Compare commits
7 Commits
release-47
...
release-55
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
dacf73a804 | ||
|
|
5812399e29 | ||
|
|
6a6908518d | ||
|
|
d065910ff3 | ||
|
|
490433b002 | ||
|
|
e17bc8a521 | ||
|
|
ec1215f2ae |
53
.clinerules
53
.clinerules
@@ -1,53 +0,0 @@
|
|||||||
# Cursor Rules
|
|
||||||
|
|
||||||
- When merging tailwind classes, use the `cn` function.
|
|
||||||
- When using Tailwind and you need to merge classes use the `cn` function if avilable.
|
|
||||||
- We use Tailwind 4 (the latest version), make sure to not use outdated classes.
|
|
||||||
- Instead of using the syntax`Array<T>`, use `T[]`.
|
|
||||||
- Use TypeScript `type` over `interface`.
|
|
||||||
- You are forbiddent o add comments unless explicitly stated by the user.
|
|
||||||
- Avoid sending JavaScript to the client. The JS send should be optional.
|
|
||||||
- In prisma preffer `select` over `include` when making queries.
|
|
||||||
- Import the types from prisma instead of hardcoding duplicates.
|
|
||||||
- Avoid duplicating similar html code, and parametrize it when possible or create separate components.
|
|
||||||
- Remember to check the prisma schema when doing things related to the database.
|
|
||||||
- Avoid hardcoding enums from the database, import them from prisma.
|
|
||||||
- Avoid using client-side JavaScript as much as possible. And if it has to be done, make it optional.
|
|
||||||
- The admin pages can use client-side JavaScript.
|
|
||||||
- Keep README.md in sync with new capabilities.
|
|
||||||
- The package manager is npm.
|
|
||||||
- For icons use the `Icon` component from `astro-icon/components`.
|
|
||||||
- For icons use the Remix Icon library preferably.
|
|
||||||
- Use the `Image` component from `astro:assets` for images.
|
|
||||||
- Use the `zod` library for schema validation.
|
|
||||||
- In the astro actions return, don't return success: true, or similar, just return an object with the newly created/edited objects or nothing.
|
|
||||||
- When adding actions, don't create and export a new variable called actions. Notice that Astro already provides that variable from `import { actions } from 'astro:actions'`. So just add the new actions to the `server` variable in `web/src/actions/index.ts` and that's it.
|
|
||||||
- Don't forget that the astro files have thre dashes (`---`) at the begining of the file and where the server js ends. I noticed that sometimes you forget them.
|
|
||||||
- The admin actions go into a separate folder.
|
|
||||||
- In Actro actions when throwing errors use ActionError.
|
|
||||||
- @deprecated Don't import this object, use {@link actions} instead, like: `import { actions } from 'astro:actions'`. Example:
|
|
||||||
|
|
||||||
```ts
|
|
||||||
import { actions } from "astro:actions"; /* CORRECT */
|
|
||||||
import { server } from "~/actions"; /* WRONG!!!! DON'T DO THIS */
|
|
||||||
import { adminAttributeActions } from "~/actions/admin/attribute.ts"; /* WRONG!!!! DON'T DO THIS */
|
|
||||||
|
|
||||||
const result = Astro.getActionResult(actions.admin.attribute.create);
|
|
||||||
```
|
|
||||||
|
|
||||||
- Always use Astro actions instead of with API routes or `if (Astro.request.method === "POST")`.
|
|
||||||
- When adding clientside js do it with HTMX.
|
|
||||||
- When adding HTMX, the layout component BaseLayout accepts a prop htmx to load it in that page. No need to use a cdn.
|
|
||||||
- When redirecting to login use the `makeLoginUrl` function from web/src/lib/redirectUrls.ts
|
|
||||||
|
|
||||||
```ts
|
|
||||||
function makeLoginUrl(
|
|
||||||
currentUrl: URL,
|
|
||||||
options: {
|
|
||||||
redirect?: URL | string | null;
|
|
||||||
error?: string | null;
|
|
||||||
logout?: boolean;
|
|
||||||
message?: string | null;
|
|
||||||
} = {}
|
|
||||||
);
|
|
||||||
```
|
|
||||||
36
justfile
36
justfile
@@ -51,42 +51,6 @@ import-db file="":
|
|||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
echo "Restoring database from $BACKUP_FILE..."
|
|
||||||
# First drop all connections to the database
|
|
||||||
docker compose exec -T database psql -U ${POSTGRES_USER:-kycnot} -c "SELECT pg_terminate_backend(pg_stat_activity.pid) FROM pg_stat_activity WHERE pg_stat_activity.datname = '${POSTGRES_DATABASE:-kycnot}' AND pid <> pg_backend_pid();" postgres
|
|
||||||
|
|
||||||
# Drop and recreate database
|
|
||||||
echo "Dropping and recreating the database..."
|
|
||||||
docker compose exec -T database psql -U ${POSTGRES_USER:-kycnot} -c "DROP DATABASE IF EXISTS ${POSTGRES_DATABASE:-kycnot};" postgres
|
|
||||||
docker compose exec -T database psql -U ${POSTGRES_USER:-kycnot} -c "CREATE DATABASE ${POSTGRES_DATABASE:-kycnot};" postgres
|
|
||||||
|
|
||||||
# Restore the database
|
|
||||||
cat "$BACKUP_FILE" | docker compose exec -T database pg_restore -U ${POSTGRES_USER:-kycnot} -d ${POSTGRES_DATABASE:-kycnot} --no-owner
|
|
||||||
echo "Database restored successfully!"
|
|
||||||
|
|
||||||
# Import triggers
|
|
||||||
echo "Importing triggers..."
|
|
||||||
just import-triggers
|
|
||||||
|
|
||||||
echo "Database import completed!"
|
|
||||||
# Check if migrations need to be run
|
|
||||||
cd web && npx prisma migrate status
|
|
||||||
|
|
||||||
#!/bin/bash
|
|
||||||
if [ -z "{{file}}" ]; then
|
|
||||||
BACKUP_FILE=$(find backups/ -name 'db_backup_*.dump' | sort -r | head -n 1)
|
|
||||||
if [ -z "$BACKUP_FILE" ]; then
|
|
||||||
echo "Error: No backup files found in the backups directory"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
BACKUP_FILE="{{file}}"
|
|
||||||
if [ ! -f "$BACKUP_FILE" ]; then
|
|
||||||
echo "Error: Backup file '$BACKUP_FILE' not found"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "=== STEP 1: PREPARING DATABASE ==="
|
echo "=== STEP 1: PREPARING DATABASE ==="
|
||||||
# Drop all connections to the database
|
# Drop all connections to the database
|
||||||
docker compose exec -T database psql -U ${POSTGRES_USER:-kycnot} -c "SELECT pg_terminate_backend(pg_stat_activity.pid) FROM pg_stat_activity WHERE pg_stat_activity.datname = '${POSTGRES_DATABASE:-kycnot}' AND pid <> pg_backend_pid();" postgres
|
docker compose exec -T database psql -U ${POSTGRES_USER:-kycnot} -c "SELECT pg_terminate_backend(pg_stat_activity.pid) FROM pg_stat_activity WHERE pg_stat_activity.datname = '${POSTGRES_DATABASE:-kycnot}' AND pid <> pg_backend_pid();" postgres
|
||||||
|
|||||||
@@ -2,8 +2,12 @@ DATABASE_URL="postgresql://kycnot:kycnot@localhost:3399/kycnot?schema=public"
|
|||||||
REDIS_URL="redis://localhost:6379"
|
REDIS_URL="redis://localhost:6379"
|
||||||
SOURCE_CODE_URL="https://github.com"
|
SOURCE_CODE_URL="https://github.com"
|
||||||
DATABASE_UI_URL="http://localhost:5555"
|
DATABASE_UI_URL="http://localhost:5555"
|
||||||
SITE_URL="https://localhost:4321"
|
SITE_URL="http://localhost:4321"
|
||||||
ONION_ADDRESS="http://kycnotmezdiftahfmc34pqbpicxlnx3jbf5p7jypge7gdvduu7i6qjqd.onion"
|
ONION_ADDRESS="http://kycnotmezdiftahfmc34pqbpicxlnx3jbf5p7jypge7gdvduu7i6qjqd.onion"
|
||||||
I2P_ADDRESS="http://nti3rj4j4disjcm2kvp4eno7otcejbbxv3ggxwr5tpfk4jucah7q.b32.i2p"
|
I2P_ADDRESS="http://nti3rj4j4disjcm2kvp4eno7otcejbbxv3ggxwr5tpfk4jucah7q.b32.i2p"
|
||||||
RELEASE_NUMBER=123
|
RELEASE_NUMBER=123
|
||||||
RELEASE_DATE="2025-05-23T19:00:00.000Z"
|
RELEASE_DATE="2025-05-23T19:00:00.000Z"
|
||||||
|
# Generated with `npx web-push generate-vapid-keys`
|
||||||
|
VAPID_PUBLIC_KEY="BPmJbRXzG9zT181vyg1GlpyV8qu7rjVjfg6vkkOgtqeTZECyt6lR4MuzmlarEHSBF6gPpc77ZA0_tTVtmYh65iM"
|
||||||
|
VAPID_PRIVATE_KEY="eN_S2SMXDB2hpwVXbgDkDrPIPMqirllZaJcUgYTt9w0"
|
||||||
|
VAPID_SUBJECT="mailto:no-reply@kycnot.me"
|
||||||
|
|||||||
@@ -6,11 +6,11 @@ import sitemap from '@astrojs/sitemap'
|
|||||||
import tailwindcss from '@tailwindcss/vite'
|
import tailwindcss from '@tailwindcss/vite'
|
||||||
import { defineConfig, envField } from 'astro/config'
|
import { defineConfig, envField } from 'astro/config'
|
||||||
import icon from 'astro-icon'
|
import icon from 'astro-icon'
|
||||||
import { loadEnv } from 'vite'
|
|
||||||
|
|
||||||
// @ts-expect-error process.env actually exists
|
import { postgresListener } from './src/lib/postgresListenerIntegration'
|
||||||
const { SITE_URL } = loadEnv(process.env.NODE_ENV, process.cwd(), '')
|
import { getServerEnvVariable } from './src/lib/serverEnvVariables'
|
||||||
if (!SITE_URL) throw new Error('SITE_URL environment variable is not set')
|
|
||||||
|
const SITE_URL = getServerEnvVariable('SITE_URL')
|
||||||
|
|
||||||
export default defineConfig({
|
export default defineConfig({
|
||||||
site: SITE_URL,
|
site: SITE_URL,
|
||||||
@@ -22,6 +22,7 @@ export default defineConfig({
|
|||||||
plugins: [tailwindcss()],
|
plugins: [tailwindcss()],
|
||||||
},
|
},
|
||||||
integrations: [
|
integrations: [
|
||||||
|
postgresListener(),
|
||||||
icon(),
|
icon(),
|
||||||
mdx(),
|
mdx(),
|
||||||
sitemap({
|
sitemap({
|
||||||
@@ -195,6 +196,27 @@ export default defineConfig({
|
|||||||
access: 'public',
|
access: 'public',
|
||||||
optional: true,
|
optional: true,
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
// Generated with `npx web-push generate-vapid-keys`
|
||||||
|
VAPID_PUBLIC_KEY: envField.string({
|
||||||
|
context: 'server',
|
||||||
|
access: 'public',
|
||||||
|
min: 1,
|
||||||
|
optional: false,
|
||||||
|
}),
|
||||||
|
// Generated with `npx web-push generate-vapid-keys`
|
||||||
|
VAPID_PRIVATE_KEY: envField.string({
|
||||||
|
context: 'server',
|
||||||
|
access: 'secret',
|
||||||
|
min: 1,
|
||||||
|
optional: false,
|
||||||
|
}),
|
||||||
|
VAPID_SUBJECT: envField.string({
|
||||||
|
context: 'server',
|
||||||
|
access: 'secret',
|
||||||
|
min: 1,
|
||||||
|
optional: false,
|
||||||
|
}),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|||||||
224
web/package-lock.json
generated
224
web/package-lock.json
generated
@@ -19,6 +19,7 @@
|
|||||||
"@prisma/client": "6.8.2",
|
"@prisma/client": "6.8.2",
|
||||||
"@tailwindcss/vite": "4.1.7",
|
"@tailwindcss/vite": "4.1.7",
|
||||||
"@types/mime-types": "2.1.4",
|
"@types/mime-types": "2.1.4",
|
||||||
|
"@types/pg": "8.15.4",
|
||||||
"@vercel/og": "0.6.8",
|
"@vercel/og": "0.6.8",
|
||||||
"astro": "5.7.13",
|
"astro": "5.7.13",
|
||||||
"astro-loading-indicator": "0.7.0",
|
"astro-loading-indicator": "0.7.0",
|
||||||
@@ -32,6 +33,7 @@
|
|||||||
"lodash-es": "4.17.21",
|
"lodash-es": "4.17.21",
|
||||||
"mime-types": "3.0.1",
|
"mime-types": "3.0.1",
|
||||||
"object-to-formdata": "4.5.1",
|
"object-to-formdata": "4.5.1",
|
||||||
|
"pg": "8.16.0",
|
||||||
"qrcode": "1.5.4",
|
"qrcode": "1.5.4",
|
||||||
"react": "19.1.0",
|
"react": "19.1.0",
|
||||||
"redis": "5.0.1",
|
"redis": "5.0.1",
|
||||||
@@ -44,6 +46,7 @@
|
|||||||
"tailwindcss": "4.1.7",
|
"tailwindcss": "4.1.7",
|
||||||
"typescript": "5.8.3",
|
"typescript": "5.8.3",
|
||||||
"unique-username-generator": "1.4.0",
|
"unique-username-generator": "1.4.0",
|
||||||
|
"web-push": "3.6.7",
|
||||||
"zod-form-data": "2.0.7"
|
"zod-form-data": "2.0.7"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
@@ -60,6 +63,7 @@
|
|||||||
"@types/qrcode": "1.5.5",
|
"@types/qrcode": "1.5.5",
|
||||||
"@types/react": "19.1.4",
|
"@types/react": "19.1.4",
|
||||||
"@types/seedrandom": "3.0.8",
|
"@types/seedrandom": "3.0.8",
|
||||||
|
"@types/web-push": "3.6.4",
|
||||||
"@typescript-eslint/parser": "8.32.1",
|
"@typescript-eslint/parser": "8.32.1",
|
||||||
"astro-icon": "1.1.5",
|
"astro-icon": "1.1.5",
|
||||||
"date-fns": "4.1.0",
|
"date-fns": "4.1.0",
|
||||||
@@ -3150,12 +3154,10 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@types/pg": {
|
"node_modules/@types/pg": {
|
||||||
"version": "8.6.1",
|
"version": "8.15.4",
|
||||||
"resolved": "https://registry.npmjs.org/@types/pg/-/pg-8.6.1.tgz",
|
"resolved": "https://registry.npmjs.org/@types/pg/-/pg-8.15.4.tgz",
|
||||||
"integrity": "sha512-1Kc4oAGzAl7uqUStZCDvaLFqZrW9qWSjXOmBfdgyBP5La7Us6Mg4GBvRlSoaZMhQF/zSj1C8CtKMBkoiT8eL8w==",
|
"integrity": "sha512-I6UNVBAoYbvuWkkU3oosC8yxqH21f4/Jc4DK71JLG3dT2mdlGe1z+ep/LQGXaKaOgcvUrsQoPRqfgtMcvZiJhg==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"optional": true,
|
|
||||||
"peer": true,
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@types/node": "*",
|
"@types/node": "*",
|
||||||
"pg-protocol": "*",
|
"pg-protocol": "*",
|
||||||
@@ -3215,6 +3217,16 @@
|
|||||||
"integrity": "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q==",
|
"integrity": "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/@types/web-push": {
|
||||||
|
"version": "3.6.4",
|
||||||
|
"resolved": "https://registry.npmjs.org/@types/web-push/-/web-push-3.6.4.tgz",
|
||||||
|
"integrity": "sha512-GnJmSr40H3RAnj0s34FNTcJi1hmWFV5KXugE0mYWnYhgTAHLJ/dJKAwDmvPJYMke0RplY2XE9LnM4hqSqKIjhQ==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@types/node": "*"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@types/ws": {
|
"node_modules/@types/ws": {
|
||||||
"version": "8.5.13",
|
"version": "8.5.13",
|
||||||
"resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.13.tgz",
|
"resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.13.tgz",
|
||||||
@@ -3797,6 +3809,15 @@
|
|||||||
"acorn": "^6.0.0 || ^7.0.0 || ^8.0.0"
|
"acorn": "^6.0.0 || ^7.0.0 || ^8.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/agent-base": {
|
||||||
|
"version": "7.1.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.3.tgz",
|
||||||
|
"integrity": "sha512-jRR5wdylq8CkOe6hei19GGZnxM6rBGwFl3Bg0YItGDimvjGtAvdZk4Pu6Cl4u4Igsws4a1fd1Vq3ezrhn4KmFw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 14"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/ajv": {
|
"node_modules/ajv": {
|
||||||
"version": "6.12.6",
|
"version": "6.12.6",
|
||||||
"resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
|
"resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
|
||||||
@@ -4063,6 +4084,18 @@
|
|||||||
"url": "https://github.com/sponsors/ljharb"
|
"url": "https://github.com/sponsors/ljharb"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/asn1.js": {
|
||||||
|
"version": "5.4.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/asn1.js/-/asn1.js-5.4.1.tgz",
|
||||||
|
"integrity": "sha512-+I//4cYPccV8LdmBLiX8CYvf9Sp3vQsrqu2QNXRcrbiWvcx/UdlFiqUJJzxRQxgsZmvhXhn4cSKeSmoFjVdupA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"bn.js": "^4.0.0",
|
||||||
|
"inherits": "^2.0.1",
|
||||||
|
"minimalistic-assert": "^1.0.0",
|
||||||
|
"safer-buffer": "^2.1.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/ast-types-flow": {
|
"node_modules/ast-types-flow": {
|
||||||
"version": "0.0.8",
|
"version": "0.0.8",
|
||||||
"resolved": "https://registry.npmjs.org/ast-types-flow/-/ast-types-flow-0.0.8.tgz",
|
"resolved": "https://registry.npmjs.org/ast-types-flow/-/ast-types-flow-0.0.8.tgz",
|
||||||
@@ -4822,6 +4855,12 @@
|
|||||||
],
|
],
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/bn.js": {
|
||||||
|
"version": "4.12.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.2.tgz",
|
||||||
|
"integrity": "sha512-n4DSx829VRTRByMRGdjQ9iqsN0Bh4OolPsFnaZBLcbi8iXcB+kJ9s7EnRt4wILZNV3kPLHkRVfOc/HvhC3ovDw==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/boolbase": {
|
"node_modules/boolbase": {
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz",
|
||||||
@@ -4916,6 +4955,12 @@
|
|||||||
"node": "*"
|
"node": "*"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/buffer-equal-constant-time": {
|
||||||
|
"version": "1.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz",
|
||||||
|
"integrity": "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==",
|
||||||
|
"license": "BSD-3-Clause"
|
||||||
|
},
|
||||||
"node_modules/bundle-name": {
|
"node_modules/bundle-name": {
|
||||||
"version": "4.1.0",
|
"version": "4.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/bundle-name/-/bundle-name-4.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/bundle-name/-/bundle-name-4.1.0.tgz",
|
||||||
@@ -6157,6 +6202,15 @@
|
|||||||
"node": ">= 0.4"
|
"node": ">= 0.4"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/ecdsa-sig-formatter": {
|
||||||
|
"version": "1.0.11",
|
||||||
|
"resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz",
|
||||||
|
"integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==",
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"dependencies": {
|
||||||
|
"safe-buffer": "^5.0.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/ee-first": {
|
"node_modules/ee-first": {
|
||||||
"version": "1.1.1",
|
"version": "1.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
|
||||||
@@ -8181,6 +8235,15 @@
|
|||||||
"integrity": "sha512-VZAohXyF7xPGS52IM8d1T1283y+X4D+Owf3qY1NZ9RuBypyu9l8cGsxUMAG5fEAb/DhT7rDoJ9Hpu5/HxFD3cw==",
|
"integrity": "sha512-VZAohXyF7xPGS52IM8d1T1283y+X4D+Owf3qY1NZ9RuBypyu9l8cGsxUMAG5fEAb/DhT7rDoJ9Hpu5/HxFD3cw==",
|
||||||
"license": "0BSD"
|
"license": "0BSD"
|
||||||
},
|
},
|
||||||
|
"node_modules/http_ece": {
|
||||||
|
"version": "1.2.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/http_ece/-/http_ece-1.2.0.tgz",
|
||||||
|
"integrity": "sha512-JrF8SSLVmcvc5NducxgyOrKXe3EsyHMgBFgSaIUGmArKe+rwr0uphRkRXvwiom3I+fpIfoItveHrfudL8/rxuA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=16"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/http-cache-semantics": {
|
"node_modules/http-cache-semantics": {
|
||||||
"version": "4.1.1",
|
"version": "4.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.1.tgz",
|
||||||
@@ -8203,6 +8266,19 @@
|
|||||||
"node": ">= 0.8"
|
"node": ">= 0.8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/https-proxy-agent": {
|
||||||
|
"version": "7.0.6",
|
||||||
|
"resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-7.0.6.tgz",
|
||||||
|
"integrity": "sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"agent-base": "^7.1.2",
|
||||||
|
"debug": "4"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 14"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/iconv-lite": {
|
"node_modules/iconv-lite": {
|
||||||
"version": "0.6.3",
|
"version": "0.6.3",
|
||||||
"resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz",
|
"resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz",
|
||||||
@@ -8935,6 +9011,27 @@
|
|||||||
"node": ">=4.0"
|
"node": ">=4.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/jwa": {
|
||||||
|
"version": "2.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/jwa/-/jwa-2.0.1.tgz",
|
||||||
|
"integrity": "sha512-hRF04fqJIP8Abbkq5NKGN0Bbr3JxlQ+qhZufXVr0DvujKy93ZCbXZMHDL4EOtodSbCWxOqR8MS1tXA5hwqCXDg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"buffer-equal-constant-time": "^1.0.1",
|
||||||
|
"ecdsa-sig-formatter": "1.0.11",
|
||||||
|
"safe-buffer": "^5.0.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/jws": {
|
||||||
|
"version": "4.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/jws/-/jws-4.0.0.tgz",
|
||||||
|
"integrity": "sha512-KDncfTmOZoOMTFG4mBlG0qUIOlc03fmzH+ru6RgYVZhPkyiy/92Owlt/8UEN+a4TXR1FQetfIpJE8ApdvdVxTg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"jwa": "^2.0.0",
|
||||||
|
"safe-buffer": "^5.0.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/keyv": {
|
"node_modules/keyv": {
|
||||||
"version": "4.5.4",
|
"version": "4.5.4",
|
||||||
"resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz",
|
"resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz",
|
||||||
@@ -10556,6 +10653,12 @@
|
|||||||
"mini-svg-data-uri": "cli.js"
|
"mini-svg-data-uri": "cli.js"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/minimalistic-assert": {
|
||||||
|
"version": "1.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz",
|
||||||
|
"integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==",
|
||||||
|
"license": "ISC"
|
||||||
|
},
|
||||||
"node_modules/minimatch": {
|
"node_modules/minimatch": {
|
||||||
"version": "9.0.5",
|
"version": "9.0.5",
|
||||||
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz",
|
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz",
|
||||||
@@ -11310,32 +11413,75 @@
|
|||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/pg": {
|
||||||
|
"version": "8.16.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/pg/-/pg-8.16.0.tgz",
|
||||||
|
"integrity": "sha512-7SKfdvP8CTNXjMUzfcVTaI+TDzBEeaUnVwiVGZQD1Hh33Kpev7liQba9uLd4CfN8r9mCVsD0JIpq03+Unpz+kg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"pg-connection-string": "^2.9.0",
|
||||||
|
"pg-pool": "^3.10.0",
|
||||||
|
"pg-protocol": "^1.10.0",
|
||||||
|
"pg-types": "2.2.0",
|
||||||
|
"pgpass": "1.0.5"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 8.0.0"
|
||||||
|
},
|
||||||
|
"optionalDependencies": {
|
||||||
|
"pg-cloudflare": "^1.2.5"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"pg-native": ">=3.0.1"
|
||||||
|
},
|
||||||
|
"peerDependenciesMeta": {
|
||||||
|
"pg-native": {
|
||||||
|
"optional": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/pg-cloudflare": {
|
||||||
|
"version": "1.2.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/pg-cloudflare/-/pg-cloudflare-1.2.5.tgz",
|
||||||
|
"integrity": "sha512-OOX22Vt0vOSRrdoUPKJ8Wi2OpE/o/h9T8X1s4qSkCedbNah9ei2W2765be8iMVxQUsvgT7zIAT2eIa9fs5+vtg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true
|
||||||
|
},
|
||||||
|
"node_modules/pg-connection-string": {
|
||||||
|
"version": "2.9.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/pg-connection-string/-/pg-connection-string-2.9.0.tgz",
|
||||||
|
"integrity": "sha512-P2DEBKuvh5RClafLngkAuGe9OUlFV7ebu8w1kmaaOgPcpJd1RIFh7otETfI6hAR8YupOLFTY7nuvvIn7PLciUQ==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/pg-int8": {
|
"node_modules/pg-int8": {
|
||||||
"version": "1.0.1",
|
"version": "1.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/pg-int8/-/pg-int8-1.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/pg-int8/-/pg-int8-1.0.1.tgz",
|
||||||
"integrity": "sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==",
|
"integrity": "sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==",
|
||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"optional": true,
|
|
||||||
"peer": true,
|
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=4.0.0"
|
"node": ">=4.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/pg-protocol": {
|
"node_modules/pg-pool": {
|
||||||
"version": "1.8.0",
|
"version": "3.10.0",
|
||||||
"resolved": "https://registry.npmjs.org/pg-protocol/-/pg-protocol-1.8.0.tgz",
|
"resolved": "https://registry.npmjs.org/pg-pool/-/pg-pool-3.10.0.tgz",
|
||||||
"integrity": "sha512-jvuYlEkL03NRvOoyoRktBK7+qU5kOvlAwvmrH8sr3wbLrOdVWsRxQfz8mMy9sZFsqJ1hEWNfdWKI4SAmoL+j7g==",
|
"integrity": "sha512-DzZ26On4sQ0KmqnO34muPcmKbhrjmyiO4lCCR0VwEd7MjmiKf5NTg/6+apUEu0NF7ESa37CGzFxH513CoUmWnA==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"optional": true,
|
"peerDependencies": {
|
||||||
"peer": true
|
"pg": ">=8.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/pg-protocol": {
|
||||||
|
"version": "1.10.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/pg-protocol/-/pg-protocol-1.10.0.tgz",
|
||||||
|
"integrity": "sha512-IpdytjudNuLv8nhlHs/UrVBhU0e78J0oIS/0AVdTbWxSOkFUVdsHC/NrorO6nXsQNDTT1kzDSOMJubBQviX18Q==",
|
||||||
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/pg-types": {
|
"node_modules/pg-types": {
|
||||||
"version": "2.2.0",
|
"version": "2.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/pg-types/-/pg-types-2.2.0.tgz",
|
"resolved": "https://registry.npmjs.org/pg-types/-/pg-types-2.2.0.tgz",
|
||||||
"integrity": "sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==",
|
"integrity": "sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"optional": true,
|
|
||||||
"peer": true,
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"pg-int8": "1.0.1",
|
"pg-int8": "1.0.1",
|
||||||
"postgres-array": "~2.0.0",
|
"postgres-array": "~2.0.0",
|
||||||
@@ -11347,6 +11493,15 @@
|
|||||||
"node": ">=4"
|
"node": ">=4"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/pgpass": {
|
||||||
|
"version": "1.0.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/pgpass/-/pgpass-1.0.5.tgz",
|
||||||
|
"integrity": "sha512-FdW9r/jQZhSeohs1Z3sI1yxFQNFvMcnmfuj4WBMUTxOrAyLMaTcE1aAMBiTlbMNaXvBCQuVi0R7hd8udDSP7ug==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"split2": "^4.1.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/picocolors": {
|
"node_modules/picocolors": {
|
||||||
"version": "1.1.1",
|
"version": "1.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
|
||||||
@@ -11456,8 +11611,6 @@
|
|||||||
"resolved": "https://registry.npmjs.org/postgres-array/-/postgres-array-2.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/postgres-array/-/postgres-array-2.0.0.tgz",
|
||||||
"integrity": "sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==",
|
"integrity": "sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"optional": true,
|
|
||||||
"peer": true,
|
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=4"
|
"node": ">=4"
|
||||||
}
|
}
|
||||||
@@ -11467,8 +11620,6 @@
|
|||||||
"resolved": "https://registry.npmjs.org/postgres-bytea/-/postgres-bytea-1.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/postgres-bytea/-/postgres-bytea-1.0.0.tgz",
|
||||||
"integrity": "sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w==",
|
"integrity": "sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"optional": true,
|
|
||||||
"peer": true,
|
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=0.10.0"
|
"node": ">=0.10.0"
|
||||||
}
|
}
|
||||||
@@ -11478,8 +11629,6 @@
|
|||||||
"resolved": "https://registry.npmjs.org/postgres-date/-/postgres-date-1.0.7.tgz",
|
"resolved": "https://registry.npmjs.org/postgres-date/-/postgres-date-1.0.7.tgz",
|
||||||
"integrity": "sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==",
|
"integrity": "sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"optional": true,
|
|
||||||
"peer": true,
|
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=0.10.0"
|
"node": ">=0.10.0"
|
||||||
}
|
}
|
||||||
@@ -11489,8 +11638,6 @@
|
|||||||
"resolved": "https://registry.npmjs.org/postgres-interval/-/postgres-interval-1.2.0.tgz",
|
"resolved": "https://registry.npmjs.org/postgres-interval/-/postgres-interval-1.2.0.tgz",
|
||||||
"integrity": "sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==",
|
"integrity": "sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"optional": true,
|
|
||||||
"peer": true,
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"xtend": "^4.0.0"
|
"xtend": "^4.0.0"
|
||||||
},
|
},
|
||||||
@@ -12696,7 +12843,6 @@
|
|||||||
"version": "2.1.2",
|
"version": "2.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
|
||||||
"integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==",
|
"integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==",
|
||||||
"dev": true,
|
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/sass-formatter": {
|
"node_modules/sass-formatter": {
|
||||||
@@ -13149,6 +13295,15 @@
|
|||||||
"url": "https://github.com/sponsors/wooorm"
|
"url": "https://github.com/sponsors/wooorm"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/split2": {
|
||||||
|
"version": "4.2.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/split2/-/split2-4.2.0.tgz",
|
||||||
|
"integrity": "sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==",
|
||||||
|
"license": "ISC",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 10.x"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/stable-hash": {
|
"node_modules/stable-hash": {
|
||||||
"version": "0.0.5",
|
"version": "0.0.5",
|
||||||
"resolved": "https://registry.npmjs.org/stable-hash/-/stable-hash-0.0.5.tgz",
|
"resolved": "https://registry.npmjs.org/stable-hash/-/stable-hash-0.0.5.tgz",
|
||||||
@@ -14675,6 +14830,25 @@
|
|||||||
"url": "https://github.com/sponsors/wooorm"
|
"url": "https://github.com/sponsors/wooorm"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/web-push": {
|
||||||
|
"version": "3.6.7",
|
||||||
|
"resolved": "https://registry.npmjs.org/web-push/-/web-push-3.6.7.tgz",
|
||||||
|
"integrity": "sha512-OpiIUe8cuGjrj3mMBFWY+e4MMIkW3SVT+7vEIjvD9kejGUypv8GPDf84JdPWskK8zMRIJ6xYGm+Kxr8YkPyA0A==",
|
||||||
|
"license": "MPL-2.0",
|
||||||
|
"dependencies": {
|
||||||
|
"asn1.js": "^5.3.0",
|
||||||
|
"http_ece": "1.2.0",
|
||||||
|
"https-proxy-agent": "^7.0.0",
|
||||||
|
"jws": "^4.0.0",
|
||||||
|
"minimist": "^1.2.5"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"web-push": "src/cli.js"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 16"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/web-streams-polyfill": {
|
"node_modules/web-streams-polyfill": {
|
||||||
"version": "3.3.3",
|
"version": "3.3.3",
|
||||||
"resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-3.3.3.tgz",
|
"resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-3.3.3.tgz",
|
||||||
@@ -14916,8 +15090,6 @@
|
|||||||
"resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz",
|
"resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz",
|
||||||
"integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==",
|
"integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"optional": true,
|
|
||||||
"peer": true,
|
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=0.4"
|
"node": ">=0.4"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -33,6 +33,7 @@
|
|||||||
"@prisma/client": "6.8.2",
|
"@prisma/client": "6.8.2",
|
||||||
"@tailwindcss/vite": "4.1.7",
|
"@tailwindcss/vite": "4.1.7",
|
||||||
"@types/mime-types": "2.1.4",
|
"@types/mime-types": "2.1.4",
|
||||||
|
"@types/pg": "8.15.4",
|
||||||
"@vercel/og": "0.6.8",
|
"@vercel/og": "0.6.8",
|
||||||
"astro": "5.7.13",
|
"astro": "5.7.13",
|
||||||
"astro-loading-indicator": "0.7.0",
|
"astro-loading-indicator": "0.7.0",
|
||||||
@@ -46,6 +47,7 @@
|
|||||||
"lodash-es": "4.17.21",
|
"lodash-es": "4.17.21",
|
||||||
"mime-types": "3.0.1",
|
"mime-types": "3.0.1",
|
||||||
"object-to-formdata": "4.5.1",
|
"object-to-formdata": "4.5.1",
|
||||||
|
"pg": "8.16.0",
|
||||||
"qrcode": "1.5.4",
|
"qrcode": "1.5.4",
|
||||||
"react": "19.1.0",
|
"react": "19.1.0",
|
||||||
"redis": "5.0.1",
|
"redis": "5.0.1",
|
||||||
@@ -58,6 +60,7 @@
|
|||||||
"tailwindcss": "4.1.7",
|
"tailwindcss": "4.1.7",
|
||||||
"typescript": "5.8.3",
|
"typescript": "5.8.3",
|
||||||
"unique-username-generator": "1.4.0",
|
"unique-username-generator": "1.4.0",
|
||||||
|
"web-push": "3.6.7",
|
||||||
"zod-form-data": "2.0.7"
|
"zod-form-data": "2.0.7"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
@@ -74,6 +77,7 @@
|
|||||||
"@types/qrcode": "1.5.5",
|
"@types/qrcode": "1.5.5",
|
||||||
"@types/react": "19.1.4",
|
"@types/react": "19.1.4",
|
||||||
"@types/seedrandom": "3.0.8",
|
"@types/seedrandom": "3.0.8",
|
||||||
|
"@types/web-push": "3.6.4",
|
||||||
"@typescript-eslint/parser": "8.32.1",
|
"@typescript-eslint/parser": "8.32.1",
|
||||||
"astro-icon": "1.1.5",
|
"astro-icon": "1.1.5",
|
||||||
"date-fns": "4.1.0",
|
"date-fns": "4.1.0",
|
||||||
|
|||||||
@@ -0,0 +1,5 @@
|
|||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "Service" ADD COLUMN "previousSlugs" TEXT[] DEFAULT ARRAY[]::TEXT[];
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "Service_previousSlugs_idx" ON "Service"("previousSlugs");
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
-- CreateEnum
|
||||||
|
CREATE TYPE "KycLevelClarification" AS ENUM ('NONE', 'DEPENDS_ON_PARTNERS');
|
||||||
|
|
||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "Service" ADD COLUMN "kycLevelClarification" "KycLevelClarification",
|
||||||
|
ADD COLUMN "kycLevelDetailsId" INTEGER;
|
||||||
@@ -0,0 +1,25 @@
|
|||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "PushSubscription" (
|
||||||
|
"id" SERIAL NOT NULL,
|
||||||
|
"userId" INTEGER NOT NULL,
|
||||||
|
"endpoint" TEXT NOT NULL,
|
||||||
|
"p256dh" TEXT NOT NULL,
|
||||||
|
"auth" TEXT NOT NULL,
|
||||||
|
"userAgent" TEXT,
|
||||||
|
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"updatedAt" TIMESTAMP(3) NOT NULL,
|
||||||
|
|
||||||
|
CONSTRAINT "PushSubscription_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "PushSubscription_endpoint_key" ON "PushSubscription"("endpoint");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "PushSubscription_userId_idx" ON "PushSubscription"("userId");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "PushSubscription_endpoint_idx" ON "PushSubscription"("endpoint");
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "PushSubscription" ADD CONSTRAINT "PushSubscription_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
/*
|
||||||
|
Warnings:
|
||||||
|
|
||||||
|
- You are about to drop the column `kycLevelDetailsId` on the `Service` table. All the data in the column will be lost.
|
||||||
|
- Made the column `kycLevelClarification` on table `Service` required. This step will fail if there are existing NULL values in that column.
|
||||||
|
|
||||||
|
*/
|
||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "Service" DROP COLUMN "kycLevelDetailsId",
|
||||||
|
ALTER COLUMN "kycLevelClarification" SET NOT NULL,
|
||||||
|
ALTER COLUMN "kycLevelClarification" SET DEFAULT 'NONE';
|
||||||
@@ -297,6 +297,11 @@ enum ServiceSuggestionType {
|
|||||||
EDIT_SERVICE
|
EDIT_SERVICE
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum KycLevelClarification {
|
||||||
|
NONE
|
||||||
|
DEPENDS_ON_PARTNERS
|
||||||
|
}
|
||||||
|
|
||||||
model ServiceSuggestion {
|
model ServiceSuggestion {
|
||||||
id Int @id @default(autoincrement())
|
id Int @id @default(autoincrement())
|
||||||
type ServiceSuggestionType
|
type ServiceSuggestionType
|
||||||
@@ -336,9 +341,11 @@ model Service {
|
|||||||
id Int @id @default(autoincrement())
|
id Int @id @default(autoincrement())
|
||||||
name String
|
name String
|
||||||
slug String @unique
|
slug String @unique
|
||||||
|
previousSlugs String[] @default([])
|
||||||
description String
|
description String
|
||||||
categories Category[] @relation("ServiceToCategory")
|
categories Category[] @relation("ServiceToCategory")
|
||||||
kycLevel Int @default(4)
|
kycLevel Int @default(4)
|
||||||
|
kycLevelClarification KycLevelClarification @default(NONE)
|
||||||
overallScore Int @default(0)
|
overallScore Int @default(0)
|
||||||
privacyScore Int @default(0)
|
privacyScore Int @default(0)
|
||||||
trustScore Int @default(0)
|
trustScore Int @default(0)
|
||||||
@@ -396,6 +403,7 @@ model Service {
|
|||||||
@@index([createdAt])
|
@@index([createdAt])
|
||||||
@@index([updatedAt])
|
@@index([updatedAt])
|
||||||
@@index([slug])
|
@@index([slug])
|
||||||
|
@@index([previousSlugs])
|
||||||
}
|
}
|
||||||
|
|
||||||
model ServiceContactMethod {
|
model ServiceContactMethod {
|
||||||
@@ -506,6 +514,7 @@ model User {
|
|||||||
notifications Notification[] @relation("NotificationOwner")
|
notifications Notification[] @relation("NotificationOwner")
|
||||||
notificationPreferences NotificationPreferences?
|
notificationPreferences NotificationPreferences?
|
||||||
serviceAffiliations ServiceUser[] @relation("UserServices")
|
serviceAffiliations ServiceUser[] @relation("UserServices")
|
||||||
|
pushSubscriptions PushSubscription[]
|
||||||
|
|
||||||
@@index([createdAt])
|
@@index([createdAt])
|
||||||
@@index([totalKarma])
|
@@index([totalKarma])
|
||||||
@@ -654,3 +663,21 @@ model Announcement {
|
|||||||
|
|
||||||
@@index([isActive, startDate, endDate])
|
@@index([isActive, startDate, endDate])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
model PushSubscription {
|
||||||
|
id Int @id @default(autoincrement())
|
||||||
|
userId Int
|
||||||
|
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
||||||
|
endpoint String @unique
|
||||||
|
/// Public key for encryption
|
||||||
|
p256dh String
|
||||||
|
/// Authentication secret
|
||||||
|
auth String
|
||||||
|
/// To identify different devices
|
||||||
|
userAgent String?
|
||||||
|
createdAt DateTime @default(now())
|
||||||
|
updatedAt DateTime @updatedAt
|
||||||
|
|
||||||
|
@@index([userId])
|
||||||
|
@@index([endpoint])
|
||||||
|
}
|
||||||
|
|||||||
@@ -18,8 +18,9 @@ import {
|
|||||||
type User,
|
type User,
|
||||||
type ServiceVisibility,
|
type ServiceVisibility,
|
||||||
ServiceSuggestionType,
|
ServiceSuggestionType,
|
||||||
|
KycLevelClarification,
|
||||||
} from '@prisma/client'
|
} from '@prisma/client'
|
||||||
import { uniqBy } from 'lodash-es'
|
import { omit, uniqBy } from 'lodash-es'
|
||||||
import { generateUsername } from 'unique-username-generator'
|
import { generateUsername } from 'unique-username-generator'
|
||||||
|
|
||||||
import { kycLevels } from '../src/constants/kycLevels'
|
import { kycLevels } from '../src/constants/kycLevels'
|
||||||
@@ -612,8 +613,14 @@ const generateFakeService = (users: User[]) => {
|
|||||||
return {
|
return {
|
||||||
name,
|
name,
|
||||||
slug,
|
slug,
|
||||||
|
previousSlugs: faker.helpers.maybe(() => [`${slug}-old`], { probability: 0.5 }),
|
||||||
description: faker.helpers.arrayElement(serviceDescriptions),
|
description: faker.helpers.arrayElement(serviceDescriptions),
|
||||||
kycLevel: faker.helpers.arrayElement(kycLevels.map((level) => level.value)),
|
kycLevel: faker.helpers.arrayElement(kycLevels.map((level) => level.value)),
|
||||||
|
kycLevelClarification: faker.helpers.maybe(
|
||||||
|
() =>
|
||||||
|
faker.helpers.arrayElement(omit(Object.values(KycLevelClarification), [KycLevelClarification.NONE])),
|
||||||
|
{ probability: 0.25 }
|
||||||
|
),
|
||||||
overallScore: 0,
|
overallScore: 0,
|
||||||
privacyScore: 0,
|
privacyScore: 0,
|
||||||
trustScore: 0,
|
trustScore: 0,
|
||||||
@@ -1134,7 +1141,7 @@ async function main() {
|
|||||||
// ---- Create services ----
|
// ---- Create services ----
|
||||||
const services = await Promise.all(
|
const services = await Promise.all(
|
||||||
Array.from({ length: numServices }, async () => {
|
Array.from({ length: numServices }, async () => {
|
||||||
const serviceData = generateFakeService(users)
|
const serviceData = generateFakeService([...users, ...Object.values(specialUsers)])
|
||||||
const randomCategories = faker.helpers.arrayElements(categories, { min: 1, max: 3 })
|
const randomCategories = faker.helpers.arrayElements(categories, { min: 1, max: 3 })
|
||||||
|
|
||||||
const service = await prisma.service.create({
|
const service = await prisma.service.create({
|
||||||
@@ -1274,7 +1281,7 @@ async function main() {
|
|||||||
// ---- Create service suggestions for normal_dev user ----
|
// ---- Create service suggestions for normal_dev user ----
|
||||||
// First create 3 CREATE_SERVICE suggestions with their services
|
// First create 3 CREATE_SERVICE suggestions with their services
|
||||||
for (let i = 0; i < 3; i++) {
|
for (let i = 0; i < 3; i++) {
|
||||||
const serviceData = generateFakeService(users)
|
const serviceData = generateFakeService([...users, ...Object.values(specialUsers)])
|
||||||
const randomCategories = faker.helpers.arrayElements(categories, { min: 1, max: 3 })
|
const randomCategories = faker.helpers.arrayElements(categories, { min: 1, max: 3 })
|
||||||
|
|
||||||
const service = await prisma.service.create({
|
const service = await prisma.service.create({
|
||||||
|
|||||||
16
web/prisma/triggers/12_notification_push_trigger.sql
Normal file
16
web/prisma/triggers/12_notification_push_trigger.sql
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
CREATE OR REPLACE FUNCTION trigger_notification_push()
|
||||||
|
RETURNS TRIGGER AS $$
|
||||||
|
BEGIN
|
||||||
|
PERFORM pg_notify('notification_created', json_build_object('id', NEW.id)::text);
|
||||||
|
RETURN NEW;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
-- Drop the trigger if it exists to ensure a clean setup
|
||||||
|
DROP TRIGGER IF EXISTS notification_push_trigger ON "Notification";
|
||||||
|
|
||||||
|
-- Create the trigger to fire after inserts
|
||||||
|
CREATE TRIGGER notification_push_trigger
|
||||||
|
AFTER INSERT ON "Notification"
|
||||||
|
FOR EACH ROW
|
||||||
|
EXECUTE FUNCTION trigger_notification_push();
|
||||||
83
web/public/sw.js
Normal file
83
web/public/sw.js
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
// @ts-check
|
||||||
|
|
||||||
|
/// <reference lib="webworker" />
|
||||||
|
|
||||||
|
/** @type {ServiceWorkerGlobalScope} */
|
||||||
|
// @ts-expect-error
|
||||||
|
const typedSelf = self
|
||||||
|
|
||||||
|
const CACHE_NAME = 'kycnot-sw-push-notifications-v1'
|
||||||
|
|
||||||
|
typedSelf.addEventListener('install', (event) => {
|
||||||
|
console.log('Service Worker installing')
|
||||||
|
typedSelf.skipWaiting()
|
||||||
|
})
|
||||||
|
|
||||||
|
typedSelf.addEventListener('activate', (event) => {
|
||||||
|
console.log('Service Worker activating')
|
||||||
|
event.waitUntil(typedSelf.clients.claim())
|
||||||
|
})
|
||||||
|
|
||||||
|
typedSelf.addEventListener('push', (event) => {
|
||||||
|
console.log('Push event received:', event)
|
||||||
|
|
||||||
|
if (!event.data) {
|
||||||
|
console.log('Push event but no data')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
let notificationData
|
||||||
|
try {
|
||||||
|
notificationData = event.data.json()
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error parsing push data:', error)
|
||||||
|
notificationData = {
|
||||||
|
title: 'New Notification',
|
||||||
|
options: {
|
||||||
|
body: event.data.text() || 'You have a new notification',
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const { title, options } = notificationData
|
||||||
|
|
||||||
|
const notificationOptions = {
|
||||||
|
body: options.body || '',
|
||||||
|
icon: options.icon || '/favicon.svg',
|
||||||
|
badge: options.badge || '/favicon.svg',
|
||||||
|
data: options.data || {},
|
||||||
|
requireInteraction: false,
|
||||||
|
silent: false,
|
||||||
|
...options,
|
||||||
|
}
|
||||||
|
|
||||||
|
event.waitUntil(typedSelf.registration.showNotification(title, notificationOptions))
|
||||||
|
})
|
||||||
|
|
||||||
|
typedSelf.addEventListener('notificationclick', (event) => {
|
||||||
|
console.log('Notification clicked:', event)
|
||||||
|
|
||||||
|
event.notification.close()
|
||||||
|
|
||||||
|
const url = event.notification.data?.url || '/'
|
||||||
|
|
||||||
|
event.waitUntil(
|
||||||
|
typedSelf.clients.matchAll({ type: 'window' }).then((clientList) => {
|
||||||
|
// If a window is already open, focus it
|
||||||
|
for (const client of clientList) {
|
||||||
|
if (client.url === url && 'focus' in client) {
|
||||||
|
return client.focus()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Otherwise, open a new window
|
||||||
|
if (typedSelf.clients.openWindow) {
|
||||||
|
return typedSelf.clients.openWindow(url)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
typedSelf.addEventListener('notificationclose', (event) => {
|
||||||
|
console.log('Notification closed:', event)
|
||||||
|
})
|
||||||
@@ -1,11 +1,9 @@
|
|||||||
import { type Prisma, type PrismaClient } from '@prisma/client'
|
import { type Prisma } from '@prisma/client'
|
||||||
import { ActionError } from 'astro:actions'
|
import { ActionError } from 'astro:actions'
|
||||||
import { z } from 'zod'
|
import { z } from 'zod'
|
||||||
|
|
||||||
import { defineProtectedAction } from '../../lib/defineProtectedAction'
|
import { defineProtectedAction } from '../../lib/defineProtectedAction'
|
||||||
import { prisma as prismaInstance } from '../../lib/prisma'
|
import { prisma } from '../../lib/prisma'
|
||||||
|
|
||||||
const prisma = prismaInstance as PrismaClient
|
|
||||||
|
|
||||||
const selectAnnouncementReturnFields = {
|
const selectAnnouncementReturnFields = {
|
||||||
id: true,
|
id: true,
|
||||||
|
|||||||
@@ -1,15 +1,17 @@
|
|||||||
import { adminAnnouncementActions } from './announcement'
|
import { adminAnnouncementActions } from './announcement'
|
||||||
import { adminAttributeActions } from './attribute'
|
import { adminAttributeActions } from './attribute'
|
||||||
import { adminEventActions } from './event'
|
import { adminEventActions } from './event'
|
||||||
|
import { adminNotificationActions } from './notification'
|
||||||
import { adminServiceActions } from './service'
|
import { adminServiceActions } from './service'
|
||||||
import { adminServiceSuggestionActions } from './serviceSuggestion'
|
import { adminServiceSuggestionActions } from './serviceSuggestion'
|
||||||
import { adminUserActions } from './user'
|
import { adminUserActions } from './user'
|
||||||
import { verificationStep } from './verificationStep'
|
import { verificationStep } from './verificationStep'
|
||||||
|
|
||||||
export const adminActions = {
|
export const adminActions = {
|
||||||
attribute: adminAttributeActions,
|
|
||||||
announcement: adminAnnouncementActions,
|
announcement: adminAnnouncementActions,
|
||||||
|
attribute: adminAttributeActions,
|
||||||
event: adminEventActions,
|
event: adminEventActions,
|
||||||
|
notification: adminNotificationActions,
|
||||||
service: adminServiceActions,
|
service: adminServiceActions,
|
||||||
serviceSuggestions: adminServiceSuggestionActions,
|
serviceSuggestions: adminServiceSuggestionActions,
|
||||||
user: adminUserActions,
|
user: adminUserActions,
|
||||||
|
|||||||
80
web/src/actions/admin/notification.ts
Normal file
80
web/src/actions/admin/notification.ts
Normal file
@@ -0,0 +1,80 @@
|
|||||||
|
import { z } from 'astro/zod'
|
||||||
|
import { sumBy } from 'lodash-es'
|
||||||
|
|
||||||
|
import { defineProtectedAction } from '../../lib/defineProtectedAction'
|
||||||
|
import { prisma } from '../../lib/prisma'
|
||||||
|
import { sendPushNotification } from '../../lib/webPush'
|
||||||
|
import { stringListOfSlugsSchemaRequired } from '../../lib/zodUtils'
|
||||||
|
|
||||||
|
export const adminNotificationActions = {
|
||||||
|
webPush: {
|
||||||
|
test: defineProtectedAction({
|
||||||
|
accept: 'form',
|
||||||
|
permissions: 'admin',
|
||||||
|
input: z.object({
|
||||||
|
userNames: stringListOfSlugsSchemaRequired,
|
||||||
|
title: z.string().min(1).nullable(),
|
||||||
|
body: z.string().nullable(),
|
||||||
|
url: z.string().url().optional(),
|
||||||
|
}),
|
||||||
|
handler: async (input) => {
|
||||||
|
const subscriptions = await prisma.pushSubscription.findMany({
|
||||||
|
where: { user: { name: { in: input.userNames } } },
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
endpoint: true,
|
||||||
|
p256dh: true,
|
||||||
|
auth: true,
|
||||||
|
userAgent: true,
|
||||||
|
user: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
name: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const results = await Promise.allSettled(
|
||||||
|
subscriptions.map(async (subscription) => {
|
||||||
|
const result = await sendPushNotification(
|
||||||
|
{
|
||||||
|
endpoint: subscription.endpoint,
|
||||||
|
keys: {
|
||||||
|
p256dh: subscription.p256dh,
|
||||||
|
auth: subscription.auth,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: input.title ?? 'Test Notification',
|
||||||
|
body: input.body ?? 'This is a test push notification from KYCNot.me',
|
||||||
|
url: input.url ?? '/',
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// If subscription is invalid, remove it from database
|
||||||
|
if (result.error && (result.error.statusCode === 410 || result.error.statusCode === 404)) {
|
||||||
|
await prisma.pushSubscription.delete({
|
||||||
|
where: { id: subscription.id },
|
||||||
|
})
|
||||||
|
console.info(`Removed invalid subscription for user ${subscription.user.name}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
return result.success
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
|
const successCount = sumBy(results, (r) => (r.status === 'fulfilled' && r.value ? 1 : 0))
|
||||||
|
const failureCount = sumBy(results, (r) => (r.status === 'fulfilled' && r.value ? 0 : 1))
|
||||||
|
const now = new Date()
|
||||||
|
return {
|
||||||
|
message: `Sent to ${successCount.toLocaleString()} devices, ${failureCount.toLocaleString()} failed. Sent at ${now.toLocaleString()}`,
|
||||||
|
totalSubscriptions: subscriptions.length,
|
||||||
|
successCount,
|
||||||
|
failureCount,
|
||||||
|
sentAt: now,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
}
|
||||||
@@ -1,42 +1,19 @@
|
|||||||
import { Currency, ServiceVisibility, VerificationStatus } from '@prisma/client'
|
import { Currency, ServiceVisibility, VerificationStatus, KycLevelClarification } from '@prisma/client'
|
||||||
import { z } from 'astro/zod'
|
import { z } from 'astro/zod'
|
||||||
import { ActionError } from 'astro:actions'
|
import { ActionError } from 'astro:actions'
|
||||||
|
import { uniq } from 'lodash-es'
|
||||||
import slugify from 'slugify'
|
import slugify from 'slugify'
|
||||||
|
|
||||||
import { defineProtectedAction } from '../../lib/defineProtectedAction'
|
import { defineProtectedAction } from '../../lib/defineProtectedAction'
|
||||||
import { saveFileLocally } from '../../lib/fileStorage'
|
import { saveFileLocally, deleteFileLocally } from '../../lib/fileStorage'
|
||||||
import { prisma } from '../../lib/prisma'
|
import { prisma } from '../../lib/prisma'
|
||||||
import { separateServiceUrlsByType } from '../../lib/urls'
|
import { separateServiceUrlsByType } from '../../lib/urls'
|
||||||
import { imageFileSchema, stringListOfUrlsSchemaRequired, zodCohercedNumber } from '../../lib/zodUtils'
|
import {
|
||||||
|
imageFileSchema,
|
||||||
const serviceSchemaBase = z.object({
|
stringListOfUrlsSchemaRequired,
|
||||||
id: z.number().int().positive(),
|
zodCohercedNumber,
|
||||||
slug: z
|
zodContactMethod,
|
||||||
.string()
|
} from '../../lib/zodUtils'
|
||||||
.regex(/^[a-z0-9-]+$/, 'Allowed characters: lowercase letters, numbers, and hyphens')
|
|
||||||
.optional(),
|
|
||||||
name: z.string().min(1).max(40),
|
|
||||||
description: z.string().min(1),
|
|
||||||
allServiceUrls: stringListOfUrlsSchemaRequired,
|
|
||||||
tosUrls: stringListOfUrlsSchemaRequired,
|
|
||||||
kycLevel: z.coerce.number().int().min(0).max(4),
|
|
||||||
attributes: z.array(z.coerce.number().int().positive()),
|
|
||||||
categories: z.array(z.coerce.number().int().positive()).min(1),
|
|
||||||
verificationStatus: z.nativeEnum(VerificationStatus),
|
|
||||||
verificationSummary: z.string().optional().nullable().default(null),
|
|
||||||
verificationProofMd: z.string().optional().nullable().default(null),
|
|
||||||
acceptedCurrencies: z.array(z.nativeEnum(Currency)),
|
|
||||||
referral: z
|
|
||||||
.string()
|
|
||||||
.regex(/^(\?\w+=.|\/.+)/, 'Referral must be a valid URL parameter or path, not a full URL')
|
|
||||||
.optional()
|
|
||||||
.nullable()
|
|
||||||
.default(null),
|
|
||||||
imageFile: imageFileSchema,
|
|
||||||
overallScore: zodCohercedNumber(z.number().int().min(0).max(10)).optional(),
|
|
||||||
serviceVisibility: z.nativeEnum(ServiceVisibility),
|
|
||||||
internalNote: z.string().optional(),
|
|
||||||
})
|
|
||||||
|
|
||||||
const addSlugIfMissing = <
|
const addSlugIfMissing = <
|
||||||
T extends {
|
T extends {
|
||||||
@@ -57,12 +34,61 @@ const addSlugIfMissing = <
|
|||||||
}),
|
}),
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const serviceSchemaBase = z.object({
|
||||||
|
id: z.number().int().positive(),
|
||||||
|
slug: z
|
||||||
|
.string()
|
||||||
|
.regex(/^[a-z0-9-]+$/, 'Allowed characters: lowercase letters, numbers, and hyphens')
|
||||||
|
.optional(),
|
||||||
|
name: z.string().min(1).max(40),
|
||||||
|
description: z.string().min(1),
|
||||||
|
allServiceUrls: stringListOfUrlsSchemaRequired,
|
||||||
|
tosUrls: stringListOfUrlsSchemaRequired,
|
||||||
|
kycLevel: z.coerce.number().int().min(0).max(4),
|
||||||
|
kycLevelClarification: z.nativeEnum(KycLevelClarification).optional().nullable().default(null),
|
||||||
|
attributes: z.array(z.coerce.number().int().positive()),
|
||||||
|
categories: z.array(z.coerce.number().int().positive()).min(1),
|
||||||
|
verificationStatus: z.nativeEnum(VerificationStatus),
|
||||||
|
verificationSummary: z.string().optional().nullable().default(null),
|
||||||
|
verificationProofMd: z.string().optional().nullable().default(null),
|
||||||
|
acceptedCurrencies: z.array(z.nativeEnum(Currency)),
|
||||||
|
referral: z
|
||||||
|
.string()
|
||||||
|
.regex(/^(\?\w+=.|\/.+)/, 'Referral must be a valid URL parameter or path, not a full URL')
|
||||||
|
.optional()
|
||||||
|
.nullable()
|
||||||
|
.default(null),
|
||||||
|
imageFile: imageFileSchema,
|
||||||
|
overallScore: zodCohercedNumber(z.number().int().min(0).max(10)).optional(),
|
||||||
|
serviceVisibility: z.nativeEnum(ServiceVisibility),
|
||||||
|
internalNote: z.string().optional(),
|
||||||
|
})
|
||||||
|
|
||||||
|
// Define schema for the create action input
|
||||||
|
const createServiceInputSchema = serviceSchemaBase.omit({ id: true }).transform(addSlugIfMissing)
|
||||||
|
|
||||||
|
// Define schema for the update action input
|
||||||
|
const updateServiceInputSchema = serviceSchemaBase
|
||||||
|
.extend({
|
||||||
|
removeImage: z.boolean().optional(),
|
||||||
|
})
|
||||||
|
.transform(addSlugIfMissing)
|
||||||
|
|
||||||
|
const evidenceImageAddSchema = z.object({
|
||||||
|
serviceId: z.number().int().positive(),
|
||||||
|
imageFile: imageFileSchema,
|
||||||
|
})
|
||||||
|
|
||||||
|
const evidenceImageDeleteSchema = z.object({
|
||||||
|
fileUrl: z.string().startsWith('/files/evidence/', 'Must be a valid evidence file URL'),
|
||||||
|
})
|
||||||
|
|
||||||
export const adminServiceActions = {
|
export const adminServiceActions = {
|
||||||
create: defineProtectedAction({
|
create: defineProtectedAction({
|
||||||
accept: 'form',
|
accept: 'form',
|
||||||
permissions: 'admin',
|
permissions: 'admin',
|
||||||
input: serviceSchemaBase.omit({ id: true }).transform(addSlugIfMissing),
|
input: createServiceInputSchema,
|
||||||
handler: async (input, context) => {
|
handler: async (input: z.infer<typeof createServiceInputSchema>, context) => {
|
||||||
const existing = await prisma.service.findUnique({
|
const existing = await prisma.service.findUnique({
|
||||||
where: {
|
where: {
|
||||||
slug: input.slug,
|
slug: input.slug,
|
||||||
@@ -95,6 +121,7 @@ export const adminServiceActions = {
|
|||||||
onionUrls,
|
onionUrls,
|
||||||
i2pUrls,
|
i2pUrls,
|
||||||
kycLevel: input.kycLevel,
|
kycLevel: input.kycLevel,
|
||||||
|
kycLevelClarification: input.kycLevelClarification ?? undefined,
|
||||||
verificationStatus: input.verificationStatus,
|
verificationStatus: input.verificationStatus,
|
||||||
verificationSummary: input.verificationSummary,
|
verificationSummary: input.verificationSummary,
|
||||||
verificationProofMd: input.verificationProofMd,
|
verificationProofMd: input.verificationProofMd,
|
||||||
@@ -136,12 +163,8 @@ export const adminServiceActions = {
|
|||||||
update: defineProtectedAction({
|
update: defineProtectedAction({
|
||||||
accept: 'form',
|
accept: 'form',
|
||||||
permissions: 'admin',
|
permissions: 'admin',
|
||||||
input: serviceSchemaBase
|
input: updateServiceInputSchema,
|
||||||
.extend({
|
handler: async (input: z.infer<typeof updateServiceInputSchema>) => {
|
||||||
removeImage: z.boolean().optional(),
|
|
||||||
})
|
|
||||||
.transform(addSlugIfMissing),
|
|
||||||
handler: async (input) => {
|
|
||||||
const anotherServiceWithNewSlug = await prisma.service.findUnique({
|
const anotherServiceWithNewSlug = await prisma.service.findUnique({
|
||||||
where: {
|
where: {
|
||||||
slug: input.slug,
|
slug: input.slug,
|
||||||
@@ -156,19 +179,24 @@ export const adminServiceActions = {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
const imageUrl = input.removeImage
|
|
||||||
? null
|
|
||||||
: input.imageFile
|
|
||||||
? await saveFileLocally(input.imageFile, input.imageFile.name)
|
|
||||||
: undefined
|
|
||||||
|
|
||||||
const existingService = await prisma.service.findUnique({
|
const existingService = await prisma.service.findUnique({
|
||||||
where: { id: input.id },
|
where: { id: input.id },
|
||||||
include: {
|
select: {
|
||||||
categories: true,
|
slug: true,
|
||||||
|
previousSlugs: true,
|
||||||
|
categories: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
attributes: {
|
attributes: {
|
||||||
include: {
|
select: {
|
||||||
attribute: true,
|
attributeId: true,
|
||||||
|
attribute: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -189,6 +217,12 @@ export const adminServiceActions = {
|
|||||||
const attributesToAdd = input.attributes.filter((aId) => !existingAttributeIds.includes(aId))
|
const attributesToAdd = input.attributes.filter((aId) => !existingAttributeIds.includes(aId))
|
||||||
const attributesToRemove = existingAttributeIds.filter((aId) => !input.attributes.includes(aId))
|
const attributesToRemove = existingAttributeIds.filter((aId) => !input.attributes.includes(aId))
|
||||||
|
|
||||||
|
const imageUrl = input.removeImage
|
||||||
|
? null
|
||||||
|
: input.imageFile
|
||||||
|
? await saveFileLocally(input.imageFile, input.imageFile.name)
|
||||||
|
: undefined
|
||||||
|
|
||||||
const {
|
const {
|
||||||
web: serviceUrls,
|
web: serviceUrls,
|
||||||
onion: onionUrls,
|
onion: onionUrls,
|
||||||
@@ -205,6 +239,7 @@ export const adminServiceActions = {
|
|||||||
onionUrls,
|
onionUrls,
|
||||||
i2pUrls,
|
i2pUrls,
|
||||||
kycLevel: input.kycLevel,
|
kycLevel: input.kycLevel,
|
||||||
|
kycLevelClarification: input.kycLevelClarification ?? undefined,
|
||||||
verificationStatus: input.verificationStatus,
|
verificationStatus: input.verificationStatus,
|
||||||
verificationSummary: input.verificationSummary,
|
verificationSummary: input.verificationSummary,
|
||||||
verificationProofMd: input.verificationProofMd,
|
verificationProofMd: input.verificationProofMd,
|
||||||
@@ -213,6 +248,14 @@ export const adminServiceActions = {
|
|||||||
serviceVisibility: input.serviceVisibility,
|
serviceVisibility: input.serviceVisibility,
|
||||||
slug: input.slug,
|
slug: input.slug,
|
||||||
overallScore: input.overallScore,
|
overallScore: input.overallScore,
|
||||||
|
previousSlugs:
|
||||||
|
existingService.slug !== input.slug
|
||||||
|
? {
|
||||||
|
set: uniq([...existingService.previousSlugs, existingService.slug]).filter(
|
||||||
|
(slug) => slug !== input.slug
|
||||||
|
),
|
||||||
|
}
|
||||||
|
: undefined,
|
||||||
|
|
||||||
imageUrl,
|
imageUrl,
|
||||||
categories: {
|
categories: {
|
||||||
@@ -243,7 +286,7 @@ export const adminServiceActions = {
|
|||||||
permissions: 'admin',
|
permissions: 'admin',
|
||||||
input: z.object({
|
input: z.object({
|
||||||
label: z.string().min(1).max(50).nullable(),
|
label: z.string().min(1).max(50).nullable(),
|
||||||
value: z.string().url(),
|
value: zodContactMethod,
|
||||||
serviceId: z.number().int().positive(),
|
serviceId: z.number().int().positive(),
|
||||||
}),
|
}),
|
||||||
handler: async (input) => {
|
handler: async (input) => {
|
||||||
@@ -375,4 +418,50 @@ export const adminServiceActions = {
|
|||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
|
|
||||||
|
evidenceImage: {
|
||||||
|
add: defineProtectedAction({
|
||||||
|
accept: 'form',
|
||||||
|
permissions: 'admin',
|
||||||
|
input: evidenceImageAddSchema,
|
||||||
|
handler: async (input) => {
|
||||||
|
const service = await prisma.service.findUnique({
|
||||||
|
where: { id: input.serviceId },
|
||||||
|
select: { slug: true },
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!service) {
|
||||||
|
throw new ActionError({
|
||||||
|
code: 'NOT_FOUND',
|
||||||
|
message: 'Service not found to associate image with.',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!input.imageFile) {
|
||||||
|
throw new ActionError({
|
||||||
|
code: 'BAD_REQUEST',
|
||||||
|
message: 'Image file is required.',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const imageUrl = await saveFileLocally(
|
||||||
|
input.imageFile,
|
||||||
|
input.imageFile.name,
|
||||||
|
`evidence/${service.slug}`
|
||||||
|
)
|
||||||
|
|
||||||
|
return { imageUrl }
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
delete: defineProtectedAction({
|
||||||
|
accept: 'form',
|
||||||
|
permissions: 'admin',
|
||||||
|
input: evidenceImageDeleteSchema,
|
||||||
|
handler: async (input) => {
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-call
|
||||||
|
await deleteFileLocally(input.fileUrl)
|
||||||
|
return { success: true }
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import { z } from 'astro/zod'
|
|||||||
import { ActionError } from 'astro:actions'
|
import { ActionError } from 'astro:actions'
|
||||||
import { pick } from 'lodash-es'
|
import { pick } from 'lodash-es'
|
||||||
|
|
||||||
|
import { getKycLevelClarificationInfo } from '../../constants/kycLevelClarifications'
|
||||||
import { getKycLevelInfo } from '../../constants/kycLevels'
|
import { getKycLevelInfo } from '../../constants/kycLevels'
|
||||||
import { getVerificationStatusInfo } from '../../constants/verificationStatus'
|
import { getVerificationStatusInfo } from '../../constants/verificationStatus'
|
||||||
import { defineProtectedAction } from '../../lib/defineProtectedAction'
|
import { defineProtectedAction } from '../../lib/defineProtectedAction'
|
||||||
@@ -44,7 +45,31 @@ export const apiServiceActions = {
|
|||||||
.flatMap((url) => [url, url.endsWith('/') ? url.slice(0, -1) : `${url}/`])
|
.flatMap((url) => [url, url.endsWith('/') ? url.slice(0, -1) : `${url}/`])
|
||||||
: undefined
|
: undefined
|
||||||
|
|
||||||
const service = await prisma.service.findFirst({
|
const select = {
|
||||||
|
id: true,
|
||||||
|
name: true,
|
||||||
|
slug: true,
|
||||||
|
description: true,
|
||||||
|
kycLevel: true,
|
||||||
|
kycLevelClarification: true,
|
||||||
|
verificationStatus: true,
|
||||||
|
categories: {
|
||||||
|
select: {
|
||||||
|
name: true,
|
||||||
|
slug: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
serviceUrls: true,
|
||||||
|
onionUrls: true,
|
||||||
|
i2pUrls: true,
|
||||||
|
tosUrls: true,
|
||||||
|
referral: true,
|
||||||
|
listedAt: true,
|
||||||
|
verifiedAt: true,
|
||||||
|
serviceVisibility: true,
|
||||||
|
} as const satisfies Prisma.ServiceSelect
|
||||||
|
|
||||||
|
let service = await prisma.service.findFirst({
|
||||||
where: {
|
where: {
|
||||||
listedAt: { lte: new Date() },
|
listedAt: { lte: new Date() },
|
||||||
serviceVisibility: { in: ['PUBLIC', 'ARCHIVED', 'UNLISTED'] },
|
serviceVisibility: { in: ['PUBLIC', 'ARCHIVED', 'UNLISTED'] },
|
||||||
@@ -61,30 +86,21 @@ export const apiServiceActions = {
|
|||||||
: []),
|
: []),
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
select: {
|
select,
|
||||||
id: true,
|
|
||||||
name: true,
|
|
||||||
slug: true,
|
|
||||||
description: true,
|
|
||||||
kycLevel: true,
|
|
||||||
verificationStatus: true,
|
|
||||||
categories: {
|
|
||||||
select: {
|
|
||||||
name: true,
|
|
||||||
slug: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
serviceUrls: true,
|
|
||||||
onionUrls: true,
|
|
||||||
i2pUrls: true,
|
|
||||||
tosUrls: true,
|
|
||||||
referral: true,
|
|
||||||
listedAt: true,
|
|
||||||
verifiedAt: true,
|
|
||||||
serviceVisibility: true,
|
|
||||||
},
|
|
||||||
})
|
})
|
||||||
|
|
||||||
|
if (!service && input.slug) {
|
||||||
|
service = await prisma.service.findFirst({
|
||||||
|
where: {
|
||||||
|
listedAt: { lte: new Date() },
|
||||||
|
serviceVisibility: { in: ['PUBLIC', 'ARCHIVED', 'UNLISTED'] },
|
||||||
|
|
||||||
|
previousSlugs: { has: input.slug },
|
||||||
|
},
|
||||||
|
select,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
if (
|
if (
|
||||||
!service ||
|
!service ||
|
||||||
(service.serviceVisibility !== 'PUBLIC' &&
|
(service.serviceVisibility !== 'PUBLIC' &&
|
||||||
@@ -116,6 +132,12 @@ export const apiServiceActions = {
|
|||||||
verifiedAt: service.verifiedAt,
|
verifiedAt: service.verifiedAt,
|
||||||
kycLevel: service.kycLevel,
|
kycLevel: service.kycLevel,
|
||||||
kycLevelInfo: pick(getKycLevelInfo(service.kycLevel.toString()), ['value', 'name', 'description']),
|
kycLevelInfo: pick(getKycLevelInfo(service.kycLevel.toString()), ['value', 'name', 'description']),
|
||||||
|
kycLevelClarification: service.kycLevelClarification,
|
||||||
|
kycLevelClarificationInfo: pick(getKycLevelClarificationInfo(service.kycLevelClarification), [
|
||||||
|
'value',
|
||||||
|
'name',
|
||||||
|
'description',
|
||||||
|
]),
|
||||||
categories: service.categories,
|
categories: service.categories,
|
||||||
listedAt: service.listedAt,
|
listedAt: service.listedAt,
|
||||||
serviceUrls: [...service.serviceUrls, ...service.onionUrls, ...service.i2pUrls].map(
|
serviceUrls: [...service.serviceUrls, ...service.onionUrls, ...service.i2pUrls].map(
|
||||||
|
|||||||
@@ -23,6 +23,63 @@ export const notificationActions = {
|
|||||||
})
|
})
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
webPush: {
|
||||||
|
subscribe: defineProtectedAction({
|
||||||
|
accept: 'json',
|
||||||
|
permissions: 'user',
|
||||||
|
input: z.object({
|
||||||
|
endpoint: z.string(),
|
||||||
|
p256dhKey: z.string(),
|
||||||
|
authKey: z.string(),
|
||||||
|
userAgent: z.string().optional(),
|
||||||
|
}),
|
||||||
|
handler: async (input, context) => {
|
||||||
|
await prisma.pushSubscription.upsert({
|
||||||
|
where: {
|
||||||
|
userId: context.locals.user.id,
|
||||||
|
endpoint: input.endpoint,
|
||||||
|
},
|
||||||
|
update: {
|
||||||
|
p256dh: input.p256dhKey,
|
||||||
|
auth: input.authKey,
|
||||||
|
userAgent: input.userAgent,
|
||||||
|
},
|
||||||
|
create: {
|
||||||
|
userId: context.locals.user.id,
|
||||||
|
endpoint: input.endpoint,
|
||||||
|
p256dh: input.p256dhKey,
|
||||||
|
auth: input.authKey,
|
||||||
|
userAgent: input.userAgent,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
|
||||||
|
unsubscribe: defineProtectedAction({
|
||||||
|
accept: 'json',
|
||||||
|
permissions: 'user',
|
||||||
|
input: z.object({
|
||||||
|
endpoint: z.string().optional(),
|
||||||
|
}),
|
||||||
|
handler: async (input, context) => {
|
||||||
|
if (input.endpoint) {
|
||||||
|
await prisma.pushSubscription.deleteMany({
|
||||||
|
where: {
|
||||||
|
userId: context.locals.user.id,
|
||||||
|
endpoint: input.endpoint,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
await prisma.pushSubscription.deleteMany({
|
||||||
|
where: {
|
||||||
|
userId: context.locals.user.id,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
},
|
||||||
preferences: {
|
preferences: {
|
||||||
update: defineProtectedAction({
|
update: defineProtectedAction({
|
||||||
accept: 'form',
|
accept: 'form',
|
||||||
@@ -31,7 +88,7 @@ export const notificationActions = {
|
|||||||
enableOnMyCommentStatusChange: z.coerce.boolean().optional(),
|
enableOnMyCommentStatusChange: z.coerce.boolean().optional(),
|
||||||
enableAutowatchMyComments: z.coerce.boolean().optional(),
|
enableAutowatchMyComments: z.coerce.boolean().optional(),
|
||||||
enableNotifyPendingRepliesOnWatch: z.coerce.boolean().optional(),
|
enableNotifyPendingRepliesOnWatch: z.coerce.boolean().optional(),
|
||||||
karmaNotificationThreshold: z.coerce.number().int().min(1).optional(),
|
karmaNotificationThreshold: z.coerce.number().int().min(1).max(1_000_000).optional(),
|
||||||
}),
|
}),
|
||||||
handler: async (input, context) => {
|
handler: async (input, context) => {
|
||||||
await prisma.notificationPreferences.upsert({
|
await prisma.notificationPreferences.upsert({
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { Currency } from '@prisma/client'
|
import { Currency, KycLevelClarification } from '@prisma/client'
|
||||||
import { z } from 'astro/zod'
|
import { z } from 'astro/zod'
|
||||||
import { ActionError } from 'astro:actions'
|
import { ActionError } from 'astro:actions'
|
||||||
import { formatDistanceStrict } from 'date-fns'
|
import { formatDistanceStrict } from 'date-fns'
|
||||||
@@ -10,7 +10,12 @@ import { findServicesBySimilarity } from '../lib/findServicesBySimilarity'
|
|||||||
import { handleHoneypotTrap } from '../lib/honeypot'
|
import { handleHoneypotTrap } from '../lib/honeypot'
|
||||||
import { prisma } from '../lib/prisma'
|
import { prisma } from '../lib/prisma'
|
||||||
import { separateServiceUrlsByType } from '../lib/urls'
|
import { separateServiceUrlsByType } from '../lib/urls'
|
||||||
import { imageFileSchemaRequired, stringListOfUrlsSchemaRequired, zodCohercedNumber } from '../lib/zodUtils'
|
import {
|
||||||
|
imageFileSchemaRequired,
|
||||||
|
stringListOfContactMethodsSchema,
|
||||||
|
stringListOfUrlsSchemaRequired,
|
||||||
|
zodCohercedNumber,
|
||||||
|
} from '../lib/zodUtils'
|
||||||
|
|
||||||
import type { Prisma } from '@prisma/client'
|
import type { Prisma } from '@prisma/client'
|
||||||
|
|
||||||
@@ -153,7 +158,9 @@ export const serviceSuggestionActions = {
|
|||||||
description: z.string().min(1).max(SUGGESTION_DESCRIPTION_MAX_LENGTH),
|
description: z.string().min(1).max(SUGGESTION_DESCRIPTION_MAX_LENGTH),
|
||||||
allServiceUrls: stringListOfUrlsSchemaRequired,
|
allServiceUrls: stringListOfUrlsSchemaRequired,
|
||||||
tosUrls: stringListOfUrlsSchemaRequired,
|
tosUrls: stringListOfUrlsSchemaRequired,
|
||||||
|
contactMethods: stringListOfContactMethodsSchema,
|
||||||
kycLevel: zodCohercedNumber(z.coerce.number().int().min(0).max(4)),
|
kycLevel: zodCohercedNumber(z.coerce.number().int().min(0).max(4)),
|
||||||
|
kycLevelClarification: z.nativeEnum(KycLevelClarification),
|
||||||
attributes: z.array(z.coerce.number().int().positive()),
|
attributes: z.array(z.coerce.number().int().positive()),
|
||||||
categories: z.array(z.coerce.number().int().positive()).min(1),
|
categories: z.array(z.coerce.number().int().positive()).min(1),
|
||||||
acceptedCurrencies: z.array(z.nativeEnum(Currency)).min(1),
|
acceptedCurrencies: z.array(z.nativeEnum(Currency)).min(1),
|
||||||
@@ -221,6 +228,7 @@ export const serviceSuggestionActions = {
|
|||||||
onionUrls,
|
onionUrls,
|
||||||
i2pUrls,
|
i2pUrls,
|
||||||
kycLevel: input.kycLevel,
|
kycLevel: input.kycLevel,
|
||||||
|
kycLevelClarification: input.kycLevelClarification,
|
||||||
acceptedCurrencies: input.acceptedCurrencies,
|
acceptedCurrencies: input.acceptedCurrencies,
|
||||||
imageUrl,
|
imageUrl,
|
||||||
verificationStatus: 'COMMUNITY_CONTRIBUTED',
|
verificationStatus: 'COMMUNITY_CONTRIBUTED',
|
||||||
@@ -237,6 +245,11 @@ export const serviceSuggestionActions = {
|
|||||||
attributeId: id,
|
attributeId: id,
|
||||||
})),
|
})),
|
||||||
},
|
},
|
||||||
|
contactMethods: {
|
||||||
|
create: input.contactMethods.map((value) => ({
|
||||||
|
value,
|
||||||
|
})),
|
||||||
|
},
|
||||||
},
|
},
|
||||||
select: serviceSelect,
|
select: serviceSelect,
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -2,13 +2,14 @@
|
|||||||
import { Icon } from 'astro-icon/components'
|
import { Icon } from 'astro-icon/components'
|
||||||
import { tv, type VariantProps } from 'tailwind-variants'
|
import { tv, type VariantProps } from 'tailwind-variants'
|
||||||
|
|
||||||
|
import type { AstroChildren } from '../lib/astro'
|
||||||
import type { Polymorphic } from 'astro/types'
|
import type { Polymorphic } from 'astro/types'
|
||||||
|
|
||||||
const badge = tv({
|
const badge = tv({
|
||||||
slots: {
|
slots: {
|
||||||
base: 'inline-flex h-4 items-center justify-center gap-0.75 rounded-full px-1.25 text-[10px] font-medium',
|
base: 'inline-flex h-4 items-center justify-center gap-0.75 rounded-full px-1.25 text-[10px] font-medium',
|
||||||
icon: 'size-3 shrink-0',
|
icon: 'size-3 shrink-0',
|
||||||
text: 'mx-0.25 overflow-hidden text-ellipsis whitespace-nowrap',
|
text: 'mx-0.25 overflow-hidden text-ellipsis whitespace-nowrap [&>a]:hover:underline [&>a]:focus-visible:underline',
|
||||||
},
|
},
|
||||||
variants: {
|
variants: {
|
||||||
color: {
|
color: {
|
||||||
@@ -122,18 +123,22 @@ const badge = tv({
|
|||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
type Props<Tag extends 'a' | 'div' | 'li' = 'div'> = Polymorphic<
|
type Props<
|
||||||
|
Tag extends 'a' | 'div' | 'li' = 'div',
|
||||||
|
Text extends string | undefined = string | undefined,
|
||||||
|
> = Polymorphic<
|
||||||
VariantProps<typeof badge> & {
|
VariantProps<typeof badge> & {
|
||||||
as: Tag
|
as: Tag
|
||||||
icon?: string
|
icon?: string
|
||||||
endIcon?: string
|
endIcon?: string
|
||||||
text: string
|
text?: Text
|
||||||
inlineIcon?: boolean
|
inlineIcon?: boolean
|
||||||
classNames?: {
|
classNames?: {
|
||||||
icon?: string
|
icon?: string
|
||||||
text?: string
|
text?: string
|
||||||
endIcon?: string
|
endIcon?: string
|
||||||
}
|
}
|
||||||
|
children?: Text extends string ? never : AstroChildren
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
|
|
||||||
@@ -161,7 +166,7 @@ const { base, icon: iconSlot, text: textSlot } = badge({ color, variant })
|
|||||||
<Icon name={iconName} class={iconSlot({ class: classNames?.icon })} is:inline={inlineIcon} />
|
<Icon name={iconName} class={iconSlot({ class: classNames?.icon })} is:inline={inlineIcon} />
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
<span class={textSlot({ class: classNames?.text })}>{textContent}</span>
|
<span class={textSlot({ class: classNames?.text })}>{textContent ?? <slot />}</span>
|
||||||
{
|
{
|
||||||
!!endIconName && (
|
!!endIconName && (
|
||||||
<Icon name={endIconName} class={iconSlot({ class: classNames?.endIcon })} is:inline={inlineIcon} />
|
<Icon name={endIconName} class={iconSlot({ class: classNames?.endIcon })} is:inline={inlineIcon} />
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ import { cn } from '../lib/cn'
|
|||||||
|
|
||||||
import type { HTMLAttributes, Polymorphic } from 'astro/types'
|
import type { HTMLAttributes, Polymorphic } from 'astro/types'
|
||||||
|
|
||||||
type Props<Tag extends 'a' | 'button' | 'label' = 'button'> = Polymorphic<
|
type Props<Tag extends 'a' | 'button' | 'label' | 'span' = 'button'> = Polymorphic<
|
||||||
Required<Pick<HTMLAttributes<'label'>, Tag extends 'label' ? 'for' : never>> &
|
Required<Pick<HTMLAttributes<'label'>, Tag extends 'label' ? 'for' : never>> &
|
||||||
VariantProps<typeof button> & {
|
VariantProps<typeof button> & {
|
||||||
as: Tag
|
as: Tag
|
||||||
@@ -249,7 +249,7 @@ const button = tv({
|
|||||||
})
|
})
|
||||||
|
|
||||||
const {
|
const {
|
||||||
as: Tag = 'button' as 'a' | 'button' | 'label',
|
as: Tag = 'button' as 'a' | 'button' | 'label' | 'span',
|
||||||
label,
|
label,
|
||||||
icon,
|
icon,
|
||||||
endIcon,
|
endIcon,
|
||||||
@@ -286,8 +286,7 @@ const ActualTag = disabled && Tag === 'a' ? 'span' : Tag
|
|||||||
|
|
||||||
<ActualTag
|
<ActualTag
|
||||||
class={base({ class: cn({ 'opacity-20 hover:opacity-50': disabled }, className) })}
|
class={base({ class: cn({ 'opacity-20 hover:opacity-50': disabled }, className) })}
|
||||||
role={role ??
|
role={role ?? (Tag === 'button' || Tag === 'label' || (disabled && Tag === 'a') ? undefined : 'button')}
|
||||||
(ActualTag === 'button' || ActualTag === 'label' || ActualTag === 'span' ? undefined : 'button')}
|
|
||||||
aria-disabled={disabled}
|
aria-disabled={disabled}
|
||||||
{...dataAstroReload && { 'data-astro-reload': dataAstroReload }}
|
{...dataAstroReload && { 'data-astro-reload': dataAstroReload }}
|
||||||
{...htmlProps}
|
{...htmlProps}
|
||||||
|
|||||||
@@ -102,7 +102,7 @@ const commentUrl = makeCommentUrl({ serviceSlug, commentId: comment.id, origin:
|
|||||||
{...htmlProps}
|
{...htmlProps}
|
||||||
id={`comment-${comment.id.toString()}`}
|
id={`comment-${comment.id.toString()}`}
|
||||||
class={cn([
|
class={cn([
|
||||||
'group',
|
'group bg-night-700',
|
||||||
depth > 0 && 'ml-3 border-b-0 border-l border-zinc-800 pt-2 pl-2 sm:ml-4',
|
depth > 0 && 'ml-3 border-b-0 border-l border-zinc-800 pt-2 pl-2 sm:ml-4',
|
||||||
comment.author.serviceAffiliations.some((affiliation) => affiliation.service.slug === serviceSlug) &&
|
comment.author.serviceAffiliations.some((affiliation) => affiliation.service.slug === serviceSlug) &&
|
||||||
'bg-[#182a1f]',
|
'bg-[#182a1f]',
|
||||||
@@ -243,13 +243,10 @@ const commentUrl = makeCommentUrl({ serviceSlug, commentId: comment.id, origin:
|
|||||||
comment.author.serviceAffiliations.map((affiliation) => {
|
comment.author.serviceAffiliations.map((affiliation) => {
|
||||||
const roleInfo = getServiceUserRoleInfo(affiliation.role)
|
const roleInfo = getServiceUserRoleInfo(affiliation.role)
|
||||||
return (
|
return (
|
||||||
<BadgeSmall
|
<BadgeSmall icon={roleInfo.icon} color={roleInfo.color} variant="faded" inlineIcon>
|
||||||
icon={roleInfo.icon}
|
{roleInfo.label} at
|
||||||
color={roleInfo.color}
|
<a href={`/service/${affiliation.service.slug}`}>{affiliation.service.name}</a>
|
||||||
text={`${roleInfo.label} at ${affiliation.service.name}`}
|
</BadgeSmall>
|
||||||
variant="faded"
|
|
||||||
inlineIcon
|
|
||||||
/>
|
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@@ -270,12 +267,6 @@ const commentUrl = makeCommentUrl({ serviceSlug, commentId: comment.id, origin:
|
|||||||
|
|
||||||
{comment.suspicious && <BadgeSmall icon="ri:spam-2-fill" color="red" text="Potential SPAM" inlineIcon />}
|
{comment.suspicious && <BadgeSmall icon="ri:spam-2-fill" color="red" text="Potential SPAM" inlineIcon />}
|
||||||
|
|
||||||
{
|
|
||||||
comment.requiresAdminReview && isAuthorOrPrivileged && (
|
|
||||||
<BadgeSmall icon="ri:alert-fill" color="yellow" text="Reported" inlineIcon />
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
{
|
||||||
comment.rating !== null && !comment.parentId && (
|
comment.rating !== null && !comment.parentId && (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
@@ -320,6 +311,19 @@ const commentUrl = makeCommentUrl({ serviceSlug, commentId: comment.id, origin:
|
|||||||
color={commentStatusById.REJECTED.color}
|
color={commentStatusById.REJECTED.color}
|
||||||
text={commentStatusById.REJECTED.label}
|
text={commentStatusById.REJECTED.label}
|
||||||
inlineIcon
|
inlineIcon
|
||||||
|
endIcon="ri:lock-line"
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
comment.requiresAdminReview && isAuthorOrPrivileged && (
|
||||||
|
<BadgeSmall
|
||||||
|
icon="ri:alert-fill"
|
||||||
|
color="yellow"
|
||||||
|
text="Needs admin review"
|
||||||
|
inlineIcon
|
||||||
|
endIcon="ri:lock-line"
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -89,7 +89,7 @@ if (!user || !user.admin || !user.moderator) return null
|
|||||||
data-comment-id={comment.id}
|
data-comment-id={comment.id}
|
||||||
data-user-id={user.id}
|
data-user-id={user.id}
|
||||||
>
|
>
|
||||||
{comment.requiresAdminReview ? 'No Admin Review' : 'Admin Review'}
|
{comment.requiresAdminReview ? 'No Admin Review' : 'Needs Admin Review'}
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<button
|
<button
|
||||||
|
|||||||
@@ -92,7 +92,7 @@ const userCommentsDisabled = user ? user.karmaUnlocks.commentsDisabled : false
|
|||||||
<div class="flex flex-wrap gap-4">
|
<div class="flex flex-wrap gap-4">
|
||||||
<InputRating name="rating" label="Rating" />
|
<InputRating name="rating" label="Rating" />
|
||||||
|
|
||||||
<InputWrapper label="Tags" name="tags">
|
<InputWrapper label="I experienced..." name="tags">
|
||||||
<label class="flex cursor-pointer items-center gap-2">
|
<label class="flex cursor-pointer items-center gap-2">
|
||||||
<input type="checkbox" name="issueKycRequested" class="text-red-400" />
|
<input type="checkbox" name="issueKycRequested" class="text-red-400" />
|
||||||
<span class="flex items-center gap-1 text-xs text-red-400">
|
<span class="flex items-center gap-1 text-xs text-red-400">
|
||||||
|
|||||||
@@ -3,24 +3,28 @@ import { cn } from '../lib/cn'
|
|||||||
|
|
||||||
import Button from './Button.astro'
|
import Button from './Button.astro'
|
||||||
|
|
||||||
import type { HTMLAttributes } from 'astro/types'
|
import type { ComponentProps, HTMLAttributes } from 'astro/types'
|
||||||
|
|
||||||
type Props = HTMLAttributes<'div'> & {
|
type Props = HTMLAttributes<'div'> & {
|
||||||
hideCancel?: boolean
|
hideCancel?: boolean
|
||||||
icon?: string
|
icon?: string
|
||||||
label?: string
|
label?: string
|
||||||
|
disabled?: boolean
|
||||||
|
color?: ComponentProps<typeof Button>['color']
|
||||||
}
|
}
|
||||||
|
|
||||||
const {
|
const {
|
||||||
hideCancel = false,
|
hideCancel = false,
|
||||||
icon = 'ri:send-plane-2-line',
|
icon = 'ri:send-plane-2-line',
|
||||||
label = 'Submit',
|
label = 'Submit',
|
||||||
|
disabled = false,
|
||||||
class: className,
|
class: className,
|
||||||
|
color = 'success',
|
||||||
...htmlProps
|
...htmlProps
|
||||||
} = Astro.props
|
} = Astro.props
|
||||||
---
|
---
|
||||||
|
|
||||||
<div class={cn('flex justify-between gap-2', className)} {...htmlProps}>
|
<div class={cn('flex justify-between gap-2', className)} {...htmlProps}>
|
||||||
{!hideCancel && <Button as="a" href="/" label="Cancel" icon="ri:close-line" color="gray" />}
|
{!hideCancel && <Button as="a" href="/" label="Cancel" icon="ri:close-line" color="gray" />}
|
||||||
<Button type="submit" label={label} icon={icon} class="ml-auto" color="success" />
|
<Button type="submit" label={label} icon={icon} class="ml-auto" color={color} disabled={disabled} />
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
374
web/src/components/PushNotificationBanner.astro
Normal file
374
web/src/components/PushNotificationBanner.astro
Normal file
@@ -0,0 +1,374 @@
|
|||||||
|
---
|
||||||
|
import { Icon } from 'astro-icon/components'
|
||||||
|
import { VAPID_PUBLIC_KEY } from 'astro:env/server'
|
||||||
|
|
||||||
|
import { cn } from '../lib/cn'
|
||||||
|
|
||||||
|
import Button from './Button.astro'
|
||||||
|
|
||||||
|
import type { Prisma } from '@prisma/client'
|
||||||
|
import type { HTMLAttributes } from 'astro/types'
|
||||||
|
|
||||||
|
type Props = HTMLAttributes<'div'> & {
|
||||||
|
dismissable?: boolean
|
||||||
|
hideIfEnabled?: boolean
|
||||||
|
pushSubscriptions: Prisma.PushSubscriptionGetPayload<{
|
||||||
|
select: {
|
||||||
|
endpoint: true
|
||||||
|
userAgent: true
|
||||||
|
}
|
||||||
|
}>[]
|
||||||
|
}
|
||||||
|
|
||||||
|
const { class: className, dismissable = false, pushSubscriptions, hideIfEnabled, ...props } = Astro.props
|
||||||
|
|
||||||
|
// TODO: Feature flag, enabled only for admins
|
||||||
|
if (!Astro.locals.user?.admin) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
---
|
||||||
|
|
||||||
|
<div
|
||||||
|
data-push-notification-banner
|
||||||
|
data-dismissed={undefined /* Updated by client script */}
|
||||||
|
data-supports-push-notifications={undefined /* Updated by client script */}
|
||||||
|
data-push-subscriptions={JSON.stringify(pushSubscriptions)}
|
||||||
|
data-is-enabled={undefined /* Updated by client script */}
|
||||||
|
class={cn(
|
||||||
|
'no-js:hidden relative isolate flex items-center justify-between gap-x-4 overflow-hidden rounded-xl bg-gradient-to-r from-blue-950/80 to-blue-900/60 p-4',
|
||||||
|
'data-dismissed:hidden',
|
||||||
|
hideIfEnabled && 'data-is-enabled:hidden',
|
||||||
|
'not-data-supports-push-notifications:hidden',
|
||||||
|
'data-is-enabled:**:data-show-if-disabled:hidden not-data-is-enabled:**:data-show-if-enabled:hidden',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<div aria-hidden="true" class="pointer-events-none absolute inset-0 -z-10 overflow-hidden">
|
||||||
|
<div
|
||||||
|
class="absolute top-0 -left-16 h-full w-1/3 bg-gradient-to-r from-blue-500/20 to-transparent opacity-50 blur-xl"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="absolute top-0 -right-16 h-full w-1/3 bg-gradient-to-l from-blue-500/20 to-transparent opacity-50 blur-xl"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="flex items-center gap-x-3">
|
||||||
|
<div class="rounded-md bg-blue-800/30 p-2">
|
||||||
|
<Icon name="ri:notification-4-line" class="size-5 text-blue-300" />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<h3 class="font-medium text-blue-100">
|
||||||
|
<span data-show-if-enabled>Push notifications enabled</span>
|
||||||
|
<span data-show-if-disabled>Turn on push notifications?</span>
|
||||||
|
</h3>
|
||||||
|
<p class="text-sm text-blue-200/80">
|
||||||
|
<span data-show-if-enabled>Turn notifications off for this device?</span>
|
||||||
|
<span data-show-if-disabled>Get notifications on this device.</span>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="flex items-center gap-2">
|
||||||
|
{dismissable && <Button as="span" label="Skip" variant="faded" data-dismiss-button />}
|
||||||
|
<Button
|
||||||
|
as="span"
|
||||||
|
label="Yes, notify me"
|
||||||
|
color="white"
|
||||||
|
data-push-action="subscribe"
|
||||||
|
data-vapid-public-key={VAPID_PUBLIC_KEY}
|
||||||
|
data-show-if-disabled
|
||||||
|
/>
|
||||||
|
<Button
|
||||||
|
as="span"
|
||||||
|
label="Stop notifications"
|
||||||
|
color="white"
|
||||||
|
variant="faded"
|
||||||
|
data-push-action="unsubscribe"
|
||||||
|
data-vapid-public-key={VAPID_PUBLIC_KEY}
|
||||||
|
data-show-if-enabled
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
/////////////////////////////////////////////////////////////
|
||||||
|
// Script to handle push notification banner dismissal. //
|
||||||
|
/////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
import { typedLocalStorage } from '../lib/localstorage'
|
||||||
|
|
||||||
|
document.addEventListener('astro:page-load', () => {
|
||||||
|
let pushNotificationsBannerDismissedAt = typedLocalStorage.pushNotificationsBannerDismissedAt.get()
|
||||||
|
|
||||||
|
if (
|
||||||
|
pushNotificationsBannerDismissedAt &&
|
||||||
|
pushNotificationsBannerDismissedAt < new Date(Date.now() - 1000 * 60 * 60 * 24 * 365) // 1 year
|
||||||
|
) {
|
||||||
|
typedLocalStorage.pushNotificationsBannerDismissedAt.remove()
|
||||||
|
pushNotificationsBannerDismissedAt = undefined
|
||||||
|
}
|
||||||
|
|
||||||
|
document.querySelectorAll<HTMLElement>('[data-push-notification-banner]').forEach((banner) => {
|
||||||
|
const skipButton = banner.querySelector<HTMLElement>('[data-dismiss-button]')
|
||||||
|
if (!skipButton) return
|
||||||
|
|
||||||
|
if (pushNotificationsBannerDismissedAt) {
|
||||||
|
banner.dataset.dismissed = ''
|
||||||
|
}
|
||||||
|
|
||||||
|
skipButton.addEventListener('click', (event) => {
|
||||||
|
event.preventDefault()
|
||||||
|
event.stopPropagation()
|
||||||
|
|
||||||
|
banner.dataset.dismissed = ''
|
||||||
|
|
||||||
|
const now = new Date()
|
||||||
|
typedLocalStorage.pushNotificationsBannerDismissedAt.set(now)
|
||||||
|
pushNotificationsBannerDismissedAt = now
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
/////////////////////////////////////////////////////////
|
||||||
|
// Script to style when notifications enabled. //
|
||||||
|
////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
type ServerSubscription = {
|
||||||
|
endpoint: string
|
||||||
|
userAgent: string | null
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Parse push subscriptions from string */
|
||||||
|
function parsePushSubscriptions(subscriptionsAsString: string | undefined) {
|
||||||
|
try {
|
||||||
|
if (typeof subscriptionsAsString !== 'string') throw new Error('Push subscriptions must be a string')
|
||||||
|
|
||||||
|
const subscriptions = JSON.parse(subscriptionsAsString)
|
||||||
|
|
||||||
|
if (!Array.isArray(subscriptions)) throw new Error('Push subscriptions must be an array')
|
||||||
|
if (!subscriptions.every((s) => typeof s === 'object' && s !== null)) {
|
||||||
|
throw new Error('Push subscriptions must be an array of objects')
|
||||||
|
}
|
||||||
|
if (!subscriptions.every((s) => typeof s.endpoint === 'string')) {
|
||||||
|
throw new Error('Push subscriptions must be an array of objects with endpoint property')
|
||||||
|
}
|
||||||
|
if (!subscriptions.every((s) => typeof s.userAgent === 'string' || s.userAgent === null)) {
|
||||||
|
throw new Error('Push subscriptions must be an array of objects with userAgent property')
|
||||||
|
}
|
||||||
|
|
||||||
|
return subscriptions as ServerSubscription[]
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to parse push subscriptions:', error)
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Check if current device has an active push subscription */
|
||||||
|
async function getCurrentPushSubscription() {
|
||||||
|
try {
|
||||||
|
const registration = await navigator.serviceWorker.getRegistration()
|
||||||
|
if (!registration) return null
|
||||||
|
|
||||||
|
return await registration.pushManager.getSubscription()
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error getting current push subscription:', error)
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Check if current subscription matches any server subscription */
|
||||||
|
function isCurrentDeviceSubscribed(
|
||||||
|
currentSubscription: PushSubscription | null,
|
||||||
|
serverSubscriptions: ServerSubscription[]
|
||||||
|
) {
|
||||||
|
if (!currentSubscription || serverSubscriptions.length === 0) return false
|
||||||
|
|
||||||
|
const currentEndpoint = currentSubscription.endpoint
|
||||||
|
const currentUserAgent = navigator.userAgent
|
||||||
|
|
||||||
|
return serverSubscriptions.some(
|
||||||
|
(sub) =>
|
||||||
|
sub.endpoint === currentEndpoint && (sub.userAgent === currentUserAgent || sub.userAgent === null)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener('astro:page-load', async () => {
|
||||||
|
document.querySelectorAll<HTMLElement>('[data-push-notification-banner]').forEach(async (banner) => {
|
||||||
|
const serverSubscriptions = parsePushSubscriptions(banner.dataset.pushSubscriptions)
|
||||||
|
const currentSubscription = await getCurrentPushSubscription()
|
||||||
|
const isSubscribed = isCurrentDeviceSubscribed(currentSubscription, serverSubscriptions)
|
||||||
|
|
||||||
|
if (isSubscribed) banner.dataset.isEnabled = ''
|
||||||
|
})
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
/////////////////////////////////////////////////////////////
|
||||||
|
// Script to handle push notification subscription. //
|
||||||
|
/////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
import type { actions } from 'astro:actions'
|
||||||
|
import type { ActionInput } from '../lib/astroActions'
|
||||||
|
|
||||||
|
/** Utility function to convert VAPID key */
|
||||||
|
function urlB64ToUint8Array(base64String: string) {
|
||||||
|
const cleaned = base64String.trim().replace(/\s+/g, '').replace(/\-/g, '+').replace(/_/g, '/')
|
||||||
|
const padding = '='.repeat((4 - (cleaned.length % 4)) % 4)
|
||||||
|
const base64 = cleaned + padding
|
||||||
|
|
||||||
|
const rawData = window.atob(base64)
|
||||||
|
const outputArray = new Uint8Array(rawData.length)
|
||||||
|
|
||||||
|
for (let i = 0; i < rawData.length; ++i) {
|
||||||
|
outputArray[i] = rawData.charCodeAt(i)
|
||||||
|
}
|
||||||
|
return outputArray
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Check for browser support */
|
||||||
|
function checkSupport() {
|
||||||
|
const isSecure =
|
||||||
|
window.isSecureContext ||
|
||||||
|
window.location.hostname === 'localhost' ||
|
||||||
|
window.location.hostname === '127.0.0.1'
|
||||||
|
return isSecure && 'serviceWorker' in navigator && 'PushManager' in window && 'Notification' in window
|
||||||
|
}
|
||||||
|
|
||||||
|
async function registerServiceWorker() {
|
||||||
|
try {
|
||||||
|
const registration = await navigator.serviceWorker.register('/sw.js')
|
||||||
|
console.log('Service Worker registered:', registration)
|
||||||
|
|
||||||
|
const readyRegistration = await navigator.serviceWorker.ready
|
||||||
|
console.log('Service Worker is active and ready:', readyRegistration)
|
||||||
|
|
||||||
|
return readyRegistration
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Service Worker registration failed:', error)
|
||||||
|
throw error
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function subscribeToPush(vapidPublicKey: string) {
|
||||||
|
try {
|
||||||
|
if (!checkSupport()) return
|
||||||
|
|
||||||
|
// Request notification permission
|
||||||
|
const permission = await Notification.requestPermission()
|
||||||
|
if (permission !== 'granted') {
|
||||||
|
alert('Push notifications permission denied')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const registration = await registerServiceWorker()
|
||||||
|
|
||||||
|
// Subscribe to push manager
|
||||||
|
const subscription = await registration.pushManager.subscribe({
|
||||||
|
userVisibleOnly: true,
|
||||||
|
applicationServerKey: urlB64ToUint8Array(vapidPublicKey),
|
||||||
|
})
|
||||||
|
|
||||||
|
const p256dh = subscription.getKey('p256dh')
|
||||||
|
const auth = subscription.getKey('auth')
|
||||||
|
|
||||||
|
// Send subscription to server
|
||||||
|
const response = await fetch('/internal-api/notifications/subscribe', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({
|
||||||
|
endpoint: subscription.endpoint,
|
||||||
|
userAgent: navigator.userAgent,
|
||||||
|
p256dhKey: p256dh ? btoa(String.fromCharCode(...new Uint8Array(p256dh))) : '',
|
||||||
|
authKey: auth ? btoa(String.fromCharCode(...new Uint8Array(auth))) : '',
|
||||||
|
} satisfies ActionInput<typeof actions.notification.webPush.subscribe>),
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error(`HTTP error! status: ${response.status}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('Push subscription successful')
|
||||||
|
|
||||||
|
// Reload page to update UI
|
||||||
|
window.location.reload()
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Push subscription failed:', error)
|
||||||
|
alert('Error enabling push notifications. This may be due to browser settings or other restrictions.')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function unsubscribeFromPush() {
|
||||||
|
try {
|
||||||
|
const registration = await navigator.serviceWorker.getRegistration()
|
||||||
|
if (!registration) {
|
||||||
|
console.log('No service worker registration found')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const subscription = await registration.pushManager.getSubscription()
|
||||||
|
if (!subscription) {
|
||||||
|
console.log('No push subscription found')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Unsubscribe from browser
|
||||||
|
await subscription.unsubscribe()
|
||||||
|
|
||||||
|
// Remove from server
|
||||||
|
const response = await fetch('/internal-api/notifications/unsubscribe', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({
|
||||||
|
endpoint: subscription.endpoint,
|
||||||
|
} satisfies ActionInput<typeof actions.notification.webPush.unsubscribe>),
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error(`HTTP error! status: ${response.status}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('Push unsubscription successful')
|
||||||
|
|
||||||
|
// Reload page to update UI
|
||||||
|
window.location.reload()
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Push unsubscription failed:', error)
|
||||||
|
alert('Failed to unsubscribe from push notifications')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener('astro:page-load', () => {
|
||||||
|
const supportsPushNotifications = checkSupport()
|
||||||
|
if (supportsPushNotifications) {
|
||||||
|
document.querySelectorAll<HTMLElement>('[data-push-notification-banner]').forEach((element) => {
|
||||||
|
element.dataset.supportsPushNotifications = ''
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
document.querySelectorAll<HTMLElement>('[data-push-action]').forEach((button) => {
|
||||||
|
const vapidPublicKey = button.dataset.vapidPublicKey
|
||||||
|
if (!vapidPublicKey) {
|
||||||
|
console.error('Environment variable VAPID_PUBLIC_KEY is not set')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
button.addEventListener('click', async (event) => {
|
||||||
|
event.preventDefault()
|
||||||
|
event.stopPropagation()
|
||||||
|
|
||||||
|
const action = button.dataset.pushAction
|
||||||
|
if (action === 'subscribe') {
|
||||||
|
await subscribeToPush(vapidPublicKey)
|
||||||
|
} else if (action === 'unsubscribe') {
|
||||||
|
await unsubscribeFromPush()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
</script>
|
||||||
@@ -87,6 +87,25 @@ function makeLink(url: string, referral: string | null) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const bitcointalkMatch = /^(?:https?:\/\/)?(?:www\.)?bitcointalk\.org$/.exec(hostname)
|
||||||
|
if (bitcointalkMatch) {
|
||||||
|
return {
|
||||||
|
type: 'clearnet' as const,
|
||||||
|
url: urlWithReferral,
|
||||||
|
textBits: [
|
||||||
|
{
|
||||||
|
style: 'normal',
|
||||||
|
text: 'BitcoinTalk ',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
style: 'irrelevant',
|
||||||
|
text: 'thread',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
icon: networksBySlug.clearnet.icon,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
type: 'clearnet' as const,
|
type: 'clearnet' as const,
|
||||||
url: urlWithReferral,
|
url: urlWithReferral,
|
||||||
|
|||||||
@@ -56,7 +56,15 @@ const wasRecentlyAdded = isPast(listedDate) && differenceInDays(new Date(), list
|
|||||||
) : service.verificationStatus === 'COMMUNITY_CONTRIBUTED' ? (
|
) : service.verificationStatus === 'COMMUNITY_CONTRIBUTED' ? (
|
||||||
<div class="mb-3 flex items-center gap-2 rounded-md bg-yellow-600/30 p-2 text-sm text-yellow-200">
|
<div class="mb-3 flex items-center gap-2 rounded-md bg-yellow-600/30 p-2 text-sm text-yellow-200">
|
||||||
<Icon name="ri:alert-line" class="size-5 text-yellow-400" />
|
<Icon name="ri:alert-line" class="size-5 text-yellow-400" />
|
||||||
<span>Community-contributed. Information not reviewed.</span>
|
<span>
|
||||||
|
Community-contributed. Information not reviewed.
|
||||||
|
<a
|
||||||
|
href="/about#suggestion-review-process"
|
||||||
|
class="text-yellow-100 underline opacity-50 transition-opacity hover:opacity-100 focus-visible:opacity-100"
|
||||||
|
>
|
||||||
|
Learn more
|
||||||
|
</a>
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
) : wasRecentlyAdded ? (
|
) : wasRecentlyAdded ? (
|
||||||
<div class="mb-3 rounded-md bg-red-900/50 p-2 text-sm text-red-400">
|
<div class="mb-3 rounded-md bg-red-900/50 p-2 text-sm text-red-400">
|
||||||
@@ -64,11 +72,25 @@ const wasRecentlyAdded = isPast(listedDate) && differenceInDays(new Date(), list
|
|||||||
<TimeFormatted date={listedDate} daysUntilDate={RECENTLY_ADDED_DAYS} />
|
<TimeFormatted date={listedDate} daysUntilDate={RECENTLY_ADDED_DAYS} />
|
||||||
{service.verificationStatus !== 'VERIFICATION_SUCCESS' && ' and it is not verified'}. Proceed with
|
{service.verificationStatus !== 'VERIFICATION_SUCCESS' && ' and it is not verified'}. Proceed with
|
||||||
caution.
|
caution.
|
||||||
|
<a
|
||||||
|
href="/about#suggestion-review-process"
|
||||||
|
class="text-yellow-100 underline opacity-50 transition-opacity hover:opacity-100 focus-visible:opacity-100"
|
||||||
|
>
|
||||||
|
Learn more
|
||||||
|
</a>
|
||||||
</div>
|
</div>
|
||||||
) : service.verificationStatus !== 'VERIFICATION_SUCCESS' ? (
|
) : service.verificationStatus !== 'VERIFICATION_SUCCESS' ? (
|
||||||
<div class="mb-3 flex items-center gap-2 rounded-md bg-blue-600/30 p-2 text-sm text-blue-200">
|
<div class="mb-3 flex items-center gap-2 rounded-md bg-blue-600/30 p-2 text-sm text-blue-200">
|
||||||
<Icon name="ri:information-line" class="size-5 text-blue-400" />
|
<Icon name="ri:information-line" class="size-5 text-blue-400" />
|
||||||
<span>Basic checks passed, but not fully verified.</span>
|
<span>
|
||||||
|
Basic checks passed, but not fully verified.
|
||||||
|
<a
|
||||||
|
href="/about#suggestion-review-process"
|
||||||
|
class="text-yellow-100 underline opacity-50 transition-opacity hover:opacity-100 focus-visible:opacity-100"
|
||||||
|
>
|
||||||
|
Learn more
|
||||||
|
</a>
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
) : null
|
) : null
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,9 @@ import { parsePhoneNumberWithError } from 'libphonenumber-js'
|
|||||||
import { makeHelpersForOptions } from '../lib/makeHelpersForOptions'
|
import { makeHelpersForOptions } from '../lib/makeHelpersForOptions'
|
||||||
import { transformCase } from '../lib/strings'
|
import { transformCase } from '../lib/strings'
|
||||||
|
|
||||||
|
import type { Assert } from '../lib/assert'
|
||||||
|
import type { Equals } from 'ts-toolbelt/out/Any/Equals'
|
||||||
|
|
||||||
type ContactMethodInfo<T extends string | null | undefined = string> = {
|
type ContactMethodInfo<T extends string | null | undefined = string> = {
|
||||||
type: T
|
type: T
|
||||||
label: string
|
label: string
|
||||||
@@ -10,6 +13,7 @@ type ContactMethodInfo<T extends string | null | undefined = string> = {
|
|||||||
matcher: RegExp
|
matcher: RegExp
|
||||||
formatter: (match: RegExpMatchArray) => string | null
|
formatter: (match: RegExpMatchArray) => string | null
|
||||||
icon: string
|
icon: string
|
||||||
|
urlType: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export const {
|
export const {
|
||||||
@@ -22,9 +26,10 @@ export const {
|
|||||||
(type): ContactMethodInfo<typeof type> => ({
|
(type): ContactMethodInfo<typeof type> => ({
|
||||||
type,
|
type,
|
||||||
label: type ? transformCase(type, 'title') : String(type),
|
label: type ? transformCase(type, 'title') : String(type),
|
||||||
icon: 'ri:shield-fill',
|
icon: 'ri:link',
|
||||||
matcher: /(.*)/,
|
matcher: /(.*)/,
|
||||||
formatter: ([, value]) => value ?? String(value),
|
formatter: ([, value]) => value ?? String(value),
|
||||||
|
urlType: type ?? 'unknown',
|
||||||
}),
|
}),
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
@@ -33,24 +38,37 @@ export const {
|
|||||||
matcher: /mailto:(.+)/,
|
matcher: /mailto:(.+)/,
|
||||||
formatter: ([, value]) => value ?? 'Email',
|
formatter: ([, value]) => value ?? 'Email',
|
||||||
icon: 'ri:mail-line',
|
icon: 'ri:mail-line',
|
||||||
|
urlType: 'email',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: 'telephone',
|
type: 'telephone',
|
||||||
label: 'Telephone',
|
label: 'Telephone',
|
||||||
matcher: /tel:(.+)/,
|
matcher: /tel:(.+)/,
|
||||||
formatter: ([, value]) => {
|
formatter: ([, value]) => {
|
||||||
return value ? parsePhoneNumberWithError(value).formatInternational() : 'Telephone'
|
try {
|
||||||
|
return value ? parsePhoneNumberWithError(value).formatInternational() : 'Telephone'
|
||||||
|
} catch (_error) {
|
||||||
|
console.error(`Invalid telephone number: ${value ?? 'undefined'}`, _error)
|
||||||
|
return value ?? 'Telephone'
|
||||||
|
}
|
||||||
},
|
},
|
||||||
icon: 'ri:phone-line',
|
icon: 'ri:phone-line',
|
||||||
|
urlType: 'telephone',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: 'whatsapp',
|
type: 'whatsapp',
|
||||||
label: 'WhatsApp',
|
label: 'WhatsApp',
|
||||||
matcher: /^https?:\/\/(?:www\.)?wa\.me\/(.+)/,
|
matcher: /^https?:\/\/(?:www\.)?wa\.me\/(.+)/,
|
||||||
formatter: ([, value]) => {
|
formatter: ([, value]) => {
|
||||||
return value ? parsePhoneNumberWithError(value).formatInternational() : 'WhatsApp'
|
try {
|
||||||
|
return value ? parsePhoneNumberWithError(value).formatInternational() : 'WhatsApp'
|
||||||
|
} catch (_error) {
|
||||||
|
console.error(`Invalid WhatsApp number: ${value ?? 'undefined'}`, _error)
|
||||||
|
return value ?? 'WhatsApp'
|
||||||
|
}
|
||||||
},
|
},
|
||||||
icon: 'ri:whatsapp-line',
|
icon: 'ri:whatsapp-line',
|
||||||
|
urlType: 'url',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: 'telegram',
|
type: 'telegram',
|
||||||
@@ -58,6 +76,7 @@ export const {
|
|||||||
matcher: /^https?:\/\/(?:www\.)?t\.me\/(.+)/,
|
matcher: /^https?:\/\/(?:www\.)?t\.me\/(.+)/,
|
||||||
formatter: ([, value]) => (value ? `t.me/${value}` : 'Telegram'),
|
formatter: ([, value]) => (value ? `t.me/${value}` : 'Telegram'),
|
||||||
icon: 'ri:telegram-line',
|
icon: 'ri:telegram-line',
|
||||||
|
urlType: 'url',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: 'linkedin',
|
type: 'linkedin',
|
||||||
@@ -65,6 +84,7 @@ export const {
|
|||||||
matcher: /^https?:\/\/(?:www\.)?linkedin\.com\/(?:in|company)\/(.+)/,
|
matcher: /^https?:\/\/(?:www\.)?linkedin\.com\/(?:in|company)\/(.+)/,
|
||||||
formatter: ([, value]) => (value ? `in/${value}` : 'LinkedIn'),
|
formatter: ([, value]) => (value ? `in/${value}` : 'LinkedIn'),
|
||||||
icon: 'ri:linkedin-box-line',
|
icon: 'ri:linkedin-box-line',
|
||||||
|
urlType: 'url',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: 'x',
|
type: 'x',
|
||||||
@@ -72,6 +92,7 @@ export const {
|
|||||||
matcher: /^https?:\/\/(?:www\.)?x\.com\/(.+)/,
|
matcher: /^https?:\/\/(?:www\.)?x\.com\/(.+)/,
|
||||||
formatter: ([, value]) => (value ? `@${value}` : 'X'),
|
formatter: ([, value]) => (value ? `@${value}` : 'X'),
|
||||||
icon: 'ri:twitter-x-line',
|
icon: 'ri:twitter-x-line',
|
||||||
|
urlType: 'url',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: 'instagram',
|
type: 'instagram',
|
||||||
@@ -79,6 +100,7 @@ export const {
|
|||||||
matcher: /^https?:\/\/(?:www\.)?instagram\.com\/(.+)/,
|
matcher: /^https?:\/\/(?:www\.)?instagram\.com\/(.+)/,
|
||||||
formatter: ([, value]) => (value ? `@${value}` : 'Instagram'),
|
formatter: ([, value]) => (value ? `@${value}` : 'Instagram'),
|
||||||
icon: 'ri:instagram-line',
|
icon: 'ri:instagram-line',
|
||||||
|
urlType: 'url',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: 'matrix',
|
type: 'matrix',
|
||||||
@@ -86,6 +108,7 @@ export const {
|
|||||||
matcher: /^https?:\/\/(?:www\.)?matrix\.to\/#\/(.+)/,
|
matcher: /^https?:\/\/(?:www\.)?matrix\.to\/#\/(.+)/,
|
||||||
formatter: ([, value]) => (value ? `#${value}` : 'Matrix'),
|
formatter: ([, value]) => (value ? `#${value}` : 'Matrix'),
|
||||||
icon: 'ri:hashtag',
|
icon: 'ri:hashtag',
|
||||||
|
urlType: 'url',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: 'bitcointalk',
|
type: 'bitcointalk',
|
||||||
@@ -93,6 +116,7 @@ export const {
|
|||||||
matcher: /^https?:\/\/(?:www\.)?bitcointalk\.org/,
|
matcher: /^https?:\/\/(?:www\.)?bitcointalk\.org/,
|
||||||
formatter: () => 'BitcoinTalk',
|
formatter: () => 'BitcoinTalk',
|
||||||
icon: 'ri:btc-line',
|
icon: 'ri:btc-line',
|
||||||
|
urlType: 'url',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: 'simplex',
|
type: 'simplex',
|
||||||
@@ -100,6 +124,7 @@ export const {
|
|||||||
matcher: /^https?:\/\/(?:www\.)?(simplex\.chat)\//,
|
matcher: /^https?:\/\/(?:www\.)?(simplex\.chat)\//,
|
||||||
formatter: () => 'SimpleX Chat',
|
formatter: () => 'SimpleX Chat',
|
||||||
icon: 'simplex',
|
icon: 'simplex',
|
||||||
|
urlType: 'url',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: 'nostr',
|
type: 'nostr',
|
||||||
@@ -107,6 +132,7 @@ export const {
|
|||||||
matcher: /\b(npub1[a-zA-Z0-9]{58})\b/,
|
matcher: /\b(npub1[a-zA-Z0-9]{58})\b/,
|
||||||
formatter: () => 'Nostr',
|
formatter: () => 'Nostr',
|
||||||
icon: 'nostr',
|
icon: 'nostr',
|
||||||
|
urlType: 'url',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
// Website must go last because it's a catch-all
|
// Website must go last because it's a catch-all
|
||||||
@@ -115,6 +141,7 @@ export const {
|
|||||||
matcher: /^https?:\/\/(?:www\.)?((?:[a-zA-Z0-9-]+\.)+[a-zA-Z]+)/,
|
matcher: /^https?:\/\/(?:www\.)?((?:[a-zA-Z0-9-]+\.)+[a-zA-Z]+)/,
|
||||||
formatter: ([, value]) => value ?? 'Website',
|
formatter: ([, value]) => value ?? 'Website',
|
||||||
icon: 'ri:global-line',
|
icon: 'ri:global-line',
|
||||||
|
urlType: 'url',
|
||||||
},
|
},
|
||||||
] as const satisfies ContactMethodInfo[]
|
] as const satisfies ContactMethodInfo[]
|
||||||
)
|
)
|
||||||
@@ -135,3 +162,38 @@ export function formatContactMethod(url: string) {
|
|||||||
|
|
||||||
return { ...getContactMethodInfo('unknown'), formattedValue: url } as const
|
return { ...getContactMethodInfo('unknown'), formattedValue: url } as const
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type ContactMethodUrlTypeInfo<T extends string | null | undefined = string> = {
|
||||||
|
value: T
|
||||||
|
labelPlural: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export const {
|
||||||
|
dataArray: contactMethodUrlTypes,
|
||||||
|
dataObject: contactMethodUrlTypesById,
|
||||||
|
getFn: getContactMethodUrlTypeInfo,
|
||||||
|
} = makeHelpersForOptions(
|
||||||
|
'value',
|
||||||
|
(value): ContactMethodUrlTypeInfo<typeof value> => ({
|
||||||
|
value,
|
||||||
|
labelPlural: value ? transformCase(value, 'title') : String(value),
|
||||||
|
}),
|
||||||
|
[
|
||||||
|
{
|
||||||
|
value: 'email',
|
||||||
|
labelPlural: 'emails',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: 'telephone',
|
||||||
|
labelPlural: 'phone numbers',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: 'url',
|
||||||
|
labelPlural: 'URLs',
|
||||||
|
},
|
||||||
|
] as const satisfies ContactMethodUrlTypeInfo<(typeof contactMethods)[number]['urlType']>[]
|
||||||
|
)
|
||||||
|
|
||||||
|
type _ExpectUrlTypesToHaveAllValues = Assert<
|
||||||
|
Equals<(typeof contactMethods)[number]['urlType'], keyof typeof contactMethodUrlTypesById>
|
||||||
|
>
|
||||||
|
|||||||
39
web/src/constants/kycLevelClarifications.ts
Normal file
39
web/src/constants/kycLevelClarifications.ts
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
import { makeHelpersForOptions } from '../lib/makeHelpersForOptions'
|
||||||
|
import { transformCase } from '../lib/strings'
|
||||||
|
|
||||||
|
import type { KycLevelClarification } from '@prisma/client'
|
||||||
|
|
||||||
|
type KycLevelClarificationInfo<T extends string | null | undefined = string> = {
|
||||||
|
value: T
|
||||||
|
label: string
|
||||||
|
description: string
|
||||||
|
icon: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export const {
|
||||||
|
dataArray: kycLevelClarifications,
|
||||||
|
dataObject: kycLevelClarificationsById,
|
||||||
|
getFn: getKycLevelClarificationInfo,
|
||||||
|
} = makeHelpersForOptions(
|
||||||
|
'value',
|
||||||
|
(value): KycLevelClarificationInfo<typeof value> => ({
|
||||||
|
value,
|
||||||
|
label: value ? transformCase(value.replace('_', ' '), 'title') : String(value),
|
||||||
|
description: '',
|
||||||
|
icon: 'ri:question-line',
|
||||||
|
}),
|
||||||
|
[
|
||||||
|
{
|
||||||
|
value: 'NONE',
|
||||||
|
label: 'None',
|
||||||
|
description: 'No clarification needed.',
|
||||||
|
icon: 'ri:file-copy-line',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: 'DEPENDS_ON_PARTNERS',
|
||||||
|
label: 'Depends on partners',
|
||||||
|
description: 'May vary across partners.',
|
||||||
|
icon: 'ri:share-forward-line',
|
||||||
|
},
|
||||||
|
] as const satisfies KycLevelClarificationInfo<KycLevelClarification>[]
|
||||||
|
)
|
||||||
10
web/src/lib/assert.ts
Normal file
10
web/src/lib/assert.ts
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
/**
|
||||||
|
* Gives an error if the type is not equal to 1.
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* ```ts
|
||||||
|
* type _ExpectEquals = Assert<Equals<'a' | 'b', 'a'>> // Gives an error
|
||||||
|
* type _ExpectEquals = Assert<Equals<'a' | 'b', 'b' | 'a'>> // No error
|
||||||
|
* ```
|
||||||
|
*/
|
||||||
|
export type Assert<T extends 1> = T
|
||||||
@@ -69,6 +69,53 @@ export async function saveFileLocally(
|
|||||||
return url
|
return url
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* List all files in a specific subdirectory of the upload directory.
|
||||||
|
* Returns an array of web-accessible URLs.
|
||||||
|
*/
|
||||||
|
export async function listFiles(subDir: string): Promise<string[]> {
|
||||||
|
const { fsPath: uploadDir, webPath: webUploadPath } = getUploadDir(subDir)
|
||||||
|
try {
|
||||||
|
const files = await fs.readdir(uploadDir)
|
||||||
|
return files.map((file) => sanitizePath(`${webUploadPath}/${file}`))
|
||||||
|
} catch (error: unknown) {
|
||||||
|
const err = error as NodeJS.ErrnoException
|
||||||
|
if (err.code === 'ENOENT') {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
console.error(`Error listing files in ${uploadDir}:`, error)
|
||||||
|
throw error
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete a file locally given its web-accessible URL path
|
||||||
|
*/
|
||||||
|
export async function deleteFileLocally(fileUrl: string): Promise<void> {
|
||||||
|
// Extract the subpath and filename from the webPath
|
||||||
|
// Example: /files/evidence/service-slug/image.jpg -> evidence/service-slug/image.jpg
|
||||||
|
const basePath = '/files'
|
||||||
|
if (!fileUrl.startsWith(basePath)) {
|
||||||
|
throw new Error('Invalid file URL for deletion. Must start with /files')
|
||||||
|
}
|
||||||
|
|
||||||
|
const subPathAndFile = fileUrl.substring(basePath.length).replace(/^\/+/, '') // Remove leading /files/ and any extra leading slashes
|
||||||
|
const { fsPath: uploadDirWithoutSubDir } = getUploadDir() // Get base upload directory
|
||||||
|
const filePath = path.join(uploadDirWithoutSubDir, subPathAndFile)
|
||||||
|
|
||||||
|
try {
|
||||||
|
await fs.unlink(filePath)
|
||||||
|
} catch (error: unknown) {
|
||||||
|
const err = error as NodeJS.ErrnoException
|
||||||
|
if (err.code === 'ENOENT') {
|
||||||
|
console.warn(`File not found for deletion, but treating as success: ${filePath}`)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
console.error(`Error deleting file ${filePath}:`, error)
|
||||||
|
throw error
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function sanitizePath(inputPath: string): string {
|
function sanitizePath(inputPath: string): string {
|
||||||
let sanitized = inputPath.replace(/\\+/g, '/')
|
let sanitized = inputPath.replace(/\\+/g, '/')
|
||||||
// Collapse multiple slashes, but preserve protocol (e.g., http://)
|
// Collapse multiple slashes, but preserve protocol (e.g., http://)
|
||||||
|
|||||||
35
web/src/lib/json.ts
Normal file
35
web/src/lib/json.ts
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
import type { z } from 'astro:content'
|
||||||
|
|
||||||
|
// eslint-disable-next-line @typescript-eslint/consistent-type-definitions
|
||||||
|
interface JSONObject {
|
||||||
|
[k: string]: JSONValue
|
||||||
|
}
|
||||||
|
type JSONList = JSONValue[]
|
||||||
|
type JSONPrimitive = boolean | number | string | null
|
||||||
|
type JSONValue = Date | JSONList | JSONObject | JSONPrimitive
|
||||||
|
|
||||||
|
export type ZodJSON = z.ZodType<JSONValue>
|
||||||
|
|
||||||
|
export function zodParseJSON<T extends z.ZodType<JSONValue>, D extends z.output<T> | undefined = undefined>(
|
||||||
|
schema: T,
|
||||||
|
stringValue: string | null | undefined,
|
||||||
|
defaultValue?: D
|
||||||
|
): D | z.output<T> {
|
||||||
|
if (!stringValue) return defaultValue as D
|
||||||
|
|
||||||
|
let jsonValue: D | z.output<typeof schema> = defaultValue as D
|
||||||
|
try {
|
||||||
|
jsonValue = JSON.parse(stringValue)
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error)
|
||||||
|
return defaultValue as D
|
||||||
|
}
|
||||||
|
|
||||||
|
const parsedValue = schema.safeParse(jsonValue)
|
||||||
|
if (!parsedValue.success) {
|
||||||
|
console.error(parsedValue.error)
|
||||||
|
return defaultValue as D
|
||||||
|
}
|
||||||
|
|
||||||
|
return parsedValue.data
|
||||||
|
}
|
||||||
53
web/src/lib/localstorage.ts
Normal file
53
web/src/lib/localstorage.ts
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
import { z } from 'astro:schema'
|
||||||
|
|
||||||
|
import { zodParseJSON, type ZodJSON } from './json'
|
||||||
|
import { typedObjectEntries } from './objects'
|
||||||
|
|
||||||
|
function makeTypedLocalStorage<
|
||||||
|
Schemas extends Record<string, ZodJSON>,
|
||||||
|
T extends {
|
||||||
|
[K in keyof Schemas]: {
|
||||||
|
schema: Schemas[K]
|
||||||
|
default?: z.output<Schemas[K]> | undefined
|
||||||
|
key?: string
|
||||||
|
}
|
||||||
|
},
|
||||||
|
>(options: T) {
|
||||||
|
return Object.fromEntries(
|
||||||
|
typedObjectEntries(options).map(([originalKey, option]) => {
|
||||||
|
const key = option.key ?? originalKey
|
||||||
|
|
||||||
|
return [
|
||||||
|
key,
|
||||||
|
{
|
||||||
|
get: () => {
|
||||||
|
return zodParseJSON(option.schema, localStorage.getItem(key), option.default)
|
||||||
|
},
|
||||||
|
|
||||||
|
set: (value: z.input<typeof option.schema>) => {
|
||||||
|
localStorage.setItem(key, JSON.stringify(value))
|
||||||
|
},
|
||||||
|
|
||||||
|
remove: () => {
|
||||||
|
localStorage.removeItem(key)
|
||||||
|
},
|
||||||
|
|
||||||
|
default: option.default,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
})
|
||||||
|
) as {
|
||||||
|
[K in keyof T]: {
|
||||||
|
get: () => z.output<T[K]['schema']> | (T[K] extends { default: infer D } ? D : undefined)
|
||||||
|
set: (value: z.input<T[K]['schema']>) => void
|
||||||
|
remove: () => void
|
||||||
|
default: z.output<T[K]['schema']> | undefined
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const typedLocalStorage = makeTypedLocalStorage({
|
||||||
|
pushNotificationsBannerDismissedAt: {
|
||||||
|
schema: z.coerce.date(),
|
||||||
|
},
|
||||||
|
})
|
||||||
@@ -162,3 +162,16 @@ export function areEqualObjectsWithoutOrder<T extends Record<string, unknown>>(
|
|||||||
return undefined
|
return undefined
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Same as {@link Object.entries}, but with proper typing.
|
||||||
|
* @example
|
||||||
|
* typedObjectEntries({ a: 1, b: 2 }) // [['a', 1], ['b', 2]]
|
||||||
|
*/
|
||||||
|
export function typedObjectEntries<T extends Record<string, unknown>>(obj: T) {
|
||||||
|
return Object.entries(obj) as Prettify<
|
||||||
|
{
|
||||||
|
[K in Extract<keyof T, string>]: [K, T[K]]
|
||||||
|
}[Extract<keyof T, string>]
|
||||||
|
>[]
|
||||||
|
}
|
||||||
|
|||||||
248
web/src/lib/postgresListenerIntegration.ts
Normal file
248
web/src/lib/postgresListenerIntegration.ts
Normal file
@@ -0,0 +1,248 @@
|
|||||||
|
import { z } from 'astro/zod'
|
||||||
|
import { Client } from 'pg'
|
||||||
|
|
||||||
|
import { zodParseJSON } from './json'
|
||||||
|
import { makeNotificationContent, makeNotificationLink, makeNotificationTitle } from './notifications'
|
||||||
|
import { prisma } from './prisma'
|
||||||
|
import { getServerEnvVariable } from './serverEnvVariables'
|
||||||
|
import { sendPushNotification, type NotificationData } from './webPush'
|
||||||
|
|
||||||
|
import type { AstroIntegration, HookParameters } from 'astro'
|
||||||
|
|
||||||
|
const DATABASE_URL = getServerEnvVariable('DATABASE_URL')
|
||||||
|
const SITE_URL = getServerEnvVariable('SITE_URL')
|
||||||
|
|
||||||
|
let pgClient: Client | null = null
|
||||||
|
|
||||||
|
const INTEGRATION_NAME = 'postgres-listener'
|
||||||
|
|
||||||
|
async function handleNotificationCreated(
|
||||||
|
notificationId: number,
|
||||||
|
options: HookParameters<'astro:server:start'>
|
||||||
|
) {
|
||||||
|
const logger = options.logger.fork(INTEGRATION_NAME)
|
||||||
|
try {
|
||||||
|
logger.info(`Processing notification with ID: ${String(notificationId)}`)
|
||||||
|
|
||||||
|
const notification = await prisma.notification.findUnique({
|
||||||
|
where: { id: notificationId },
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
type: true,
|
||||||
|
userId: true,
|
||||||
|
aboutAccountStatusChange: true,
|
||||||
|
aboutCommentStatusChange: true,
|
||||||
|
aboutServiceVerificationStatusChange: true,
|
||||||
|
aboutSuggestionStatusChange: true,
|
||||||
|
aboutComment: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
author: { select: { id: true } },
|
||||||
|
status: true,
|
||||||
|
content: true,
|
||||||
|
communityNote: true,
|
||||||
|
parent: {
|
||||||
|
select: {
|
||||||
|
author: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
service: {
|
||||||
|
select: {
|
||||||
|
slug: true,
|
||||||
|
name: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
aboutServiceSuggestionId: true,
|
||||||
|
aboutServiceSuggestion: {
|
||||||
|
select: {
|
||||||
|
status: true,
|
||||||
|
service: {
|
||||||
|
select: {
|
||||||
|
name: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
aboutServiceSuggestionMessage: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
content: true,
|
||||||
|
suggestion: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
service: {
|
||||||
|
select: {
|
||||||
|
name: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
aboutEvent: {
|
||||||
|
select: {
|
||||||
|
title: true,
|
||||||
|
type: true,
|
||||||
|
service: {
|
||||||
|
select: {
|
||||||
|
slug: true,
|
||||||
|
name: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
aboutService: {
|
||||||
|
select: {
|
||||||
|
slug: true,
|
||||||
|
name: true,
|
||||||
|
verificationStatus: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
aboutKarmaTransaction: {
|
||||||
|
select: {
|
||||||
|
points: true,
|
||||||
|
action: true,
|
||||||
|
description: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
user: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
name: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!notification) {
|
||||||
|
logger.warn(`Notification with ID ${String(notificationId)} not found`)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const subscriptions = await prisma.pushSubscription.findMany({
|
||||||
|
where: { userId: notification.userId },
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
endpoint: true,
|
||||||
|
p256dh: true,
|
||||||
|
auth: true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
if (subscriptions.length === 0) {
|
||||||
|
logger.info(`No push subscriptions found for user ${notification.user.name}`)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const notificationData = {
|
||||||
|
title: makeNotificationTitle(notification, notification.user),
|
||||||
|
body: makeNotificationContent(notification) ?? undefined,
|
||||||
|
url: makeNotificationLink(notification, SITE_URL) ?? undefined,
|
||||||
|
} satisfies NotificationData
|
||||||
|
|
||||||
|
const results = await Promise.allSettled(
|
||||||
|
subscriptions.map(async (subscription) => {
|
||||||
|
const result = await sendPushNotification(
|
||||||
|
{
|
||||||
|
endpoint: subscription.endpoint,
|
||||||
|
keys: {
|
||||||
|
p256dh: subscription.p256dh,
|
||||||
|
auth: subscription.auth,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
notificationData
|
||||||
|
)
|
||||||
|
|
||||||
|
// Remove invalid subscriptions
|
||||||
|
if (result.error && (result.error.statusCode === 410 || result.error.statusCode === 404)) {
|
||||||
|
await prisma.pushSubscription.delete({ where: { id: subscription.id } })
|
||||||
|
logger.info(`Removed invalid subscription for user ${notification.user.name}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
return result.success
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
|
const successCount = results.filter((r) => r.status === 'fulfilled' && r.value).length
|
||||||
|
const failureCount = results.filter((r) => !(r.status === 'fulfilled' && r.value)).length
|
||||||
|
|
||||||
|
logger.info(
|
||||||
|
`Push notification sent for notification ${String(notificationId)} to user ${notification.user.name}: ${String(successCount)} successful, ${String(failureCount)} failed`
|
||||||
|
)
|
||||||
|
} catch (error) {
|
||||||
|
logger.error(`Error processing notification ${String(notificationId)}: ${getErrorMessage(error)}`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function postgresListener(): AstroIntegration {
|
||||||
|
return {
|
||||||
|
name: 'postgres-listener',
|
||||||
|
hooks: {
|
||||||
|
'astro:server:start': async (options) => {
|
||||||
|
const logger = options.logger.fork(INTEGRATION_NAME)
|
||||||
|
|
||||||
|
try {
|
||||||
|
logger.info('Starting PostgreSQL notification listener...')
|
||||||
|
|
||||||
|
pgClient = new Client({ connectionString: DATABASE_URL })
|
||||||
|
|
||||||
|
await pgClient.connect()
|
||||||
|
logger.info('Connected to PostgreSQL for notifications')
|
||||||
|
|
||||||
|
await pgClient.query('LISTEN notification_created')
|
||||||
|
logger.info('Listening for notification_created events')
|
||||||
|
|
||||||
|
pgClient.on('notification', (msg) => {
|
||||||
|
if (msg.channel === 'notification_created') {
|
||||||
|
const payload = zodParseJSON(z.object({ id: z.number().int().positive() }), msg.payload)
|
||||||
|
if (!payload) {
|
||||||
|
logger.warn(`Invalid notification ID in payload: ${String(msg.payload)}`)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE: Don't await to avoid blocking
|
||||||
|
void handleNotificationCreated(payload.id, options)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
pgClient.on('error', (error) => {
|
||||||
|
logger.error(`PostgreSQL client error: ${getErrorMessage(error)}`)
|
||||||
|
})
|
||||||
|
|
||||||
|
pgClient.on('end', () => {
|
||||||
|
logger.info('PostgreSQL client connection ended')
|
||||||
|
})
|
||||||
|
} catch (error) {
|
||||||
|
logger.error(`Failed to start PostgreSQL listener: ${getErrorMessage(error)}`)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
'astro:server:done': async ({ logger: originalLogger }) => {
|
||||||
|
const logger = originalLogger.fork(INTEGRATION_NAME)
|
||||||
|
|
||||||
|
if (pgClient) {
|
||||||
|
try {
|
||||||
|
logger.info('Stopping PostgreSQL notification listener...')
|
||||||
|
await pgClient.end()
|
||||||
|
pgClient = null
|
||||||
|
logger.info('PostgreSQL listener stopped')
|
||||||
|
} catch (error) {
|
||||||
|
logger.error(`Error stopping PostgreSQL listener: ${getErrorMessage(error)}`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getErrorMessage(error: unknown) {
|
||||||
|
if (error instanceof Error) {
|
||||||
|
return error.message
|
||||||
|
}
|
||||||
|
return String(error)
|
||||||
|
}
|
||||||
14
web/src/lib/serverEnvVariables.ts
Normal file
14
web/src/lib/serverEnvVariables.ts
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
import { loadEnv } from 'vite'
|
||||||
|
|
||||||
|
/** Only use when you can't import the variables from `astro:env/server` */
|
||||||
|
// @ts-expect-error process.env actually exists
|
||||||
|
const untypedServerEnvVariables = loadEnv(process.env.NODE_ENV, process.cwd(), '')
|
||||||
|
|
||||||
|
/** Only use when you can't import the variables from `astro:env/server` */
|
||||||
|
export function getServerEnvVariable<T extends keyof typeof untypedServerEnvVariables>(
|
||||||
|
name: T
|
||||||
|
): NonNullable<(typeof untypedServerEnvVariables)[T]> {
|
||||||
|
const value = untypedServerEnvVariables[name]
|
||||||
|
if (!value) throw new Error(`${name} environment variable is not set`)
|
||||||
|
return value
|
||||||
|
}
|
||||||
59
web/src/lib/webPush.ts
Normal file
59
web/src/lib/webPush.ts
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
/* eslint-disable import/no-named-as-default-member */
|
||||||
|
import webpush, { WebPushError } from 'web-push'
|
||||||
|
|
||||||
|
import { getServerEnvVariable } from './serverEnvVariables'
|
||||||
|
|
||||||
|
const VAPID_PUBLIC_KEY = getServerEnvVariable('VAPID_PUBLIC_KEY')
|
||||||
|
const VAPID_PRIVATE_KEY = getServerEnvVariable('VAPID_PRIVATE_KEY')
|
||||||
|
const VAPID_SUBJECT = getServerEnvVariable('VAPID_SUBJECT')
|
||||||
|
|
||||||
|
// Configure VAPID keys
|
||||||
|
webpush.setVapidDetails(VAPID_SUBJECT, VAPID_PUBLIC_KEY, VAPID_PRIVATE_KEY)
|
||||||
|
|
||||||
|
export { webpush }
|
||||||
|
|
||||||
|
export type NotificationData = {
|
||||||
|
title: string
|
||||||
|
body?: string
|
||||||
|
icon?: string
|
||||||
|
badge?: string
|
||||||
|
url?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function sendPushNotification(
|
||||||
|
subscription: {
|
||||||
|
endpoint: string
|
||||||
|
keys: {
|
||||||
|
p256dh: string
|
||||||
|
auth: string
|
||||||
|
}
|
||||||
|
},
|
||||||
|
data: NotificationData
|
||||||
|
) {
|
||||||
|
try {
|
||||||
|
const result = await webpush.sendNotification(
|
||||||
|
subscription,
|
||||||
|
JSON.stringify({
|
||||||
|
title: data.title,
|
||||||
|
options: {
|
||||||
|
body: data.body,
|
||||||
|
icon: data.icon ?? '/favicon.svg',
|
||||||
|
badge: data.badge ?? '/favicon.svg',
|
||||||
|
data: {
|
||||||
|
url: data.url,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
{
|
||||||
|
TTL: 24 * 60 * 60, // 24 hours
|
||||||
|
}
|
||||||
|
)
|
||||||
|
return { success: true, result } as const
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error sending push notification:', error)
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
error: error instanceof WebPushError ? error : undefined,
|
||||||
|
} as const
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -13,17 +13,44 @@ const addZodPipe = (schema: ZodTypeAny, zodPipe?: ZodTypeAny) => {
|
|||||||
export const zodCohercedNumber = (zodPipe?: ZodTypeAny) =>
|
export const zodCohercedNumber = (zodPipe?: ZodTypeAny) =>
|
||||||
addZodPipe(z.number().or(z.string().nonempty()), zodPipe)
|
addZodPipe(z.number().or(z.string().nonempty()), zodPipe)
|
||||||
|
|
||||||
|
const cleanUrl = (input: unknown) => {
|
||||||
|
if (typeof input !== 'string') return input
|
||||||
|
const cleanInput = input.trim().replace(/\/$/, '')
|
||||||
|
return !/^\w+:\/\//i.test(cleanInput) ? `https://${cleanInput}` : cleanInput
|
||||||
|
}
|
||||||
|
|
||||||
export const zodUrlOptionalProtocol = z.preprocess(
|
export const zodUrlOptionalProtocol = z.preprocess(
|
||||||
(input) => {
|
cleanUrl,
|
||||||
if (typeof input !== 'string') return input
|
|
||||||
const cleanInput = input.trim().replace(/\/$/, '')
|
|
||||||
return !/^\w+:\/\//i.test(cleanInput) ? `https://${cleanInput}` : cleanInput
|
|
||||||
},
|
|
||||||
z.string().refine((value) => /^(https?):\/\/(?=.*\.[a-z0-9]{2,})[^\s$.?#].[^\s]*$/i.test(value), {
|
z.string().refine((value) => /^(https?):\/\/(?=.*\.[a-z0-9]{2,})[^\s$.?#].[^\s]*$/i.test(value), {
|
||||||
message: 'Invalid URL',
|
message: 'Invalid URL',
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
|
|
||||||
|
export const zodContactMethod = z.preprocess(
|
||||||
|
(input) => {
|
||||||
|
if (typeof input !== 'string') return input
|
||||||
|
const cleanInput = input.trim()
|
||||||
|
|
||||||
|
if (/^([\d\s+\-_/()[\]*#.,]|ext|x){7,}$/i.test(cleanInput)) return `tel:${cleanInput}`
|
||||||
|
|
||||||
|
if (/^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$/.test(cleanInput)) return `mailto:${cleanInput}`
|
||||||
|
|
||||||
|
return cleanUrl(cleanInput)
|
||||||
|
},
|
||||||
|
z
|
||||||
|
.string()
|
||||||
|
.trim()
|
||||||
|
.refine(
|
||||||
|
(value) =>
|
||||||
|
/^((https?):\/\/(?=.*\.[a-z0-9]{2,})[^\s$.?#].[^\s]|([\d\s+\-_/()[\]*#.,]|ext|x){7,}|[0-9\s+-_\\/()[\]*#.]|[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,})*$/i.test(
|
||||||
|
value
|
||||||
|
),
|
||||||
|
{
|
||||||
|
message: 'Invalid contact method',
|
||||||
|
}
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
const stringToArrayFactory = (delimiter: RegExp | string = ',') => {
|
const stringToArrayFactory = (delimiter: RegExp | string = ',') => {
|
||||||
return <T>(input: T) =>
|
return <T>(input: T) =>
|
||||||
typeof input !== 'string'
|
typeof input !== 'string'
|
||||||
@@ -34,6 +61,11 @@ const stringToArrayFactory = (delimiter: RegExp | string = ',') => {
|
|||||||
.filter((item) => item !== '')
|
.filter((item) => item !== '')
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const stringListOfSlugsSchemaRequired = z.preprocess(
|
||||||
|
stringToArrayFactory(/[\s,\n]+/),
|
||||||
|
z.array(z.string().regex(/^[a-z0-9-_A-Z]+$/)).min(1)
|
||||||
|
)
|
||||||
|
|
||||||
export const stringListOfUrlsSchema = z.preprocess(
|
export const stringListOfUrlsSchema = z.preprocess(
|
||||||
stringToArrayFactory(/[\s,\n]+/),
|
stringToArrayFactory(/[\s,\n]+/),
|
||||||
z.array(zodUrlOptionalProtocol).default([])
|
z.array(zodUrlOptionalProtocol).default([])
|
||||||
@@ -44,6 +76,11 @@ export const stringListOfUrlsSchemaRequired = z.preprocess(
|
|||||||
z.array(zodUrlOptionalProtocol).min(1)
|
z.array(zodUrlOptionalProtocol).min(1)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
export const stringListOfContactMethodsSchema = z.preprocess(
|
||||||
|
stringToArrayFactory(/[\s,\n]+/),
|
||||||
|
z.array(zodContactMethod).default([])
|
||||||
|
)
|
||||||
|
|
||||||
export const MAX_IMAGE_SIZE = 5 * 1024 * 1024 // 5MB
|
export const MAX_IMAGE_SIZE = 5 * 1024 * 1024 // 5MB
|
||||||
|
|
||||||
export const ACCEPTED_IMAGE_TYPES = [
|
export const ACCEPTED_IMAGE_TYPES = [
|
||||||
|
|||||||
@@ -271,6 +271,7 @@ if (toggleResult?.error) {
|
|||||||
label: type.label,
|
label: type.label,
|
||||||
value: type.value,
|
value: type.value,
|
||||||
icon: type.icon,
|
icon: type.icon,
|
||||||
|
noTransitionPersist: false,
|
||||||
}))}
|
}))}
|
||||||
cardSize="sm"
|
cardSize="sm"
|
||||||
required
|
required
|
||||||
@@ -305,8 +306,8 @@ if (toggleResult?.error) {
|
|||||||
label="Status"
|
label="Status"
|
||||||
error={createInputErrors.isActive}
|
error={createInputErrors.isActive}
|
||||||
options={[
|
options={[
|
||||||
{ label: 'Active', value: 'true' },
|
{ label: 'Active', value: 'true', noTransitionPersist: true },
|
||||||
{ label: 'Inactive', value: 'false' },
|
{ label: 'Inactive', value: 'false', noTransitionPersist: true },
|
||||||
]}
|
]}
|
||||||
selectedValue={newAnnouncement.isActive ? 'true' : 'false'}
|
selectedValue={newAnnouncement.isActive ? 'true' : 'false'}
|
||||||
cardSize="sm"
|
cardSize="sm"
|
||||||
@@ -627,6 +628,7 @@ if (toggleResult?.error) {
|
|||||||
label: type.label,
|
label: type.label,
|
||||||
value: type.value,
|
value: type.value,
|
||||||
icon: type.icon,
|
icon: type.icon,
|
||||||
|
noTransitionPersist: true,
|
||||||
}))}
|
}))}
|
||||||
cardSize="sm"
|
cardSize="sm"
|
||||||
required
|
required
|
||||||
@@ -659,8 +661,8 @@ if (toggleResult?.error) {
|
|||||||
name="isActive"
|
name="isActive"
|
||||||
label="Status"
|
label="Status"
|
||||||
options={[
|
options={[
|
||||||
{ label: 'Active', value: 'true' },
|
{ label: 'Active', value: 'true', noTransitionPersist: true },
|
||||||
{ label: 'Inactive', value: 'false' },
|
{ label: 'Inactive', value: 'false', noTransitionPersist: true },
|
||||||
]}
|
]}
|
||||||
selectedValue={announcement.isActive ? 'true' : 'false'}
|
selectedValue={announcement.isActive ? 'true' : 'false'}
|
||||||
cardSize="sm"
|
cardSize="sm"
|
||||||
|
|||||||
@@ -65,6 +65,14 @@ const adminLinks: AdminLink[] = [
|
|||||||
base: 'text-pink-300',
|
base: 'text-pink-300',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
icon: 'ri:notification-3-line',
|
||||||
|
title: 'Notifications',
|
||||||
|
href: '/admin/notifications',
|
||||||
|
classNames: {
|
||||||
|
base: 'text-indigo-300',
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
icon: 'ri:rocket-2-line',
|
icon: 'ri:rocket-2-line',
|
||||||
title: 'Releases',
|
title: 'Releases',
|
||||||
|
|||||||
155
web/src/pages/admin/notifications.astro
Normal file
155
web/src/pages/admin/notifications.astro
Normal file
@@ -0,0 +1,155 @@
|
|||||||
|
---
|
||||||
|
import { Icon } from 'astro-icon/components'
|
||||||
|
import { actions, isInputError } from 'astro:actions'
|
||||||
|
import { groupBy, round, uniq } from 'lodash-es'
|
||||||
|
|
||||||
|
import InputSubmitButton from '../../components/InputSubmitButton.astro'
|
||||||
|
import InputText from '../../components/InputText.astro'
|
||||||
|
import InputTextArea from '../../components/InputTextArea.astro'
|
||||||
|
import MiniLayout from '../../layouts/MiniLayout.astro'
|
||||||
|
import { cn } from '../../lib/cn'
|
||||||
|
import { prisma } from '../../lib/prisma'
|
||||||
|
|
||||||
|
// Check if user is admin
|
||||||
|
if (!Astro.locals.user?.admin) {
|
||||||
|
return Astro.redirect('/access-denied')
|
||||||
|
}
|
||||||
|
|
||||||
|
const testResult = Astro.getActionResult(actions.admin.notification.webPush.test)
|
||||||
|
const testInputErrors = isInputError(testResult?.error) ? testResult.error.fields : {}
|
||||||
|
|
||||||
|
Astro.locals.banners.addIfSuccess(testResult, (data) => data.message)
|
||||||
|
|
||||||
|
const subscriptions = await Astro.locals.banners.try(
|
||||||
|
'Error while fetching subscriptions by user',
|
||||||
|
() =>
|
||||||
|
prisma.pushSubscription.findMany({
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
user: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
name: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
[] as []
|
||||||
|
)
|
||||||
|
const totalSubscriptions = subscriptions.length
|
||||||
|
const subscriptionsByUser = groupBy(subscriptions, 'user.id')
|
||||||
|
|
||||||
|
const totalUsers = Object.keys(subscriptionsByUser).length
|
||||||
|
|
||||||
|
const adminUsers = await prisma.user.findMany({
|
||||||
|
where: {
|
||||||
|
admin: true,
|
||||||
|
},
|
||||||
|
select: {
|
||||||
|
name: true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const stats = [
|
||||||
|
{
|
||||||
|
icon: 'ri:notification-4-line',
|
||||||
|
iconClass: 'text-blue-400',
|
||||||
|
title: 'Total Subscriptions',
|
||||||
|
value: totalSubscriptions.toLocaleString(),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
icon: 'ri:user-3-line',
|
||||||
|
iconClass: 'text-green-400',
|
||||||
|
title: 'Subscribed Users',
|
||||||
|
value: totalUsers.toLocaleString(),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
icon: 'ri:smartphone-line',
|
||||||
|
iconClass: 'text-purple-400',
|
||||||
|
title: 'Avg Devices/User',
|
||||||
|
value: (totalUsers > 0 ? round(totalSubscriptions / totalUsers, 1) : 0).toLocaleString(),
|
||||||
|
},
|
||||||
|
] satisfies {
|
||||||
|
icon: string
|
||||||
|
iconClass: string
|
||||||
|
title: string
|
||||||
|
value: string
|
||||||
|
}[]
|
||||||
|
---
|
||||||
|
|
||||||
|
<MiniLayout
|
||||||
|
pageTitle="Push notifications"
|
||||||
|
description="Send test notifications"
|
||||||
|
layoutHeader={{
|
||||||
|
icon: 'ri:notification-3-line',
|
||||||
|
title: 'Push notifications',
|
||||||
|
subtitle: 'Send test notifications',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div class="mb-6 grid gap-4 sm:grid-cols-3">
|
||||||
|
{
|
||||||
|
stats.map((stat) => (
|
||||||
|
<div class="flex flex-col items-center gap-2 text-center">
|
||||||
|
<div class="flex items-end gap-1">
|
||||||
|
<span class="text-5xl leading-[0.8] font-bold text-white">{stat.value}</span>
|
||||||
|
<Icon name={stat.icon} class={cn('size-5 shrink-0', stat.iconClass)} />
|
||||||
|
</div>
|
||||||
|
<span class="text-day-200 flex grow flex-col justify-center text-sm leading-none font-medium text-balance">
|
||||||
|
{stat.title}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
))
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h2 class="text-center text-lg font-semibold text-white">Send Test Notification</h2>
|
||||||
|
|
||||||
|
<form method="POST" action={actions.admin.notification.webPush.test} class="space-y-4">
|
||||||
|
<InputTextArea
|
||||||
|
label="Users"
|
||||||
|
name="userNames"
|
||||||
|
inputProps={{
|
||||||
|
placeholder: 'john-doe, jane-doe',
|
||||||
|
class: 'leading-tight min-h-24',
|
||||||
|
}}
|
||||||
|
value={uniq([Astro.locals.user, ...adminUsers].map((user) => user.name)).join('\n')}
|
||||||
|
description={[
|
||||||
|
'- Comma-separated list of user names.',
|
||||||
|
'- Minimum 1 user name.',
|
||||||
|
'- By default, all admin users are selected.',
|
||||||
|
].join('\n')}
|
||||||
|
error={testInputErrors.userNames}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<InputText
|
||||||
|
label="Title"
|
||||||
|
name="title"
|
||||||
|
inputProps={{
|
||||||
|
value: 'Test Notification',
|
||||||
|
required: true,
|
||||||
|
}}
|
||||||
|
error={testInputErrors.title}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<InputTextArea
|
||||||
|
label="Body"
|
||||||
|
name="body"
|
||||||
|
inputProps={{
|
||||||
|
value: 'This is a test push notification from KYCNot.me',
|
||||||
|
}}
|
||||||
|
error={testInputErrors.body}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<InputText
|
||||||
|
label="Action URL"
|
||||||
|
name="url"
|
||||||
|
inputProps={{
|
||||||
|
placeholder: 'https://example.com/path',
|
||||||
|
}}
|
||||||
|
description="URL to open when the notification is clicked"
|
||||||
|
error={testInputErrors.url}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<InputSubmitButton label="Send" icon="ri:send-plane-line" hideCancel color="danger" />
|
||||||
|
</form>
|
||||||
|
</MiniLayout>
|
||||||
@@ -23,9 +23,10 @@ import Tooltip from '../../../../components/Tooltip.astro'
|
|||||||
import UserBadge from '../../../../components/UserBadge.astro'
|
import UserBadge from '../../../../components/UserBadge.astro'
|
||||||
import { getAttributeCategoryInfo } from '../../../../constants/attributeCategories'
|
import { getAttributeCategoryInfo } from '../../../../constants/attributeCategories'
|
||||||
import { getAttributeTypeInfo } from '../../../../constants/attributeTypes'
|
import { getAttributeTypeInfo } from '../../../../constants/attributeTypes'
|
||||||
import { formatContactMethod } from '../../../../constants/contactMethods'
|
import { contactMethodUrlTypes, formatContactMethod } from '../../../../constants/contactMethods'
|
||||||
import { currencies } from '../../../../constants/currencies'
|
import { currencies } from '../../../../constants/currencies'
|
||||||
import { eventTypes, getEventTypeInfo } from '../../../../constants/eventTypes'
|
import { eventTypes, getEventTypeInfo } from '../../../../constants/eventTypes'
|
||||||
|
import { kycLevelClarifications } from '../../../../constants/kycLevelClarifications'
|
||||||
import { kycLevels } from '../../../../constants/kycLevels'
|
import { kycLevels } from '../../../../constants/kycLevels'
|
||||||
import { serviceVisibilities } from '../../../../constants/serviceVisibility'
|
import { serviceVisibilities } from '../../../../constants/serviceVisibility'
|
||||||
import { verificationStatuses } from '../../../../constants/verificationStatus'
|
import { verificationStatuses } from '../../../../constants/verificationStatus'
|
||||||
@@ -34,6 +35,8 @@ import {
|
|||||||
verificationStepStatuses,
|
verificationStepStatuses,
|
||||||
} from '../../../../constants/verificationStepStatus'
|
} from '../../../../constants/verificationStepStatus'
|
||||||
import BaseLayout from '../../../../layouts/BaseLayout.astro'
|
import BaseLayout from '../../../../layouts/BaseLayout.astro'
|
||||||
|
import { DEPLOYMENT_MODE } from '../../../../lib/envVariables'
|
||||||
|
import { listFiles } from '../../../../lib/fileStorage'
|
||||||
import { makeAdminApiCallInfo } from '../../../../lib/makeAdminApiCallInfo'
|
import { makeAdminApiCallInfo } from '../../../../lib/makeAdminApiCallInfo'
|
||||||
import { pluralize } from '../../../../lib/pluralize'
|
import { pluralize } from '../../../../lib/pluralize'
|
||||||
import { prisma } from '../../../../lib/prisma'
|
import { prisma } from '../../../../lib/prisma'
|
||||||
@@ -85,9 +88,36 @@ const internalNoteInputErrors = isInputError(internalNoteCreateResult?.error)
|
|||||||
? internalNoteCreateResult.error.fields
|
? internalNoteCreateResult.error.fields
|
||||||
: {}
|
: {}
|
||||||
|
|
||||||
|
const contactMethodUpdateResult = Astro.getActionResult(actions.admin.service.contactMethod.update)
|
||||||
|
Astro.locals.banners.addIfSuccess(contactMethodUpdateResult, 'Contact method updated successfully')
|
||||||
|
const contactMethodUpdateInputErrors = isInputError(contactMethodUpdateResult?.error)
|
||||||
|
? contactMethodUpdateResult.error.fields
|
||||||
|
: {}
|
||||||
|
|
||||||
|
const contactMethodAddResult = Astro.getActionResult(actions.admin.service.contactMethod.add)
|
||||||
|
Astro.locals.banners.addIfSuccess(contactMethodAddResult, 'Contact method added successfully')
|
||||||
|
const contactMethodAddInputErrors = isInputError(contactMethodAddResult?.error)
|
||||||
|
? contactMethodAddResult.error.fields
|
||||||
|
: {}
|
||||||
|
|
||||||
const internalNoteDeleteResult = Astro.getActionResult(actions.admin.service.internalNote.delete)
|
const internalNoteDeleteResult = Astro.getActionResult(actions.admin.service.internalNote.delete)
|
||||||
Astro.locals.banners.addIfSuccess(internalNoteDeleteResult, 'Internal note deleted successfully')
|
Astro.locals.banners.addIfSuccess(internalNoteDeleteResult, 'Internal note deleted successfully')
|
||||||
|
|
||||||
|
const evidenceImageAddResult = Astro.getActionResult(actions.admin.service.evidenceImage.add)
|
||||||
|
if (evidenceImageAddResult?.data?.imageUrl) {
|
||||||
|
Astro.locals.banners.add({
|
||||||
|
uiMessage: 'Evidence image added successfully',
|
||||||
|
type: 'success',
|
||||||
|
origin: 'action',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
const evidenceImageAddInputErrors = isInputError(evidenceImageAddResult?.error)
|
||||||
|
? evidenceImageAddResult.error.fields
|
||||||
|
: {}
|
||||||
|
|
||||||
|
const evidenceImageDeleteResult = Astro.getActionResult(actions.admin.service.evidenceImage.delete)
|
||||||
|
Astro.locals.banners.addIfSuccess(evidenceImageDeleteResult, 'Evidence image deleted successfully')
|
||||||
|
|
||||||
const [service, categories, attributes] = await Astro.locals.banners.tryMany([
|
const [service, categories, attributes] = await Astro.locals.banners.tryMany([
|
||||||
[
|
[
|
||||||
'Error fetching service',
|
'Error fetching service',
|
||||||
@@ -182,7 +212,27 @@ const [service, categories, attributes] = await Astro.locals.banners.tryMany([
|
|||||||
],
|
],
|
||||||
])
|
])
|
||||||
|
|
||||||
if (!service) return Astro.rewrite('/404')
|
if (!service) {
|
||||||
|
try {
|
||||||
|
const serviceWithOldSlug = await prisma.service.findFirst({
|
||||||
|
where: { previousSlugs: { has: slug } },
|
||||||
|
select: { slug: true },
|
||||||
|
})
|
||||||
|
if (serviceWithOldSlug) {
|
||||||
|
return Astro.redirect(`/admin/services/${serviceWithOldSlug.slug}/edit`, 301)
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error)
|
||||||
|
}
|
||||||
|
|
||||||
|
return Astro.rewrite('/404')
|
||||||
|
}
|
||||||
|
|
||||||
|
const evidenceImageUrls = await Astro.locals.banners.try(
|
||||||
|
'Error listing evidence files',
|
||||||
|
() => listFiles(`evidence/${service.slug}`),
|
||||||
|
[] as string[]
|
||||||
|
)
|
||||||
|
|
||||||
const apiCalls = await Astro.locals.banners.try(
|
const apiCalls = await Astro.locals.banners.try(
|
||||||
'Error fetching api calls',
|
'Error fetching api calls',
|
||||||
@@ -263,7 +313,9 @@ const apiCalls = await Astro.locals.banners.try(
|
|||||||
|
|
||||||
<InputText
|
<InputText
|
||||||
label="Slug"
|
label="Slug"
|
||||||
description="Auto-generated if empty"
|
description={`Auto-generated if empty. ${
|
||||||
|
service.previousSlugs.length > 0 ? `Old slugs: ${service.previousSlugs.join(', ')}` : ''
|
||||||
|
}`}
|
||||||
name="slug"
|
name="slug"
|
||||||
inputProps={{
|
inputProps={{
|
||||||
value: service.slug,
|
value: service.slug,
|
||||||
@@ -389,6 +441,7 @@ const apiCalls = await Astro.locals.banners.try(
|
|||||||
value: kycLevel.id.toString(),
|
value: kycLevel.id.toString(),
|
||||||
icon: kycLevel.icon,
|
icon: kycLevel.icon,
|
||||||
description: kycLevel.description,
|
description: kycLevel.description,
|
||||||
|
noTransitionPersist: true,
|
||||||
}))}
|
}))}
|
||||||
selectedValue={service.kycLevel.toString()}
|
selectedValue={service.kycLevel.toString()}
|
||||||
iconSize="md"
|
iconSize="md"
|
||||||
@@ -397,6 +450,22 @@ const apiCalls = await Astro.locals.banners.try(
|
|||||||
class="[&>div]:grid-cols-2 [&>div]:[--card-min-size:16rem]"
|
class="[&>div]:grid-cols-2 [&>div]:[--card-min-size:16rem]"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<InputCardGroup
|
||||||
|
name="kycLevelClarification"
|
||||||
|
label="KYC Level Clarification"
|
||||||
|
options={kycLevelClarifications.map((clarification) => ({
|
||||||
|
label: clarification.label,
|
||||||
|
value: clarification.value,
|
||||||
|
icon: clarification.icon,
|
||||||
|
description: clarification.description,
|
||||||
|
noTransitionPersist: true,
|
||||||
|
}))}
|
||||||
|
selectedValue={service.kycLevelClarification}
|
||||||
|
iconSize="sm"
|
||||||
|
cardSize="sm"
|
||||||
|
error={serviceInputErrors.kycLevelClarification}
|
||||||
|
/>
|
||||||
|
|
||||||
<InputCardGroup
|
<InputCardGroup
|
||||||
name="verificationStatus"
|
name="verificationStatus"
|
||||||
label="Verification Status"
|
label="Verification Status"
|
||||||
@@ -406,6 +475,7 @@ const apiCalls = await Astro.locals.banners.try(
|
|||||||
icon: status.icon,
|
icon: status.icon,
|
||||||
iconClass: status.classNames.icon,
|
iconClass: status.classNames.icon,
|
||||||
description: status.description,
|
description: status.description,
|
||||||
|
noTransitionPersist: true,
|
||||||
}))}
|
}))}
|
||||||
selectedValue={service.verificationStatus}
|
selectedValue={service.verificationStatus}
|
||||||
error={serviceInputErrors.verificationStatus}
|
error={serviceInputErrors.verificationStatus}
|
||||||
@@ -421,6 +491,7 @@ const apiCalls = await Astro.locals.banners.try(
|
|||||||
label: currency.name,
|
label: currency.name,
|
||||||
value: currency.id,
|
value: currency.id,
|
||||||
icon: currency.icon,
|
icon: currency.icon,
|
||||||
|
noTransitionPersist: true,
|
||||||
}))}
|
}))}
|
||||||
selectedValue={service.acceptedCurrencies}
|
selectedValue={service.acceptedCurrencies}
|
||||||
error={serviceInputErrors.acceptedCurrencies}
|
error={serviceInputErrors.acceptedCurrencies}
|
||||||
@@ -461,6 +532,7 @@ const apiCalls = await Astro.locals.banners.try(
|
|||||||
icon: visibility.icon,
|
icon: visibility.icon,
|
||||||
iconClass: visibility.iconClass,
|
iconClass: visibility.iconClass,
|
||||||
description: visibility.description,
|
description: visibility.description,
|
||||||
|
noTransitionPersist: true,
|
||||||
}))}
|
}))}
|
||||||
selectedValue={service.serviceVisibility}
|
selectedValue={service.serviceVisibility}
|
||||||
error={serviceInputErrors.serviceVisibility}
|
error={serviceInputErrors.serviceVisibility}
|
||||||
@@ -1075,6 +1147,7 @@ const apiCalls = await Astro.locals.banners.try(
|
|||||||
value: method.label,
|
value: method.label,
|
||||||
placeholder: contactMethodInfo.formattedValue,
|
placeholder: contactMethodInfo.formattedValue,
|
||||||
}}
|
}}
|
||||||
|
error={contactMethodUpdateInputErrors.label}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<InputText
|
<InputText
|
||||||
@@ -1084,6 +1157,7 @@ const apiCalls = await Astro.locals.banners.try(
|
|||||||
value: method.value,
|
value: method.value,
|
||||||
placeholder: 'e.g., mailto:contact@example.com or https://t.me/example',
|
placeholder: 'e.g., mailto:contact@example.com or https://t.me/example',
|
||||||
}}
|
}}
|
||||||
|
error={contactMethodUpdateInputErrors.value}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<InputSubmitButton label="Update" icon="ri:save-line" hideCancel />
|
<InputSubmitButton label="Update" icon="ri:save-line" hideCancel />
|
||||||
@@ -1104,12 +1178,13 @@ const apiCalls = await Astro.locals.banners.try(
|
|||||||
|
|
||||||
<InputText
|
<InputText
|
||||||
label="Value"
|
label="Value"
|
||||||
description="With protocol (e.g., `mailto:contact@example.com` or `https://t.me/example`)"
|
description={`Accepts: ${contactMethodUrlTypes.map((type) => type.labelPlural).join(', ')}`}
|
||||||
name="value"
|
name="value"
|
||||||
inputProps={{
|
inputProps={{
|
||||||
required: true,
|
required: true,
|
||||||
placeholder: 'mailto:contact@example.com',
|
placeholder: 'contact@example.com',
|
||||||
}}
|
}}
|
||||||
|
error={contactMethodAddInputErrors.value}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<InputSubmitButton label="Add" icon="ri:add-line" hideCancel />
|
<InputSubmitButton label="Add" icon="ri:add-line" hideCancel />
|
||||||
@@ -1118,6 +1193,24 @@ const apiCalls = await Astro.locals.banners.try(
|
|||||||
</FormSection>
|
</FormSection>
|
||||||
|
|
||||||
<FormSection title="API">
|
<FormSection title="API">
|
||||||
|
{
|
||||||
|
DEPLOYMENT_MODE === 'staging' && (
|
||||||
|
<p class="rounded-lg bg-red-900/30 p-4 text-sm text-red-200">
|
||||||
|
<Icon name="ri:alert-line" class="inline-block size-4 align-[-0.2em] text-red-400" />
|
||||||
|
This endpoints section doesn't work in PRE. Use curl commands instead.
|
||||||
|
</p>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
<div class="mb-6 flex justify-center">
|
||||||
|
<Button
|
||||||
|
as="a"
|
||||||
|
href="/docs/api"
|
||||||
|
icon="ri:book-open-line"
|
||||||
|
color="gray"
|
||||||
|
size="sm"
|
||||||
|
label=" Documentation"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
{
|
{
|
||||||
apiCalls.map((call) => (
|
apiCalls.map((call) => (
|
||||||
<FormSubSection title={`${call.method} ${call.path}`}>
|
<FormSubSection title={`${call.method} ${call.path}`}>
|
||||||
@@ -1130,5 +1223,73 @@ const apiCalls = await Astro.locals.banners.try(
|
|||||||
))
|
))
|
||||||
}
|
}
|
||||||
</FormSection>
|
</FormSection>
|
||||||
|
|
||||||
|
<FormSection title="Evidence Images" id="evidence-images">
|
||||||
|
<FormSubSection title="Existing Evidence Images">
|
||||||
|
{
|
||||||
|
evidenceImageUrls.length === 0 ? (
|
||||||
|
<p class="border-night-600 bg-night-800 text-day-300 rounded-xl border p-6 text-center">
|
||||||
|
No evidence images yet.
|
||||||
|
</p>
|
||||||
|
) : (
|
||||||
|
<div class="grid grid-cols-2 gap-4 sm:grid-cols-3 md:grid-cols-4">
|
||||||
|
{evidenceImageUrls.map((imageUrl: string) => (
|
||||||
|
<div class="border-night-600 bg-night-800 group relative rounded-md border p-2">
|
||||||
|
<MyPicture
|
||||||
|
src={imageUrl}
|
||||||
|
alt="Evidence image"
|
||||||
|
class="aspect-square w-full rounded object-cover"
|
||||||
|
width={200}
|
||||||
|
height={200}
|
||||||
|
/>
|
||||||
|
<form
|
||||||
|
method="POST"
|
||||||
|
action={actions.admin.service.evidenceImage.delete}
|
||||||
|
class="absolute top-1 right-1"
|
||||||
|
>
|
||||||
|
<input type="hidden" name="fileUrl" value={imageUrl} />
|
||||||
|
<Button
|
||||||
|
type="submit"
|
||||||
|
variant="faded"
|
||||||
|
color="danger"
|
||||||
|
size="sm"
|
||||||
|
icon="ri:delete-bin-line"
|
||||||
|
iconOnly
|
||||||
|
label="Delete Image"
|
||||||
|
class="opacity-0 transition-opacity group-hover:opacity-100"
|
||||||
|
/>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
readonly
|
||||||
|
value={``}
|
||||||
|
class="bg-night-700 text-day-200 mt-2 w-full cursor-text rounded border p-2 font-mono text-xs select-all"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
</FormSubSection>
|
||||||
|
<FormSubSection title="Add New Evidence Image">
|
||||||
|
<form
|
||||||
|
method="POST"
|
||||||
|
action={actions.admin.service.evidenceImage.add}
|
||||||
|
class="space-y-4"
|
||||||
|
enctype="multipart/form-data"
|
||||||
|
>
|
||||||
|
<input type="hidden" name="serviceId" value={service.id} />
|
||||||
|
<InputImageFile
|
||||||
|
label="Upload Image"
|
||||||
|
name="imageFile"
|
||||||
|
description="Upload an evidence image."
|
||||||
|
error={evidenceImageAddInputErrors.imageFile}
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
<InputSubmitButton label="Add Image" icon="ri:add-line" hideCancel />
|
||||||
|
</form>
|
||||||
|
</FormSubSection>
|
||||||
|
</FormSection>
|
||||||
</div>
|
</div>
|
||||||
</BaseLayout>
|
</BaseLayout>
|
||||||
|
|||||||
@@ -226,9 +226,24 @@ if (!user) return Astro.rewrite('/404')
|
|||||||
name="type"
|
name="type"
|
||||||
label="Type"
|
label="Type"
|
||||||
options={[
|
options={[
|
||||||
{ label: 'Admin', value: 'admin', icon: 'ri:shield-star-fill' },
|
{
|
||||||
{ label: 'Moderator', value: 'moderator', icon: 'ri:graduation-cap-fill' },
|
label: 'Admin',
|
||||||
{ label: 'Spammer', value: 'spammer', icon: 'ri:alert-fill' },
|
value: 'admin',
|
||||||
|
icon: 'ri:shield-star-fill',
|
||||||
|
noTransitionPersist: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Moderator',
|
||||||
|
value: 'moderator',
|
||||||
|
icon: 'ri:graduation-cap-fill',
|
||||||
|
noTransitionPersist: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Spammer',
|
||||||
|
value: 'spammer',
|
||||||
|
icon: 'ri:alert-fill',
|
||||||
|
noTransitionPersist: true,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
label: 'Verified',
|
label: 'Verified',
|
||||||
value: 'verified',
|
value: 'verified',
|
||||||
@@ -419,6 +434,7 @@ if (!user) return Astro.rewrite('/404')
|
|||||||
label: role.label,
|
label: role.label,
|
||||||
value: role.value,
|
value: role.value,
|
||||||
icon: role.icon,
|
icon: role.icon,
|
||||||
|
noTransitionPersist: true,
|
||||||
}))}
|
}))}
|
||||||
required
|
required
|
||||||
cardSize="sm"
|
cardSize="sm"
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import { SOURCE_CODE_URL } from 'astro:env/server'
|
|||||||
import { kycLevels } from '../../constants/kycLevels'
|
import { kycLevels } from '../../constants/kycLevels'
|
||||||
import { verificationStatuses } from '../../constants/verificationStatus'
|
import { verificationStatuses } from '../../constants/verificationStatus'
|
||||||
import { serviceVisibilities } from '../../constants/serviceVisibility'
|
import { serviceVisibilities } from '../../constants/serviceVisibility'
|
||||||
|
import { kycLevelClarifications } from '../../constants/kycLevelClarifications'
|
||||||
|
|
||||||
Access basic service data via our public API.
|
Access basic service data via our public API.
|
||||||
|
|
||||||
@@ -58,6 +59,12 @@ type ServiceResponse = {
|
|||||||
name: string
|
name: string
|
||||||
description: string
|
description: string
|
||||||
}
|
}
|
||||||
|
kycLevelClarification: 'NONE' | 'DEPENDS_ON_PARTNERS' | ...
|
||||||
|
kycLevelClarificationInfo: {
|
||||||
|
value: 'NONE' | 'DEPENDS_ON_PARTNERS' | ...
|
||||||
|
name: string
|
||||||
|
description: string
|
||||||
|
}
|
||||||
categories: {
|
categories: {
|
||||||
name: string
|
name: string
|
||||||
slug: string
|
slug: string
|
||||||
@@ -99,6 +106,16 @@ type ServiceResponse = {
|
|||||||
))}
|
))}
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
|
#### KYC Level Clarifications
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
{kycLevelClarifications.map((clarification) => (
|
||||||
|
<li key={clarification.value}>
|
||||||
|
<strong>{clarification.value}</strong>: {clarification.description}
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
|
||||||
### Examples
|
### Examples
|
||||||
|
|
||||||
#### Request
|
#### Request
|
||||||
@@ -131,6 +148,11 @@ curl -X QUERY https://kycnot.me/api/v1/service/get \
|
|||||||
"name": "Guaranteed no KYC",
|
"name": "Guaranteed no KYC",
|
||||||
"description": "Terms explicitly state KYC will never be requested."
|
"description": "Terms explicitly state KYC will never be requested."
|
||||||
},
|
},
|
||||||
|
"kycLevelClarification": "NONE",
|
||||||
|
"kycLevelClarificationInfo": {
|
||||||
|
"value": "NONE",
|
||||||
|
"description": "No clarification needed."
|
||||||
|
},
|
||||||
"categories": [
|
"categories": [
|
||||||
{
|
{
|
||||||
"name": "Exchange",
|
"name": "Exchange",
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import {
|
|||||||
getEventTypeInfo,
|
getEventTypeInfo,
|
||||||
getEventTypeInfoBySlug,
|
getEventTypeInfoBySlug,
|
||||||
} from '../constants/eventTypes'
|
} from '../constants/eventTypes'
|
||||||
|
import { getServiceVisibilityInfo } from '../constants/serviceVisibility'
|
||||||
import { getVerificationStatusInfo } from '../constants/verificationStatus'
|
import { getVerificationStatusInfo } from '../constants/verificationStatus'
|
||||||
import BaseLayout from '../layouts/BaseLayout.astro'
|
import BaseLayout from '../layouts/BaseLayout.astro'
|
||||||
import { cn } from '../lib/cn'
|
import { cn } from '../lib/cn'
|
||||||
@@ -44,6 +45,8 @@ const [services, [dbEvents, totalEvents]] = await Astro.locals.banners.tryMany([
|
|||||||
async () =>
|
async () =>
|
||||||
prisma.service.findMany({
|
prisma.service.findMany({
|
||||||
where: {
|
where: {
|
||||||
|
listedAt: { lte: new Date() },
|
||||||
|
serviceVisibility: { in: ['PUBLIC', 'ARCHIVED'] },
|
||||||
events: {
|
events: {
|
||||||
some: {
|
some: {
|
||||||
visible: true,
|
visible: true,
|
||||||
@@ -72,8 +75,15 @@ const [services, [dbEvents, totalEvents]] = await Astro.locals.banners.tryMany([
|
|||||||
createdAt: {
|
createdAt: {
|
||||||
lte: params.now,
|
lte: params.now,
|
||||||
},
|
},
|
||||||
...(params.service ? { service: { slug: params.service } } : {}),
|
service: {
|
||||||
...(params.type ? { type: getEventTypeInfoBySlug(params.type).id } : {}),
|
slug: params.service ?? undefined,
|
||||||
|
listedAt: params.service ? undefined : { lte: new Date() },
|
||||||
|
serviceVisibility: {
|
||||||
|
in: params.service ? ['PUBLIC', 'ARCHIVED', 'UNLISTED'] : ['PUBLIC', 'ARCHIVED'],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
type: params.type ? getEventTypeInfoBySlug(params.type).id : undefined,
|
||||||
|
|
||||||
...(params.from || params.to
|
...(params.from || params.to
|
||||||
? {
|
? {
|
||||||
OR: [
|
OR: [
|
||||||
@@ -105,6 +115,7 @@ const [services, [dbEvents, totalEvents]] = await Astro.locals.banners.tryMany([
|
|||||||
name: true,
|
name: true,
|
||||||
imageUrl: true,
|
imageUrl: true,
|
||||||
verificationStatus: true,
|
verificationStatus: true,
|
||||||
|
serviceVisibility: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -126,6 +137,7 @@ const events = orderBy(
|
|||||||
service: {
|
service: {
|
||||||
...event.service,
|
...event.service,
|
||||||
verificationStatusInfo: getVerificationStatusInfo(event.service.verificationStatus),
|
verificationStatusInfo: getVerificationStatusInfo(event.service.verificationStatus),
|
||||||
|
serviceVisibilityInfo: getServiceVisibilityInfo(event.service.serviceVisibility),
|
||||||
},
|
},
|
||||||
})),
|
})),
|
||||||
['actualEndedAt', 'startedAt'],
|
['actualEndedAt', 'startedAt'],
|
||||||
@@ -416,6 +428,16 @@ const createUrlWithoutFilter = (paramName: keyof typeof params) => {
|
|||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
{event.service.serviceVisibility === 'ARCHIVED' && (
|
||||||
|
<Icon
|
||||||
|
name={event.service.serviceVisibilityInfo.icon}
|
||||||
|
class={cn(
|
||||||
|
'ms-1 inline-block size-3 shrink-0',
|
||||||
|
event.service.serviceVisibilityInfo.iconClass
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
{event.source && (
|
{event.source && (
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
---
|
---
|
||||||
import { z } from 'astro:schema'
|
import { z } from 'astro:schema'
|
||||||
import { groupBy, omit, orderBy, uniq } from 'lodash-es'
|
import { groupBy, omit, orderBy, sortBy, uniq } from 'lodash-es'
|
||||||
import seedrandom from 'seedrandom'
|
import seedrandom from 'seedrandom'
|
||||||
|
|
||||||
import Button from '../components/Button.astro'
|
import Button from '../components/Button.astro'
|
||||||
@@ -182,6 +182,7 @@ const {
|
|||||||
'min-score': { if: 'default' },
|
'min-score': { if: 'default' },
|
||||||
'user-rating': { if: 'default' },
|
'user-rating': { if: 'default' },
|
||||||
'max-kyc': { if: 'default' },
|
'max-kyc': { if: 'default' },
|
||||||
|
'sort-seed': { if: 'another-is-unset', prop: 'page' },
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@@ -349,7 +350,8 @@ const [categories, [services, totalServices], countCommunityOnly, attributes] =
|
|||||||
const selectedSort = sortOptions.find((sort) => sort.value === filters.sort) ?? defaultSortOption
|
const selectedSort = sortOptions.find((sort) => sort.value === filters.sort) ?? defaultSortOption
|
||||||
|
|
||||||
const sortedServices = orderBy(
|
const sortedServices = orderBy(
|
||||||
unsortedServices,
|
// NOTE: We do a first sort by id to make the seeded sort deterministic
|
||||||
|
sortBy(unsortedServices, 'id'),
|
||||||
[
|
[
|
||||||
...(filters.q ? (['similarityScore'] as const) : ([] as const)),
|
...(filters.q ? (['similarityScore'] as const) : ([] as const)),
|
||||||
selectedSort.orderBy.key,
|
selectedSort.orderBy.key,
|
||||||
|
|||||||
13
web/src/pages/internal-api/[...catchAll].ts
Normal file
13
web/src/pages/internal-api/[...catchAll].ts
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
import type { APIRoute } from 'astro'
|
||||||
|
|
||||||
|
export const ALL: APIRoute = (context) => {
|
||||||
|
console.error('Endpoint not found', { url: context.url.href, method: context.request.method })
|
||||||
|
return new Response(
|
||||||
|
JSON.stringify({
|
||||||
|
error: 'Endpoint not found',
|
||||||
|
}),
|
||||||
|
{
|
||||||
|
status: 404,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
7
web/src/pages/internal-api/notifications/subscribe.ts
Normal file
7
web/src/pages/internal-api/notifications/subscribe.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
import { actions } from 'astro:actions'
|
||||||
|
|
||||||
|
import { makeEndpointFromAction } from '../../../lib/endpoints'
|
||||||
|
|
||||||
|
import type { APIRoute } from 'astro'
|
||||||
|
|
||||||
|
export const POST: APIRoute = makeEndpointFromAction(actions.notification.webPush.subscribe)
|
||||||
7
web/src/pages/internal-api/notifications/unsubscribe.ts
Normal file
7
web/src/pages/internal-api/notifications/unsubscribe.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
import { actions } from 'astro:actions'
|
||||||
|
|
||||||
|
import { makeEndpointFromAction } from '../../../lib/endpoints'
|
||||||
|
|
||||||
|
import type { APIRoute } from 'astro'
|
||||||
|
|
||||||
|
export const POST: APIRoute = makeEndpointFromAction(actions.notification.webPush.unsubscribe)
|
||||||
@@ -4,6 +4,7 @@ import { Icon } from 'astro-icon/components'
|
|||||||
import { actions } from 'astro:actions'
|
import { actions } from 'astro:actions'
|
||||||
|
|
||||||
import Button from '../components/Button.astro'
|
import Button from '../components/Button.astro'
|
||||||
|
import PushNotificationBanner from '../components/PushNotificationBanner.astro'
|
||||||
import TimeFormatted from '../components/TimeFormatted.astro'
|
import TimeFormatted from '../components/TimeFormatted.astro'
|
||||||
import Tooltip from '../components/Tooltip.astro'
|
import Tooltip from '../components/Tooltip.astro'
|
||||||
import { getNotificationTypeInfo } from '../constants/notificationTypes'
|
import { getNotificationTypeInfo } from '../constants/notificationTypes'
|
||||||
@@ -28,131 +29,146 @@ const { data: params } = zodParseQueryParamsStoringErrors(
|
|||||||
)
|
)
|
||||||
const skip = (params.page - 1) * PAGE_SIZE
|
const skip = (params.page - 1) * PAGE_SIZE
|
||||||
|
|
||||||
const [dbNotifications, notificationPreferences, totalNotifications] = await Astro.locals.banners.tryMany([
|
const [dbNotifications, notificationPreferences, totalNotifications, pushSubscriptions] =
|
||||||
[
|
await Astro.locals.banners.tryMany([
|
||||||
'Error while fetching notifications',
|
[
|
||||||
() =>
|
'Error while fetching notifications',
|
||||||
prisma.notification.findMany({
|
() =>
|
||||||
where: {
|
prisma.notification.findMany({
|
||||||
userId: user.id,
|
where: {
|
||||||
},
|
userId: user.id,
|
||||||
orderBy: {
|
},
|
||||||
createdAt: 'desc',
|
orderBy: {
|
||||||
},
|
createdAt: 'desc',
|
||||||
skip,
|
},
|
||||||
take: PAGE_SIZE,
|
skip,
|
||||||
select: {
|
take: PAGE_SIZE,
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
type: true,
|
||||||
|
createdAt: true,
|
||||||
|
read: true,
|
||||||
|
aboutAccountStatusChange: true,
|
||||||
|
aboutCommentStatusChange: true,
|
||||||
|
aboutServiceVerificationStatusChange: true,
|
||||||
|
aboutSuggestionStatusChange: true,
|
||||||
|
aboutComment: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
author: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
status: true,
|
||||||
|
content: true,
|
||||||
|
communityNote: true,
|
||||||
|
service: {
|
||||||
|
select: {
|
||||||
|
slug: true,
|
||||||
|
name: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
parent: {
|
||||||
|
select: {
|
||||||
|
author: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
aboutServiceSuggestionId: true,
|
||||||
|
aboutServiceSuggestion: {
|
||||||
|
select: {
|
||||||
|
status: true,
|
||||||
|
service: {
|
||||||
|
select: {
|
||||||
|
name: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
aboutServiceSuggestionMessage: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
content: true,
|
||||||
|
suggestion: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
service: {
|
||||||
|
select: {
|
||||||
|
name: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
aboutEvent: {
|
||||||
|
select: {
|
||||||
|
title: true,
|
||||||
|
type: true,
|
||||||
|
service: {
|
||||||
|
select: {
|
||||||
|
slug: true,
|
||||||
|
name: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
aboutService: {
|
||||||
|
select: {
|
||||||
|
slug: true,
|
||||||
|
name: true,
|
||||||
|
verificationStatus: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
aboutKarmaTransaction: {
|
||||||
|
select: {
|
||||||
|
points: true,
|
||||||
|
action: true,
|
||||||
|
description: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
[],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
'Error while fetching notification preferences',
|
||||||
|
() =>
|
||||||
|
getOrCreateNotificationPreferences(user.id, {
|
||||||
id: true,
|
id: true,
|
||||||
type: true,
|
enableOnMyCommentStatusChange: true,
|
||||||
createdAt: true,
|
enableAutowatchMyComments: true,
|
||||||
read: true,
|
enableNotifyPendingRepliesOnWatch: true,
|
||||||
aboutAccountStatusChange: true,
|
karmaNotificationThreshold: true,
|
||||||
aboutCommentStatusChange: true,
|
}),
|
||||||
aboutServiceVerificationStatusChange: true,
|
null,
|
||||||
aboutSuggestionStatusChange: true,
|
],
|
||||||
aboutComment: {
|
[
|
||||||
select: {
|
'Error while fetching total notifications',
|
||||||
id: true,
|
() => prisma.notification.count({ where: { userId: user.id } }),
|
||||||
author: {
|
0,
|
||||||
select: {
|
],
|
||||||
id: true,
|
[
|
||||||
},
|
'Error while fetching push subscriptions',
|
||||||
},
|
() =>
|
||||||
status: true,
|
prisma.pushSubscription.findMany({
|
||||||
content: true,
|
where: { userId: user.id },
|
||||||
communityNote: true,
|
select: {
|
||||||
service: {
|
endpoint: true,
|
||||||
select: {
|
userAgent: true,
|
||||||
slug: true,
|
|
||||||
name: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
parent: {
|
|
||||||
select: {
|
|
||||||
author: {
|
|
||||||
select: {
|
|
||||||
id: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
aboutServiceSuggestionId: true,
|
}),
|
||||||
aboutServiceSuggestion: {
|
[],
|
||||||
select: {
|
],
|
||||||
status: true,
|
])
|
||||||
service: {
|
|
||||||
select: {
|
if (!notificationPreferences) console.error('No notification preferences found')
|
||||||
name: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
aboutServiceSuggestionMessage: {
|
|
||||||
select: {
|
|
||||||
id: true,
|
|
||||||
content: true,
|
|
||||||
suggestion: {
|
|
||||||
select: {
|
|
||||||
id: true,
|
|
||||||
service: {
|
|
||||||
select: {
|
|
||||||
name: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
aboutEvent: {
|
|
||||||
select: {
|
|
||||||
title: true,
|
|
||||||
type: true,
|
|
||||||
service: {
|
|
||||||
select: {
|
|
||||||
slug: true,
|
|
||||||
name: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
aboutService: {
|
|
||||||
select: {
|
|
||||||
slug: true,
|
|
||||||
name: true,
|
|
||||||
verificationStatus: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
aboutKarmaTransaction: {
|
|
||||||
select: {
|
|
||||||
points: true,
|
|
||||||
action: true,
|
|
||||||
description: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
[],
|
|
||||||
],
|
|
||||||
[
|
|
||||||
'Error while fetching notification preferences',
|
|
||||||
() =>
|
|
||||||
getOrCreateNotificationPreferences(user.id, {
|
|
||||||
id: true,
|
|
||||||
enableOnMyCommentStatusChange: true,
|
|
||||||
enableAutowatchMyComments: true,
|
|
||||||
enableNotifyPendingRepliesOnWatch: true,
|
|
||||||
karmaNotificationThreshold: true,
|
|
||||||
}),
|
|
||||||
null,
|
|
||||||
],
|
|
||||||
[
|
|
||||||
'Error while fetching total notifications',
|
|
||||||
() => prisma.notification.count({ where: { userId: user.id } }),
|
|
||||||
0,
|
|
||||||
],
|
|
||||||
])
|
|
||||||
|
|
||||||
const totalPages = Math.ceil(totalNotifications / PAGE_SIZE)
|
const totalPages = Math.ceil(totalNotifications / PAGE_SIZE)
|
||||||
|
|
||||||
@@ -199,6 +215,17 @@ const notifications = dbNotifications.map((notification) => ({
|
|||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<section class="mx-auto w-full">
|
<section class="mx-auto w-full">
|
||||||
|
{
|
||||||
|
notifications.length >= 5 && (
|
||||||
|
<PushNotificationBanner
|
||||||
|
class="mb-4"
|
||||||
|
dismissable
|
||||||
|
pushSubscriptions={pushSubscriptions}
|
||||||
|
hideIfEnabled
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
<div class="mb-4 flex items-center justify-between">
|
<div class="mb-4 flex items-center justify-between">
|
||||||
<h1 class="font-title flex items-center text-2xl leading-tight font-bold tracking-wider">
|
<h1 class="font-title flex items-center text-2xl leading-tight font-bold tracking-wider">
|
||||||
<Icon name="ri:notification-line" class="mr-2 size-6 text-zinc-400" />
|
<Icon name="ri:notification-line" class="mr-2 size-6 text-zinc-400" />
|
||||||
@@ -306,57 +333,57 @@ const notifications = dbNotifications.map((notification) => ({
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
<h2 class="font-title mt-8 mb-3 flex items-center border-b border-zinc-800 text-lg font-bold">
|
||||||
!!notificationPreferences && (
|
<Icon name="ri:settings-3-line" class="mr-2 size-5 text-zinc-400" />
|
||||||
<div class="mt-8">
|
Notification Settings
|
||||||
<h2 class="font-title mb-3 flex items-center border-b border-zinc-800 text-lg font-bold">
|
</h2>
|
||||||
<Icon name="ri:settings-3-line" class="mr-2 size-5 text-zinc-400" />
|
|
||||||
Notification Settings
|
|
||||||
</h2>
|
|
||||||
|
|
||||||
<form
|
<PushNotificationBanner class="mb-3" pushSubscriptions={pushSubscriptions} />
|
||||||
method="POST"
|
|
||||||
action={actions.notification.preferences.update}
|
|
||||||
class="rounded-lg border border-zinc-800 bg-zinc-900 p-4 shadow-sm"
|
|
||||||
>
|
|
||||||
{notificationPreferenceFields.map((field) => (
|
|
||||||
<label class="flex items-center justify-between rounded-md p-2 transition-colors duration-200 hover:bg-zinc-800">
|
|
||||||
<span class="flex items-center text-zinc-300">
|
|
||||||
<Icon name={field.icon} class="mr-2 size-5 text-zinc-400" />
|
|
||||||
{field.label}
|
|
||||||
</span>
|
|
||||||
<input
|
|
||||||
type="checkbox"
|
|
||||||
name={field.id}
|
|
||||||
checked={notificationPreferences[field.id]}
|
|
||||||
class="size-4 rounded border-zinc-700 bg-zinc-800 text-blue-600 focus:ring-blue-600"
|
|
||||||
/>
|
|
||||||
</label>
|
|
||||||
))}
|
|
||||||
|
|
||||||
<div class="mt-4 flex items-center justify-between rounded-md p-2 transition-colors duration-200 hover:bg-zinc-800">
|
<form
|
||||||
<span class="flex items-center text-zinc-300">
|
method="POST"
|
||||||
<Icon name="ri:award-line" class="mr-2 size-5 text-zinc-400" />
|
action={actions.notification.preferences.update}
|
||||||
Notify me when my karma changes by at least
|
class="rounded-lg border border-zinc-800 bg-zinc-900 p-4 shadow-sm"
|
||||||
</span>
|
>
|
||||||
<div class="flex items-center gap-2">
|
{
|
||||||
<input
|
notificationPreferenceFields.map((field) => (
|
||||||
type="number"
|
<label class="flex items-center justify-between rounded-md p-2 transition-colors duration-200 hover:bg-zinc-800">
|
||||||
name="karmaNotificationThreshold"
|
<span class="flex items-center text-zinc-300">
|
||||||
value={notificationPreferences.karmaNotificationThreshold}
|
<Icon name={field.icon} class="mr-2 size-5 text-zinc-400" />
|
||||||
min="1"
|
{field.label}
|
||||||
class="w-20 rounded border border-zinc-700 bg-zinc-800 px-2 py-1 text-zinc-200 focus:border-blue-600 focus:ring-1 focus:ring-blue-600 focus:outline-none"
|
</span>
|
||||||
/>
|
<input
|
||||||
<span class="text-zinc-400">points</span>
|
type="checkbox"
|
||||||
</div>
|
name={field.id}
|
||||||
</div>
|
checked={notificationPreferences?.[field.id]}
|
||||||
|
class="size-4 rounded border-zinc-700 bg-zinc-800 text-blue-600 focus:ring-blue-600"
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
<div class="mt-4 flex justify-end">
|
<div
|
||||||
<Button type="submit" label="Save" icon="ri:save-line" color="success" />
|
class="mt-4 flex items-center justify-between rounded-md p-2 transition-colors duration-200 hover:bg-zinc-800"
|
||||||
</div>
|
>
|
||||||
</form>
|
<span class="flex items-center text-zinc-300">
|
||||||
|
<Icon name="ri:award-line" class="mr-2 size-5 text-zinc-400" />
|
||||||
|
Notify me when my karma changes by at least
|
||||||
|
</span>
|
||||||
|
<div class="flex items-center gap-2">
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
name="karmaNotificationThreshold"
|
||||||
|
value={notificationPreferences?.karmaNotificationThreshold}
|
||||||
|
min="1"
|
||||||
|
class="w-20 rounded border border-zinc-700 bg-zinc-800 px-2 py-1 text-zinc-200 focus:border-blue-600 focus:ring-1 focus:ring-blue-600 focus:outline-none"
|
||||||
|
/>
|
||||||
|
<span class="text-zinc-400">points</span>
|
||||||
</div>
|
</div>
|
||||||
)
|
</div>
|
||||||
}
|
|
||||||
|
<div class="mt-4 flex justify-end">
|
||||||
|
<Button type="submit" label="Save" icon="ri:save-line" color="success" />
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
</section>
|
</section>
|
||||||
</BaseLayout>
|
</BaseLayout>
|
||||||
|
|||||||
@@ -19,7 +19,9 @@ import InputText from '../../components/InputText.astro'
|
|||||||
import InputTextArea from '../../components/InputTextArea.astro'
|
import InputTextArea from '../../components/InputTextArea.astro'
|
||||||
import { getAttributeCategoryInfo } from '../../constants/attributeCategories'
|
import { getAttributeCategoryInfo } from '../../constants/attributeCategories'
|
||||||
import { getAttributeTypeInfo } from '../../constants/attributeTypes'
|
import { getAttributeTypeInfo } from '../../constants/attributeTypes'
|
||||||
|
import { contactMethodUrlTypes } from '../../constants/contactMethods'
|
||||||
import { currencies } from '../../constants/currencies'
|
import { currencies } from '../../constants/currencies'
|
||||||
|
import { kycLevelClarifications } from '../../constants/kycLevelClarifications'
|
||||||
import { kycLevels } from '../../constants/kycLevels'
|
import { kycLevels } from '../../constants/kycLevels'
|
||||||
import BaseLayout from '../../layouts/BaseLayout.astro'
|
import BaseLayout from '../../layouts/BaseLayout.astro'
|
||||||
import { prisma } from '../../lib/prisma'
|
import { prisma } from '../../lib/prisma'
|
||||||
@@ -207,26 +209,42 @@ const [categories, attributes] = await Astro.locals.banners.tryMany([
|
|||||||
description="One per line. Accepts **Web**, **Onion**, and **I2P** URLs."
|
description="One per line. Accepts **Web**, **Onion**, and **I2P** URLs."
|
||||||
name="allServiceUrls"
|
name="allServiceUrls"
|
||||||
inputProps={{
|
inputProps={{
|
||||||
placeholder: 'https://example1.com\nhttps://example2.onion\nhttps://example3.b32.i2p',
|
placeholder: 'example1.com\nexample2.onion\nexample3.b32.i2p',
|
||||||
class: 'min-h-24',
|
class: 'md:min-h-20 min-h-24 h-full',
|
||||||
required: true,
|
required: true,
|
||||||
}}
|
}}
|
||||||
class="row-span-2 flex flex-col self-stretch"
|
class="flex flex-col self-stretch"
|
||||||
error={inputErrors.allServiceUrls}
|
error={inputErrors.allServiceUrls}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<InputTextArea
|
<InputTextArea
|
||||||
label="ToS URLs"
|
label="Contact Methods"
|
||||||
description="One per line"
|
description={[
|
||||||
name="tosUrls"
|
'One per line.',
|
||||||
|
`Accepts: ${contactMethodUrlTypes.map((type) => type.labelPlural).join(', ')}`,
|
||||||
|
].join('\n')}
|
||||||
|
name="contactMethods"
|
||||||
inputProps={{
|
inputProps={{
|
||||||
placeholder: 'https://example1.com/tos\nhttps://example2.com/tos',
|
placeholder: 'contact@example.com\nt.me/example\n+123 456 7890',
|
||||||
class: 'md:min-h-24',
|
class: 'h-full',
|
||||||
required: true,
|
|
||||||
}}
|
}}
|
||||||
error={inputErrors.tosUrls}
|
class="flex flex-col self-stretch"
|
||||||
|
error={inputErrors.contactMethods}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<InputTextArea
|
||||||
|
label="ToS URLs"
|
||||||
|
description="One per line"
|
||||||
|
name="tosUrls"
|
||||||
|
inputProps={{
|
||||||
|
placeholder: 'example.com/tos',
|
||||||
|
required: true,
|
||||||
|
class: 'min-h-10',
|
||||||
|
}}
|
||||||
|
error={inputErrors.tosUrls}
|
||||||
|
/>
|
||||||
|
|
||||||
<InputCardGroup
|
<InputCardGroup
|
||||||
name="kycLevel"
|
name="kycLevel"
|
||||||
label="KYC Level"
|
label="KYC Level"
|
||||||
@@ -242,6 +260,21 @@ const [categories, attributes] = await Astro.locals.banners.tryMany([
|
|||||||
error={inputErrors.kycLevel}
|
error={inputErrors.kycLevel}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<InputCardGroup
|
||||||
|
name="kycLevelClarification"
|
||||||
|
label="KYC Level Clarification"
|
||||||
|
options={kycLevelClarifications.map((clarification) => ({
|
||||||
|
label: clarification.label,
|
||||||
|
value: clarification.value,
|
||||||
|
icon: clarification.icon,
|
||||||
|
description: clarification.description,
|
||||||
|
}))}
|
||||||
|
selectedValue="NONE"
|
||||||
|
iconSize="sm"
|
||||||
|
cardSize="sm"
|
||||||
|
error={inputErrors.kycLevelClarification}
|
||||||
|
/>
|
||||||
|
|
||||||
<div class="xs:grid-cols-[1fr_2fr] grid grid-cols-1 items-stretch gap-x-4 gap-y-6">
|
<div class="xs:grid-cols-[1fr_2fr] grid grid-cols-1 items-stretch gap-x-4 gap-y-6">
|
||||||
<InputCheckboxGroup
|
<InputCheckboxGroup
|
||||||
name="categories"
|
name="categories"
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ import { getAttributeTypeInfo } from '../../constants/attributeTypes'
|
|||||||
import { formatContactMethod } from '../../constants/contactMethods'
|
import { formatContactMethod } from '../../constants/contactMethods'
|
||||||
import { currencies, getCurrencyInfo } from '../../constants/currencies'
|
import { currencies, getCurrencyInfo } from '../../constants/currencies'
|
||||||
import { getEventTypeInfo } from '../../constants/eventTypes'
|
import { getEventTypeInfo } from '../../constants/eventTypes'
|
||||||
|
import { getKycLevelClarificationInfo } from '../../constants/kycLevelClarifications'
|
||||||
import { getKycLevelInfo, kycLevels } from '../../constants/kycLevels'
|
import { getKycLevelInfo, kycLevels } from '../../constants/kycLevels'
|
||||||
import { getServiceVisibilityInfo } from '../../constants/serviceVisibility'
|
import { getServiceVisibilityInfo } from '../../constants/serviceVisibility'
|
||||||
import { getTosHighlightRatingInfo } from '../../constants/tosHighlightRating'
|
import { getTosHighlightRatingInfo } from '../../constants/tosHighlightRating'
|
||||||
@@ -67,13 +68,18 @@ const [service, dbNotificationPreferences] = await Astro.locals.banners.tryMany(
|
|||||||
'Error fetching service',
|
'Error fetching service',
|
||||||
async () =>
|
async () =>
|
||||||
prisma.service.findUnique({
|
prisma.service.findUnique({
|
||||||
where: { slug },
|
where: {
|
||||||
|
slug,
|
||||||
|
serviceVisibility: { in: ['PUBLIC', 'UNLISTED', 'ARCHIVED'] },
|
||||||
|
listedAt: { lte: new Date() },
|
||||||
|
},
|
||||||
select: {
|
select: {
|
||||||
id: true,
|
id: true,
|
||||||
slug: true,
|
slug: true,
|
||||||
name: true,
|
name: true,
|
||||||
description: true,
|
description: true,
|
||||||
kycLevel: true,
|
kycLevel: true,
|
||||||
|
kycLevelClarification: true,
|
||||||
overallScore: true,
|
overallScore: true,
|
||||||
privacyScore: true,
|
privacyScore: true,
|
||||||
trustScore: true,
|
trustScore: true,
|
||||||
@@ -219,6 +225,34 @@ const [service, dbNotificationPreferences] = await Astro.locals.banners.tryMany(
|
|||||||
],
|
],
|
||||||
])
|
])
|
||||||
|
|
||||||
|
if (!service) {
|
||||||
|
try {
|
||||||
|
const serviceWithOldSlug = await prisma.service.findFirst({
|
||||||
|
where: {
|
||||||
|
previousSlugs: { has: slug },
|
||||||
|
serviceVisibility: { in: ['PUBLIC', 'UNLISTED', 'ARCHIVED'] },
|
||||||
|
listedAt: { lte: new Date() },
|
||||||
|
},
|
||||||
|
select: { slug: true },
|
||||||
|
})
|
||||||
|
if (serviceWithOldSlug) {
|
||||||
|
return Astro.redirect(`/service/${serviceWithOldSlug.slug}`, 301)
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error)
|
||||||
|
}
|
||||||
|
|
||||||
|
return Astro.rewrite('/404')
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
service.serviceVisibility !== 'PUBLIC' &&
|
||||||
|
service.serviceVisibility !== 'UNLISTED' &&
|
||||||
|
service.serviceVisibility !== 'ARCHIVED'
|
||||||
|
) {
|
||||||
|
return Astro.rewrite('/404')
|
||||||
|
}
|
||||||
|
|
||||||
const makeWatchingDetails = (
|
const makeWatchingDetails = (
|
||||||
dbNotificationPreferences: Prisma.NotificationPreferencesGetPayload<{
|
dbNotificationPreferences: Prisma.NotificationPreferencesGetPayload<{
|
||||||
select: {
|
select: {
|
||||||
@@ -254,17 +288,7 @@ const makeWatchingDetails = (
|
|||||||
} as const
|
} as const
|
||||||
}
|
}
|
||||||
|
|
||||||
const watchingDetails = makeWatchingDetails(dbNotificationPreferences, service?.id)
|
const watchingDetails = makeWatchingDetails(dbNotificationPreferences, service.id)
|
||||||
|
|
||||||
if (!service) return Astro.rewrite('/404')
|
|
||||||
|
|
||||||
if (
|
|
||||||
service.serviceVisibility !== 'PUBLIC' &&
|
|
||||||
service.serviceVisibility !== 'UNLISTED' &&
|
|
||||||
service.serviceVisibility !== 'ARCHIVED'
|
|
||||||
) {
|
|
||||||
return Astro.rewrite('/404')
|
|
||||||
}
|
|
||||||
|
|
||||||
const statusIcon = {
|
const statusIcon = {
|
||||||
...verificationStatusesByValue,
|
...verificationStatusesByValue,
|
||||||
@@ -286,6 +310,9 @@ const hiddenLinks = [
|
|||||||
]
|
]
|
||||||
|
|
||||||
const kycLevelInfo = getKycLevelInfo(`${service.kycLevel}`)
|
const kycLevelInfo = getKycLevelInfo(`${service.kycLevel}`)
|
||||||
|
|
||||||
|
const kycLevelClarificationInfo = getKycLevelClarificationInfo(service.kycLevelClarification)
|
||||||
|
|
||||||
const userSentiment = service.userSentiment
|
const userSentiment = service.userSentiment
|
||||||
? { ...service.userSentiment, info: getUserSentimentInfo(service.userSentiment.sentiment) }
|
? { ...service.userSentiment, info: getUserSentimentInfo(service.userSentiment.sentiment) }
|
||||||
: null
|
: null
|
||||||
@@ -874,8 +901,12 @@ const activeAlertOrWarningEvents = service.events.filter(
|
|||||||
'@type': 'Review',
|
'@type': 'Review',
|
||||||
itemReviewed: { '@id': itemReviewedId },
|
itemReviewed: { '@id': itemReviewedId },
|
||||||
reviewAspect: 'KYC Level',
|
reviewAspect: 'KYC Level',
|
||||||
name: kycLevelInfo.name,
|
name:
|
||||||
reviewBody: kycLevelInfo.description,
|
kycLevelInfo.name +
|
||||||
|
(kycLevelClarificationInfo.value !== 'NONE' ? ` (${kycLevelClarificationInfo.label})` : ''),
|
||||||
|
reviewBody:
|
||||||
|
kycLevelInfo.description +
|
||||||
|
(kycLevelClarificationInfo.value !== 'NONE' ? ` ${kycLevelClarificationInfo.description}` : ''),
|
||||||
reviewRating: {
|
reviewRating: {
|
||||||
'@type': 'Rating',
|
'@type': 'Rating',
|
||||||
ratingValue: kycLevelInfo.value,
|
ratingValue: kycLevelInfo.value,
|
||||||
@@ -896,9 +927,22 @@ const activeAlertOrWarningEvents = service.events.filter(
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<dl class="flex-grow-5 basis-0">
|
<dl class="flex-grow-5 basis-0">
|
||||||
<dt class="text-base font-bold text-pretty">{kycLevelInfo.name}</dt>
|
<dt class="text-base font-bold text-pretty">
|
||||||
|
{kycLevelInfo.name}
|
||||||
|
{
|
||||||
|
kycLevelClarificationInfo.value !== 'NONE' && (
|
||||||
|
<>
|
||||||
|
<span class="text-day-400 mx-1">•</span>
|
||||||
|
<span class="text-blue-500">{kycLevelClarificationInfo.label}</span>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
</dt>
|
||||||
<dd class="text-day-700 mt-1 font-sans text-sm text-pretty">
|
<dd class="text-day-700 mt-1 font-sans text-sm text-pretty">
|
||||||
{kycLevelInfo.description}
|
{kycLevelInfo.description}
|
||||||
|
<span class="font-medium text-blue-600">
|
||||||
|
{kycLevelClarificationInfo.value !== 'NONE' && ` ${kycLevelClarificationInfo.description}`}
|
||||||
|
</span>
|
||||||
</dd>
|
</dd>
|
||||||
</dl>
|
</dl>
|
||||||
</div>
|
</div>
|
||||||
@@ -1251,9 +1295,17 @@ const activeAlertOrWarningEvents = service.events.filter(
|
|||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<h2 class="font-title border-day-500 text-day-200 mt-6 mb-3 border-b text-lg font-bold" id="verification">
|
<div class="border-day-500 mt-6 mb-3 flex items-center justify-between border-b" id="verification">
|
||||||
Verification
|
<h2 class="font-title text-day-100 text-lg font-bold">Verification</h2>
|
||||||
</h2>
|
<a
|
||||||
|
href="/about#suggestion-review-process"
|
||||||
|
class="text-day-500 hover:text-day-200 inline-flex items-center gap-1 text-xs leading-none transition-colors hover:underline"
|
||||||
|
>
|
||||||
|
Learn more
|
||||||
|
<Icon name="ri:arrow-right-s-line" class="size-4" />
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class={cn('mb-6 rounded-lg p-4', statusInfo.classNames.containerBg)}>
|
<div class={cn('mb-6 rounded-lg p-4', statusInfo.classNames.containerBg)}>
|
||||||
<div class="flex flex-col gap-4 sm:flex-row sm:items-start sm:justify-between">
|
<div class="flex flex-col gap-4 sm:flex-row sm:items-start sm:justify-between">
|
||||||
<div class="flex-1">
|
<div class="flex-1">
|
||||||
|
|||||||
Reference in New Issue
Block a user