17 Commits

20 changed files with 171 additions and 82 deletions

View File

@@ -13,3 +13,6 @@ helm-charts
.editorconfig .editorconfig
.idea .idea
coverage* coverage*
output*
dist
data

10
.env.sample Normal file
View File

@@ -0,0 +1,10 @@
ntfy_on=true
ntfy_username=chiko
ntfy_password=Blub
ntfy_host=ntfy.some-service.com
ntfy_topic=SomeTopic
dc_on=true
dc_webhook=123123123123123/ABCDEFABCDEFABCDEFABCDEFABCDEFABCDEFABCDEF
dc_botname=Botname Here
dc_avatar_url=https://pico.eix-1.n8x.sx/-tbcjZCuAtj

2
Crontab Normal file
View File

@@ -0,0 +1,2 @@
1 * * * * bun run ./src/app.ts --today > /dev/null 2>&1
0 * * * * bun run ./src/app.ts > /dev/null 2>&1

View File

@@ -1,14 +1,12 @@
FROM debian:12 AS base FROM debian:12 AS base
WORKDIR /usr/src/app WORKDIR /opt/app
RUN apt-get update && \ RUN apt-get update && \
apt-get install -y curl unzip ca-certificates python3 python3-pip && \ # apt-get install -y curl unzip cron ca-certificates python3 python3-pip && \
apt-get install -y curl unzip cron ca-certificates python3 python3-pip && \
rm -rf /var/lib/apt/lists/* rm -rf /var/lib/apt/lists/*
# install BunJs # install BunJs
RUN curl -fsSL https://bun.com/install | bash RUN curl -fsSL https://bun.com/install | bash
ENV PATH="/root/.bun/bin:$PATH" ENV PATH="/root/.bun/bin:$PATH"
# symlink python
RUN ln -s /usr/bin/python3 /usr/bin/python
# install dependencies into temp directory # install dependencies into temp directory
# this will cache them and speed up future builds # this will cache them and speed up future builds
FROM base AS install FROM base AS install
@@ -20,24 +18,34 @@ RUN cd /temp/dev && bun install --frozen-lockfile
RUN mkdir -p /temp/prod RUN mkdir -p /temp/prod
COPY package.json bun.lock /temp/prod/ COPY package.json bun.lock /temp/prod/
RUN cd /temp/prod && bun install --frozen-lockfile --production RUN cd /temp/prod && bun install --frozen-lockfile --production
# prepare python packages
RUN pip3 install -r requirements.txt # and install python dependencies
# COPY ./requirements.txt .
# RUN python3 -m pip install --break-system-packages -r requirements.txt
# RUN python3 -m pip install -U python-dotenv
# copy node_modules from temp directory # copy node_modules from temp directory
# then copy all (non-ignored) project files into the image # then copy all (non-ignored) project files into the image
FROM base AS prerelease FROM base AS prerelease
COPY --from=install /temp/dev/node_modules node_modules COPY --from=install /temp/dev/node_modules node_modules
COPY . . COPY . ./
# [optional] tests & build # [optional] tests & build
ENV NODE_ENV=production ENV NODE_ENV=production
# copy production dependencies and source code into final image # copy production dependencies and source code into final image
FROM base AS release FROM base AS release
WORKDIR /opt/app
COPY --from=install /temp/prod/node_modules node_modules COPY --from=install /temp/prod/node_modules node_modules
COPY --from=prerelease /usr/src/app/index.ts . COPY --from=prerelease /opt/app/src/app.ts .
COPY --from=prerelease /usr/src/app/package.json . COPY --from=prerelease /opt/app/package.json .
#COPY --from=prerelease .entrypoint.sh .
# run the app COPY Crontab /etc/cron.d/
USER bun RUN chmod 0644 /etc/cron.d/Crontab
ENTRYPOINT ["./entrypoint.sh"] COPY . ./
# USER bun
RUN touch /var/log/cron.log
# RUN chmod +x entrypoint.sh
# ENTRYPOINT ["./entrypoint.sh"]
VOLUME /opt/app/data/db
CMD bun run ./src/app.ts --today && cron && tail -f /var/log/cron.log

View File

@@ -1,37 +0,0 @@
from dotenv import load_dotenv
import os
load_dotenv() # Load environment variables from .env file
ntfy_username = os.getenv('ntfy_username')
ntfy_password = os.getenv('ntfy_password')
ntfy_host = os.getenv('ntfy_host')
ntfy_topic = os.getenv('ntfy_topic')
dc_webhook = os.getenv('dc_webhook')
dc_botname = os.getenv('dc_botname')
dc_avatar_url = os.getenv('dc_avatar_url')
from argparse import ArgumentParser
import apprise
parser = ArgumentParser()
parser.add_argument("--title")
parser.add_argument("--body")
parser.add_argument("--click")
args = parser.parse_args()
print(args)
apobj = apprise.Apprise()
# config = apprise.AppriseConfig()
# config.add('https://myserver:8080/path/to/config')
if ntfy_host and ntfy_topic:
ntfy_link = f"ntfys://{ntfy_username}:{ntfy_password}@{ntfy_host}/{ntfy_topic}"
if args.click:
ntfy_link = ntfy_link + "?click=" + args.click
apobj.add(ntfy_link)
if dc_webhook:
apobj.add(f"discord://{dc_webhook}?avatar_url={dc_avatar_url}&botname={dc_botname}");
apobj.notify(
body=args.body,
title=args.title
)

View File

@@ -1,14 +0,0 @@
import * as Bun from "bun";
export function sendNotification(title: string, body: string, click?: string | null) {
const command = [
"python",
"./app/notification.py",
`--title=${title}`,
`--body=${body}`,
];
if (click) {
command.push(`--click=${click}`);
}
Bun.spawn(command);
}

36
docker-compose.yml Normal file
View File

@@ -0,0 +1,36 @@
services:
app:
build: .
volumes:
- ./data/db:/opt/app/data/db
env_file:
- path: ./.env
required: true
depends_on:
apprise:
condition: service_healthy
links:
- apprise
apprise:
image: caronc/apprise:latest
hostname: apprise
environment:
- APPRISE_WORKER_COUNT=1
- APPRISE_STATEFUL_MODE=simple
# - PUID=$(id -u)
# - PGID=$(id -g)
volumes:
- ./data/apprise/config:/config
- ./data/apprise/plugin:/plugin
- ./data/apprise/attach:/attach
ports:
- 8000:8000
healthcheck:
test: ["CMD", "curl", "-f", "http://localhost:8000/status"]
interval: 30s
timeout: 10s
retries: 5
# networks:
# default:
# external: true
# name: npm

View File

@@ -1,6 +1,6 @@
#!/bin/bash #!/bin/bash
crontab -l > mycron crontab -l > mycron
echo "0 8 * * * bun run ./app/app.ts --today > /dev/null 2>&1" >> mycron echo "0 8 * * * bun run ./src/app.ts --today > /dev/null 2>&1" >> mycron
echo "0 * * * * bun run ./app/app.ts --all > /dev/null 2>&1" >> mycron echo "0 * * * * bun run ./src/app.ts > /dev/null 2>&1" >> mycron
crontab mycron crontab mycron
rm mycron rm mycron

View File

@@ -1,17 +1,20 @@
{ {
"version": "0.1.0",
"name": "eventcalender", "name": "eventcalender",
"module": "index.ts", "module": "./src/app.ts",
"type": "module", "type": "module",
"private": true, "private": true,
"devDependencies": { "devDependencies": {
"@types/bun": "^1.3.0" "@types/bun": "^1.3.0"
}, },
"scripts": { "scripts": {
"dev": "bun run ./app/app.ts", "dev": "bun run ./src/app.ts",
"dev:init": "bun run ./app/app.ts --init", "dev:init": "bun run ./src/app.ts --init",
"db:init": "bun run ./run/db_init.ts", "db:init": "bun run ./run/db_init.ts",
"db:deleteall": "bun run ./run/db_deleteall.ts", "db:deleteall": "bun run ./run/db_deleteall.ts",
"build": "bun build ./app/app.ts --compile --outfile ./build/77th_event_calendar_notification" "build": "bun build --compile --minify --sourcemap ./src/app.ts --outfile ./build/77th_event_calendar_notification",
"build:linux": "bun build --compile --minify --sourcemap --target=bun-linux-arm64 ./src/app.ts --outfile ./build/77th_event_calendar_notification",
"docker:build": "docker build -t chiko/77th_eventcalendarntfy:0.1.0 ."
}, },
"peerDependencies": { "peerDependencies": {
"typescript": "^5" "typescript": "^5"

View File

@@ -1 +1,2 @@
apprise apprise
python-dotenv

View File

@@ -1,4 +1,4 @@
import * as db from "../app/sql"; import * as db from "../src/sql";
const query = db.db.query("DELETE FROM events;"); const query = db.db.query("DELETE FROM events;");
query.run(); query.run();

View File

@@ -1,5 +1,5 @@
import { Event } from "../app/component/event/events"; import { Event } from "../src/component/event/events";
import * as db from "../app/sql"; import * as db from "../src/sql";
import { Database } from "bun:sqlite"; import { Database } from "bun:sqlite";
export function init ( db: Database ) { export function init ( db: Database ) {

View File

@@ -112,11 +112,29 @@ async function main( ) {
} }
return false; return false;
})( ev ); })( ev );
sendNotification( const title = `${today_prefix ? "TODAY " : ""}${notification_prefix ? notification_prefix + ": " : ""} ${ev.title} (${ TEventType[ ev.event_type ] })`;
`${today_prefix ? "TODAY " : ""}${notification_prefix ? notification_prefix + ": " : ""} ${ev.title} (${ TEventType[ ev.event_type ] })`,
`${body}` await fetch("http://apprise:8000/notify", {
// `${ev.link || "https://77th-jsoc.com/#/events"}` method: "POST",
); headers: {
"Content-Type": "application/json"
},
body: JSON.stringify({
urls: [
`ntfys://${process.env.ntfy_username}:${process.env.ntfy_password}@${process.env.ntfy_host}/${process.env.ntfy_topic}${ ev.link ? `?click=${ev.link}`: "?click=https://77th-jsoc.com/#/events" }`,
`discord://${process.env.dc_webhook}?avatar_url=${process.env.dc_avatar_url}&botname=${process.env.dc_botname}`
].join(","),
title: title,
body: body,
format: "text"
})
});
// await sendNotification(
// title,
// body
// // `${ev.link || "https://77th-jsoc.com/#/events"}`
// );
ev.set_notification("done", db); ev.set_notification("done", db);
} }
}; };

41
src/notification.py Normal file
View File

@@ -0,0 +1,41 @@
from dotenv import load_dotenv
import os
def main():
load_dotenv() # Load environment variables from .env file
ntfy_username = os.getenv('ntfy_username')
ntfy_password = os.getenv('ntfy_password')
ntfy_host = os.getenv('ntfy_host')
ntfy_topic = os.getenv('ntfy_topic')
dc_webhook = os.getenv('dc_webhook')
dc_botname = os.getenv('dc_botname')
dc_avatar_url = os.getenv('dc_avatar_url')
from argparse import ArgumentParser
import apprise
parser = ArgumentParser()
parser.add_argument("--title")
parser.add_argument("--body")
parser.add_argument("--click")
args = parser.parse_args()
print(args)
apobj = apprise.Apprise()
# config = apprise.AppriseConfig()
# config.add('https://myserver:8080/path/to/config')
if ntfy_host and ntfy_topic:
ntfy_link = f"ntfys://{ntfy_username}:{ntfy_password}@{ntfy_host}/{ntfy_topic}"
if args.click:
ntfy_link = ntfy_link + "?click=" + args.click
apobj.add(ntfy_link)
if dc_webhook:
apobj.add(f"discord://{dc_webhook}?avatar_url={dc_avatar_url}&botname={dc_botname}");
apobj.notify(
body=args.body,
title=args.title
)
if __name__ == '__main__':
main()

16
src/sendNotification.ts Normal file
View File

@@ -0,0 +1,16 @@
import * as Bun from "bun";
export async function sendNotification(title: string, body: string, click?: string | null) {
const command = [
"python3",
"./src/notification.py",
`--title=${title}`,
`--body=${body}`,
];
if ( click ) {
command.push(`--click=${click}`);
}
const proc = Bun.spawn(command);
const text = await proc.stdout.text();
console.log("sendNotification: " + text);
}

View File

@@ -11,3 +11,5 @@ export const db = new Database(db_filepath);
export function init () { export function init () {
Event.createTable(db); Event.createTable(db);
} }
init();