Compare commits
36 Commits
c51263c947
...
version/0.
| Author | SHA1 | Date | |
|---|---|---|---|
| 8e47a3bc8a | |||
| eb0c5e1580 | |||
| d5d2fa5836 | |||
| ca102190ea | |||
| 8fee748837 | |||
| c6ec442c2b | |||
| 3e5032caf3 | |||
| 2f805c0772 | |||
| 8aef42396e | |||
| 170695f9ff | |||
| c703911f85 | |||
| a37d95709f | |||
| 152c1bcba0 | |||
| 5cdfd0f2e3 | |||
| 1c6aad0f3a | |||
| 16593e0281 | |||
| eea37b3df5 | |||
| 1a7de55da8 | |||
| 2c34fece2c | |||
| 4bbda5dcf8 | |||
| a57e4efd4c | |||
| 9ec83d8b87 | |||
| 12e57a97f5 | |||
| c69eca5c08 | |||
| dc76e14c9d | |||
| 6e34f30d4a | |||
| f1bc30a64d | |||
| d22dbaf971 | |||
| c5c5d872d7 | |||
| 7b594614c6 | |||
| ae569b7739 | |||
| 608608aa56 | |||
| 04ef066158 | |||
| 8bcb2618a2 | |||
| 1433d37afa | |||
| 79b7cfae68 |
@@ -9,7 +9,7 @@ LICENSE
|
||||
.vscode
|
||||
Makefile
|
||||
helm-charts
|
||||
.env
|
||||
.env*
|
||||
.editorconfig
|
||||
.idea
|
||||
coverage*
|
||||
|
||||
1
.gitignore
vendored
1
.gitignore
vendored
@@ -23,6 +23,7 @@ report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json
|
||||
.env.test.local
|
||||
.env.production.local
|
||||
.env.local
|
||||
.env.*
|
||||
|
||||
# caches
|
||||
.eslintcache
|
||||
|
||||
20
README.md
20
README.md
@@ -1,15 +1,27 @@
|
||||
# 77th Event Calender Notifcations
|
||||
# 77th Event Calendar Notifcations
|
||||
|
||||
To install dependencies:
|
||||
|
||||
```bashe
|
||||
```bash
|
||||
bun install
|
||||
```
|
||||
|
||||
To run:
|
||||
|
||||
```bash
|
||||
bun run index.ts
|
||||
bun run ./src/app.ts
|
||||
bun run start
|
||||
bun run dev
|
||||
```
|
||||
|
||||
This project was created using `bun init` in bun v1.3.0. [Bun](https://bun.com) is a fast all-in-one JavaScript runtime.
|
||||
## Docker
|
||||
|
||||
```bash
|
||||
docker compose build
|
||||
docker compose up -d
|
||||
```
|
||||
|
||||
## Parameter
|
||||
|
||||
### --today
|
||||
fetch all Events, track all Changes (new, changed and deleted Events) and additionally Send a Notification for todays mission
|
||||
@@ -1,6 +1,6 @@
|
||||
services:
|
||||
app:
|
||||
image: chiko/77th_eventcalendarntfy:dev
|
||||
image: chiko/77th_eventcalendarntfy:v0.1.6
|
||||
build: .
|
||||
volumes:
|
||||
- ./data/db:/opt/app/data/db
|
||||
|
||||
@@ -2,4 +2,3 @@ SHELL=/bin/bash
|
||||
MAILTO=""
|
||||
0 8 * * * root . /etc/cron-env.sh && /opt/app/run-task.sh --today >> /proc/1/fd/1 2>&1
|
||||
*/15 * * * * root . /etc/cron-env.sh && /opt/app/run-task.sh >> /proc/1/fd/1 2>&1
|
||||
* * * * * root echo "cron test ran at $(date)" >> /proc/1/fd/1 2>&1
|
||||
|
||||
@@ -7,6 +7,7 @@ chmod +x /etc/cron-env.sh
|
||||
# Write the Env Vars into a file for cron. happens during runtime of the container and not build.
|
||||
# List your environment variables here
|
||||
env_vars=(
|
||||
NODE_ENV
|
||||
TZ
|
||||
DB_FILEPATH
|
||||
DB_FILENAME
|
||||
@@ -34,7 +35,7 @@ for var in "${env_vars[@]}"; do
|
||||
done
|
||||
|
||||
export PATH="/root/.bun/bin:$PATH"
|
||||
bun run /opt/app/src/app.ts --today
|
||||
bun run /opt/app/src/app.ts
|
||||
|
||||
# Start cron in foreground
|
||||
exec cron -f
|
||||
15
package.json
15
package.json
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"version": "0.1.1",
|
||||
"name": "77th_eventcalendernotification",
|
||||
"version": "0.1.6",
|
||||
"name": "77th_eventcalendarnotification",
|
||||
"module": "./src/app.ts",
|
||||
"type": "module",
|
||||
"private": true,
|
||||
@@ -16,12 +16,13 @@
|
||||
"typescript-eslint": "^8.46.2"
|
||||
},
|
||||
"scripts": {
|
||||
"dev": "bun run ./src/app.ts",
|
||||
"dev:init": "bun run ./src/app.ts --init",
|
||||
"start": "bun run ./src/app.ts",
|
||||
"dev": "NODE_ENV=development bun ./src/app.ts",
|
||||
"db:init": "bun run ./run/db_init.ts",
|
||||
"db:deleteall": "bun run ./run/db_deleteall.ts",
|
||||
"build": "bun build --compile --minify --sourcemap ./src/app.ts --outfile ./build/77th_eventcalendernotification",
|
||||
"build:linux": "bun build --compile --minify --sourcemap --target=bun-linux-arm64 ./src/app.ts --outfile ./build/77th_eventcalendernotification",
|
||||
"db:deleteall": "bun run ./run/db_event_deleteall.ts",
|
||||
"db:event:dedup": "bun run ./run/db_event_delete_duplicates.ts",
|
||||
"build": "bun build --compile --minify --sourcemap ./src/app.ts --outfile ./build/77th_eventcalendarnotification",
|
||||
"build:linux": "bun build --compile --minify --sourcemap --target=bun-linux-arm64 ./src/app.ts --outfile ./build/77th_eventcalendarnotification",
|
||||
"docker:build": "docker build -t chiko/77th_eventcalendarntfy:0.1.0 ."
|
||||
},
|
||||
"peerDependencies": {
|
||||
|
||||
@@ -18,7 +18,7 @@ log_info "Starting task with args: $*"
|
||||
|
||||
cd /opt/app
|
||||
|
||||
if bun run ./src/app.ts "$@" >> /proc/1/fd/1 2>> /proc/1/fd/2; then
|
||||
if bun run start "$@" >> /proc/1/fd/1 2>> /proc/1/fd/2; then
|
||||
log_info "Task completed successfully."
|
||||
else
|
||||
log_error "Task failed!"
|
||||
|
||||
10
run/db_event_delete_duplicates.ts
Normal file
10
run/db_event_delete_duplicates.ts
Normal file
@@ -0,0 +1,10 @@
|
||||
import * as db from "../src/sql";
|
||||
|
||||
const query = db.db.query(`DELETE FROM events
|
||||
WHERE rowid NOT IN (
|
||||
SELECT MIN(rowid)
|
||||
FROM events
|
||||
GROUP BY uid
|
||||
);`);
|
||||
|
||||
query.run();
|
||||
7
run/db_fix_events_deleted_set_done.ts
Normal file
7
run/db_fix_events_deleted_set_done.ts
Normal file
@@ -0,0 +1,7 @@
|
||||
import db from "../src/sql";
|
||||
|
||||
db.run(
|
||||
`UPDATE events
|
||||
SET notification = 'done'
|
||||
WHERE deleteDate IS NOT NULL;`
|
||||
);
|
||||
39
run/db_migration_v0.1.3.ts
Normal file
39
run/db_migration_v0.1.3.ts
Normal file
@@ -0,0 +1,39 @@
|
||||
import db from "../src/sql";
|
||||
|
||||
const run_migration = db.transaction(() => {
|
||||
// SQL 1: remove duplicates by uid
|
||||
db.run(`DELETE FROM events
|
||||
WHERE rowid NOT IN (
|
||||
SELECT MIN(rowid)
|
||||
FROM events
|
||||
GROUP BY uid
|
||||
);`);
|
||||
|
||||
// SQL 2: create new table with unique key
|
||||
db.run(`CREATE TABLE events_new (
|
||||
"event_uid" INTEGER PRIMARY KEY,
|
||||
"uid" TEXT NOT NULL UNIQUE,
|
||||
"title" TEXT NOT NULL,
|
||||
"date_at" DATETIME NOT NULL,
|
||||
"time_start" TEXT NOT NULL,
|
||||
"time_end" TEXT NOT NULL,
|
||||
"posted_by" TEXT NOT NULL,
|
||||
"location" TEXT NOT NULL,
|
||||
"event_type" TEXT NOT NULL,
|
||||
"link" TEXT NOT NULL,
|
||||
"description" TEXT NOT NULL,
|
||||
"timezone" TEXT NOT NULL,
|
||||
"notification" TEXT NOT NULL,
|
||||
"deleteDate" INTEGER NULL
|
||||
);`);
|
||||
|
||||
// SQL 3: Log the transaction
|
||||
db.run(`INSERT INTO events_new (event_uid, uid, title, date_at, time_start, time_end, posted_by, location, event_type, link, description, timezone, notification, deleteDate)
|
||||
SELECT event_uid, uid, title, date_at, time_start, time_end, posted_by, location, event_type, link, description, timezone, notification, deleteDate FROM events;
|
||||
`);
|
||||
db.run(`DROP TABLE events;
|
||||
ALTER TABLE events_new RENAME TO events;`);
|
||||
});
|
||||
|
||||
// Run the transaction
|
||||
run_migration();
|
||||
1
sql/events/events_create_unique_index_uid.sql
Normal file
1
sql/events/events_create_unique_index_uid.sql
Normal file
@@ -0,0 +1 @@
|
||||
CREATE UNIQUE INDEX idx_events_uid ON events(uid);
|
||||
6
sql/events/events_delete_duplicate_rows.sql
Normal file
6
sql/events/events_delete_duplicate_rows.sql
Normal file
@@ -0,0 +1,6 @@
|
||||
DELETE FROM events
|
||||
WHERE rowid NOT IN (
|
||||
SELECT MIN(rowid)
|
||||
FROM events
|
||||
GROUP BY uid
|
||||
);
|
||||
4
sql/events/events_find_duplicate_uid.sql
Normal file
4
sql/events/events_find_duplicate_uid.sql
Normal file
@@ -0,0 +1,4 @@
|
||||
SELECT uid, COUNT(*) AS count
|
||||
FROM events
|
||||
GROUP BY uid
|
||||
HAVING COUNT(*) > 1;
|
||||
29
sql/sql_migration_v0.1.3.sql
Normal file
29
sql/sql_migration_v0.1.3.sql
Normal file
@@ -0,0 +1,29 @@
|
||||
DELETE FROM events
|
||||
WHERE rowid NOT IN (
|
||||
SELECT MIN(rowid)
|
||||
FROM events
|
||||
GROUP BY uid
|
||||
);
|
||||
|
||||
CREATE TABLE events_new (
|
||||
"event_uid" INTEGER PRIMARY KEY,
|
||||
"uid" TEXT NOT NULL UNIQUE,
|
||||
"title" TEXT NOT NULL,
|
||||
"date_at" DATETIME NOT NULL,
|
||||
"time_start" TEXT NOT NULL,
|
||||
"time_end" TEXT NOT NULL,
|
||||
"posted_by" TEXT NOT NULL,
|
||||
"location" TEXT NOT NULL,
|
||||
"event_type" TEXT NOT NULL,
|
||||
"link" TEXT NOT NULL,
|
||||
"description" TEXT NOT NULL,
|
||||
"timezone" TEXT NOT NULL,
|
||||
"notification" TEXT NOT NULL,
|
||||
"deleteDate" INTEGER NULL
|
||||
);
|
||||
|
||||
INSERT INTO events_new (event_uid, uid, title, date_at, time_start, time_end, posted_by, location, event_type, link, description, timezone, notification, deleteDate)
|
||||
SELECT event_uid, uid, title, date_at, time_start, time_end, posted_by, location, event_type, link, description, timezone, notification, deleteDate FROM events;
|
||||
|
||||
DROP TABLE events;
|
||||
ALTER TABLE events_new RENAME TO events;
|
||||
28
src/app.ts
28
src/app.ts
@@ -19,10 +19,12 @@ async function events_update_db() {
|
||||
console.log("events_fetched.length: " + events_fetched.length );
|
||||
|
||||
const events_fetched_list_of_uids = events_fetched.map( event => { return event.uid; });
|
||||
console.dir({events_fetched_list_of_uids} );
|
||||
console.dir( {events_fetched_list_of_uids} );
|
||||
|
||||
const events_db_currentMonth = Event.get_events({month: {year: TODAY.year, month: TODAY.month}}, db);
|
||||
const events_removed: Event[] = events_db_currentMonth.filter( (ev) => {
|
||||
const events_db_currentMonth = Event.get_events({month: {year: TODAY.year, month: TODAY.month}, deleted: false}, db);
|
||||
const events_db_nextMonth = Event.get_events({month: {year: TODAY.year, month: (TODAY.month + 1)}, deleted: false}, db);
|
||||
const events_db = [... events_db_currentMonth, ... events_db_nextMonth];
|
||||
const events_removed: Event[] = events_db.filter( (ev) => {
|
||||
return ! events_fetched_list_of_uids.includes(ev.uid);
|
||||
});
|
||||
|
||||
@@ -40,10 +42,10 @@ async function events_update_db() {
|
||||
console.log("AllRelevantEvents.length: " + AllRelevantEvents.length );
|
||||
const eventsToInsert: TEventEntityNew[] = [];
|
||||
for ( const ev of events_fetched ) {
|
||||
console.log("loop ev: " + [ ev.uid, ev.title, ev.date_at ].join( ", " ) );
|
||||
console.log("loop ev " + ev.uid + " : " + [ ev.title, ev.date_at ].join( ", " ) );
|
||||
const found = AllRelevantEvents.find(event => event.uid === ev.uid);
|
||||
if ( found ) {
|
||||
console.log("loop ev found: " + [ found.uid, found.title, found.date_at ].join( ", " ) );
|
||||
console.log("loop ev " + ev.uid + " f: " + [ found.title, found.date_at ].join( ", " ) );
|
||||
if (
|
||||
found.title != ev.title ||
|
||||
found.description != ev.description ||
|
||||
@@ -56,12 +58,12 @@ async function events_update_db() {
|
||||
found.timezone != ev.timezone ||
|
||||
found.link != ev.link
|
||||
) {
|
||||
console.log("loop ev different (changed): " + [ ev.uid, ev.title, ev.date_at ].join( ", " ) );
|
||||
console.log("loop ev " + ev.uid + " c: " + [ ev.title, ev.date_at ].join( ", " ) );
|
||||
const newEventToInsert: TEventEntityNew = {... ev, notification: "changed"};
|
||||
eventsToInsert.push( newEventToInsert );
|
||||
}
|
||||
} else {
|
||||
console.log("loop ev added (new): " + [ ev.uid, ev.title, ev.date_at ].join( ", " ) );
|
||||
console.log("loop ev " + ev.uid + " n: " + [ ev.title, ev.date_at ].join( ", " ) );
|
||||
const newEventToInsert: TEventEntityNew = {... ev, notification: "new"};
|
||||
eventsToInsert.push( newEventToInsert );
|
||||
}
|
||||
@@ -90,12 +92,20 @@ async function events_check_for_notification() {
|
||||
for ( const ev of list_of_events ) {
|
||||
console.log("loop list_of_events - ev: " + [ ev.uid, ev.title, ev.date_at, "notification: " + ev.notification ].join( ", " ) );
|
||||
console.log("loop list_of_events - ev 'title': " + ev.get_title() );
|
||||
await sendNotification( ev.get_title(), ev.get_body() );
|
||||
const notificationOptions = {
|
||||
ntfy: null,
|
||||
discord: {
|
||||
avatar_url: ( process.env.dc_avatar_url as string),
|
||||
botname: ( process.env.dc_botname as string)
|
||||
}
|
||||
};
|
||||
await sendNotification( ev.get_title(), ev.get_body(), notificationOptions );
|
||||
if ( ev.notification == "removed" ) {
|
||||
ev.set_deleted( db );
|
||||
}
|
||||
} else {
|
||||
ev.set_notification("done", db);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
async function main ( ) {
|
||||
|
||||
@@ -19,7 +19,7 @@ export type TGetEventsOptions = {
|
||||
}
|
||||
export type TEventEntity = TEvent & {
|
||||
event_uid: number
|
||||
notification: "new" | "changed" | "removed" | "done"
|
||||
notification: "new" | "changed" | "removed" | "done" | "deleted"
|
||||
}
|
||||
|
||||
export type TEventEntityNew = Omit<TEventEntity, "event_uid">
|
||||
@@ -43,8 +43,8 @@ export class Event implements TEventEntity {
|
||||
|
||||
static createTable (db: Database): void {
|
||||
const query = db.query(`CREATE TABLE IF NOT EXISTS "events" (
|
||||
"event_uid" INTEGER NOT NULL,
|
||||
"uid" TEXT NOT NULL,
|
||||
"event_uid" INTEGER PRIMARY KEY,
|
||||
"uid" TEXT NOT NULL UNIQUE,
|
||||
"title" TEXT NOT NULL,
|
||||
"date_at" DATETIME NOT NULL,
|
||||
"time_start" TEXT NOT NULL,
|
||||
@@ -56,10 +56,8 @@ export class Event implements TEventEntity {
|
||||
"description" TEXT NOT NULL,
|
||||
"timezone" TEXT NOT NULL,
|
||||
"notification" TEXT NOT NULL,
|
||||
"deleteDate" INTEGER NULL,
|
||||
PRIMARY KEY ("event_uid")
|
||||
);
|
||||
CREATE UNIQUE INDEX "sqlite_autoindex_events_1" ON "events" ("uid");`);
|
||||
"deleteDate" INTEGER NULL
|
||||
);`);
|
||||
query.run();
|
||||
}
|
||||
|
||||
@@ -91,16 +89,16 @@ export class Event implements TEventEntity {
|
||||
return events;
|
||||
}
|
||||
|
||||
static get_events (options: TGetEventsOptions, db: Database ) {
|
||||
static get_events ( options: TGetEventsOptions, db: Database ) {
|
||||
const whereConditions: string[] = [];
|
||||
if ( options.notification ) {
|
||||
whereConditions.push( `notification IN ('${ options.notification.join("', '") }')` )
|
||||
}
|
||||
if (options.date) {
|
||||
whereConditions.push(`date_at = "${options.date.year}-${options.date.month}-${options.date.day}"`);
|
||||
if ( options.date ) {
|
||||
whereConditions.push(`date_at = "${options.date.year}-${pad_l2(options.date.month)}-${pad_l2(options.date.day)}"`);
|
||||
}
|
||||
if ( options.month ) {
|
||||
whereConditions.push( `strftime('%Y-%m', date_at) = '${options.month.year}-${options.month.month}'`)
|
||||
whereConditions.push( `strftime('%Y-%m', date_at) = '${options.month.year}-${pad_l2(options.month.month)}'`)
|
||||
}
|
||||
|
||||
const where = ( () => {
|
||||
@@ -116,6 +114,7 @@ export class Event implements TEventEntity {
|
||||
return null;
|
||||
})()
|
||||
const query = db.query(`SELECT * FROM events${ where ? ( " " + where ) : ""};`).as(Event);
|
||||
console.dir({ db: { action: {get_events: query} } })
|
||||
return query.all();
|
||||
}
|
||||
|
||||
@@ -135,6 +134,25 @@ export class Event implements TEventEntity {
|
||||
this.notification = notification;
|
||||
this.deleteDate = deleteDate;
|
||||
}
|
||||
toString() {
|
||||
return {
|
||||
event_uid: this.event_uid,
|
||||
uid: this.uid,
|
||||
title: this.title,
|
||||
description: this.description,
|
||||
date_at: this.date_at,
|
||||
time_start: this.time_start,
|
||||
time_end: this.time_end,
|
||||
posted_by: this.posted_by,
|
||||
location: this.location,
|
||||
event_type: this.event_type,
|
||||
timezone: this.timezone,
|
||||
link: this.link,
|
||||
notification: this.notification,
|
||||
deleteDate: this.deleteDate
|
||||
}
|
||||
}
|
||||
|
||||
syncWithDb ( db: Database ) {
|
||||
const query = db.prepare( `SELECT * FROM events WHERE event_uid = $event_uid;`).as(Event);
|
||||
const entity = query.get({$event_uid: this.event_uid });
|
||||
@@ -167,7 +185,8 @@ export class Event implements TEventEntity {
|
||||
set_deleted ( db: Database ) {
|
||||
const query = db.prepare(
|
||||
`UPDATE events
|
||||
SET deleteDate = $deleteDate
|
||||
SET notification = 'deleted',
|
||||
deleteDate = $deleteDate
|
||||
WHERE event_uid = $event_uid;`
|
||||
);
|
||||
query.get({
|
||||
@@ -201,7 +220,7 @@ export class Event implements TEventEntity {
|
||||
const body = [
|
||||
`Title: ${this.title}`,
|
||||
`Date: ${this.date_at}`,
|
||||
`Time: ${this.get_time_start()}${ TimeDiff ? ` (Optime ${TimeDiff})` : "" }`,
|
||||
`Time: ${this.get_time_start()} (OP Time${ TimeDiff != "00:00" ? ` ${TimeDiff}` : "" })`,
|
||||
`Type: ${ TEventType[ this.event_type ] }`,
|
||||
`Location: ${this.location}`,
|
||||
`By: ${this.posted_by}`,
|
||||
|
||||
16
src/config.ts
Normal file
16
src/config.ts
Normal file
@@ -0,0 +1,16 @@
|
||||
export const config = {
|
||||
apprise: {
|
||||
services: {
|
||||
ntfy: {
|
||||
url: `ntfys://${process.env.ntfy_username}:${process.env.ntfy_password}@${process.env.ntfy_host}/${process.env.ntfy_topic}`,
|
||||
defaults: {
|
||||
|
||||
}
|
||||
}
|
||||
},
|
||||
urls: [
|
||||
`ntfys://${process.env.ntfy_username}:${process.env.ntfy_password}@${process.env.ntfy_host}/${process.env.ntfy_topic}`,
|
||||
`discord://${process.env.dc_webhook}?avatar_url=${process.env.dc_avatar_url}&botname=${process.env.dc_botname}`
|
||||
]
|
||||
}
|
||||
} as const
|
||||
@@ -1,21 +1,37 @@
|
||||
export async function sendNotification(title: string, body: string, link?: string | null) {
|
||||
import { createQS } from "./util";
|
||||
|
||||
type TSendNotificationOptions = {
|
||||
ntfy: {
|
||||
link?: string;
|
||||
} | null,
|
||||
discord: {
|
||||
href?: string
|
||||
avatar_url: string,
|
||||
botname: string
|
||||
}
|
||||
}
|
||||
|
||||
export async function sendNotification( title: string, body: string, options: TSendNotificationOptions ) {
|
||||
console.dir({
|
||||
sendNotification: {
|
||||
title,
|
||||
body,
|
||||
link
|
||||
body
|
||||
}
|
||||
});
|
||||
const QS = {
|
||||
ntfy: options.ntfy ? createQS(options.ntfy) : null,
|
||||
discord: createQS(options.discord)
|
||||
}
|
||||
if ( ! ( process.env.notification_mock == "true" ) ) {
|
||||
const response = await fetch(`${ process.env.apprise_https == "true" ? "https" : "http"}://${process.env.apprise_host ? process.env.apprise_host : "apprise"}:${process.env.apprise_port ? String(process.env.apprise_port) : "80" }/notify"`, {
|
||||
const response = await fetch(`${ process.env.apprise_https == "true" ? "https" : "http"}://${process.env.apprise_host ? process.env.apprise_host : "apprise"}:${process.env.apprise_port ? String(process.env.apprise_port) : "80" }/notify`, {
|
||||
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}${ link ? `?click=${link}`: "?click=https://77th-jsoc.com/#/events" }`,
|
||||
`discord://${process.env.dc_webhook}?avatar_url=${process.env.dc_avatar_url}&botname=${process.env.dc_botname}`
|
||||
`ntfys://${process.env.ntfy_username}:${process.env.ntfy_password}@${process.env.ntfy_host}/${process.env.ntfy_topic}${ QS.ntfy ? "?" + QS.ntfy : ""}`,
|
||||
`discord://${process.env.dc_webhook}?${QS.discord}`
|
||||
].join(","),
|
||||
title: title,
|
||||
body: body,
|
||||
|
||||
@@ -8,6 +8,8 @@ console.log(db_filepath);
|
||||
|
||||
export const db = new Database(db_filepath);
|
||||
|
||||
export default db;
|
||||
|
||||
export function init () {
|
||||
Event.createTable(db);
|
||||
}
|
||||
|
||||
@@ -34,6 +34,7 @@ export function getTsNow() {
|
||||
year: now.getFullYear(),
|
||||
month: now.getMonth() + 1,
|
||||
day: now.getDate(),
|
||||
hour: now.getHours(),
|
||||
minute: now.getMinutes(),
|
||||
seconds: now.getSeconds()
|
||||
}
|
||||
@@ -89,3 +90,10 @@ export function isEuropeanDST( date: Date ) {
|
||||
// Return true if within DST period
|
||||
return date >= start && date < end;
|
||||
}
|
||||
|
||||
export function createQS (params: Record<string, string | number | boolean>): string {
|
||||
const queryString = Object.entries(params)
|
||||
.map(([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`)
|
||||
.join("&");
|
||||
return queryString;
|
||||
}
|
||||
Reference in New Issue
Block a user