From 88a4ec4acde87fc126bc8f1c7f8bcdef9a65fcf7 Mon Sep 17 00:00:00 2001 From: Cory Dransfeldt Date: Sun, 9 Jun 2024 10:59:57 -0700 Subject: [PATCH] # This is a combination of 3 commits. # This is the 1st commit message: fix: redirects + update root cdn url # This is the commit message #2: chore: workflow # This is the commit message #3: chore: naming --- .github/workflows/deploy-worker.yaml | 45 +++++ .gitignore | 1 + README.md | 2 +- _redirects | 30 +-- src/_data/meta.js | 2 +- src/_data/nav.js | 2 +- src/_includes/base.liquid | 8 +- src/_includes/partials/feeds/json.liquid | 2 +- src/_includes/partials/feeds/rss.liquid | 2 +- src/_includes/partials/media/grid.liquid | 20 +- .../partials/media/music/recent.liquid | 10 +- .../partials/media/watching/grid.liquid | 10 +- .../partials/media/watching/hero.liquid | 10 +- .../partials/widgets/badge-grid.liquid | 80 ++++---- src/_includes/webrings/the-claw.liquid | 10 +- src/meta/webfinger.liquid | 2 +- src/pages/errors/404.html | 12 +- src/pages/main/about.md | 10 +- src/pages/main/books/book.html | 10 +- src/pages/main/books/index.html | 10 +- src/pages/main/music/artists/artist.html | 10 +- src/pages/main/watching/movie.html | 10 +- src/pages/main/watching/show.html | 10 +- src/pages/secondary/{save.md => referrals.md} | 4 +- workers/analytics/README.md | 5 + workers/analytics/index.js | 36 ++++ workers/analytics/wrangler.toml | 13 ++ workers/contact/README.md | 8 + workers/contact/index.js | 57 ++++++ workers/contact/package-lock.json | 159 ++++++++++++++++ workers/contact/package.json | 5 + workers/contact/wrangler.toml | 12 ++ workers/now-playing/README.md | 7 + workers/now-playing/index.js | 85 +++++++++ workers/now-playing/package-lock.json | 169 +++++++++++++++++ workers/now-playing/package.json | 6 + workers/now-playing/wrangler.toml | 12 ++ workers/rebuild/README.md | 7 + workers/rebuild/index.js | 17 ++ workers/rebuild/wrangler.toml | 10 + workers/scrobble/README.md | 8 + workers/scrobble/index.js | 129 +++++++++++++ workers/scrobble/package-lock.json | 179 ++++++++++++++++++ workers/scrobble/package.json | 7 + workers/scrobble/wrangler.toml | 12 ++ 45 files changed, 1122 insertions(+), 133 deletions(-) create mode 100644 .github/workflows/deploy-worker.yaml rename src/pages/secondary/{save.md => referrals.md} (96%) create mode 100644 workers/analytics/README.md create mode 100644 workers/analytics/index.js create mode 100644 workers/analytics/wrangler.toml create mode 100644 workers/contact/README.md create mode 100644 workers/contact/index.js create mode 100644 workers/contact/package-lock.json create mode 100644 workers/contact/package.json create mode 100644 workers/contact/wrangler.toml create mode 100644 workers/now-playing/README.md create mode 100644 workers/now-playing/index.js create mode 100644 workers/now-playing/package-lock.json create mode 100644 workers/now-playing/package.json create mode 100644 workers/now-playing/wrangler.toml create mode 100644 workers/rebuild/README.md create mode 100644 workers/rebuild/index.js create mode 100644 workers/rebuild/wrangler.toml create mode 100644 workers/scrobble/README.md create mode 100644 workers/scrobble/index.js create mode 100644 workers/scrobble/package-lock.json create mode 100644 workers/scrobble/package.json create mode 100644 workers/scrobble/wrangler.toml diff --git a/.github/workflows/deploy-worker.yaml b/.github/workflows/deploy-worker.yaml new file mode 100644 index 00000000..3c765aac --- /dev/null +++ b/.github/workflows/deploy-worker.yaml @@ -0,0 +1,45 @@ +name: Deploy Cloudflare Worker +on: + workflow_dispatch: + inputs: + worker-folder: + description: 'Select the folder containing the worker to deploy' + required: true + type: choice + options: + - analytics + - contact + - now-playing + - rebuild + - scrobble +jobs: + deploy: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v2 + + - name: Setup Node.js + uses: actions/setup-node@v2 + with: + node-version: '16' + + - name: Install Wrangler + run: npm install -g wrangler + + - name: Inject environment variables into wrangler.toml + run: | + cd workers/${{ github.event.inputs.worker-folder }} + sed -i 's/^zone_id =.*/zone_id = "${{ secrets.CLOUDFLARE_ZONE_ID }}"/' wrangler.toml + sed -i 's/^account_id =.*/account_id = "${{ secrets.CLOUDFLARE_ACCOUNT_ID }}"/' wrangler.toml + + - name: Install dependencies + run: | + cd workers/${{ github.event.inputs.worker-folder }} + npm i + + - name: Deploy to Cloudflare Worker + run: | + cd workers/${{ github.event.inputs.worker-folder }} + wrangler deploy --env production + env: + CLOUDFLARE_API_TOKEN: ${{ secrets.CLOUDFLARE_API_TOKEN }} \ No newline at end of file diff --git a/.gitignore b/.gitignore index 5d7afd0f..e3e90b51 100644 --- a/.gitignore +++ b/.gitignore @@ -2,6 +2,7 @@ _site node_modules .cache +.wrangler # local dependencies .env.local diff --git a/README.md b/README.md index ee1016a2..325eff9b 100644 --- a/README.md +++ b/README.md @@ -4,6 +4,6 @@ Hi! I'm Cory. 👋🏻 This is the code for my personal website and portfolio. Built using [11ty](https://www.11ty.dev). -[![Follow @cory@social.lol on Mastodon](https://coryd-dev.b-cdn.net/assets/badges/mastodon.png?aspect_ratio=1:.35&width=88)](https://social.lol/@cory) [![Buy me a Coffee](https://coryd-dev.b-cdn.net/assets/badges/buymeacoffee.png?aspect_ratio=1:.35&width=88)](https://www.buymeacoffee.com/cory) [![Made with 11ty](https://coryd-dev.b-cdn.net/assets/badges/eleventy.png?aspect_ratio=1:.35&width=88)](https://eleventy.dev) +[![Follow @cory@social.lol on Mastodon](https://cdn.coryd.dev/assets/badges/mastodon.png?aspect_ratio=1:.35&width=88)](https://social.lol/@cory) [![Buy me a Coffee](https://cdn.coryd.dev/assets/badges/buymeacoffee.png?aspect_ratio=1:.35&width=88)](https://www.buymeacoffee.com/cory) [![Made with 11ty](https://cdn.coryd.dev/assets/badges/eleventy.png?aspect_ratio=1:.35&width=88)](https://eleventy.dev) [Music](https://coryd.dev/music) • [Watching](https://coryd.dev/watching) • [Books](https://coryd.dev/books) • [Now](https://coryd.dev/now) diff --git a/_redirects b/_redirects index 80cbfba1..e00b429a 100644 --- a/_redirects +++ b/_redirects @@ -40,20 +40,20 @@ /blog/digital-privacy-tools /posts/2021/digital-privacy-tools/ 301 # assets -/favicon.ico https://coryd-dev.b-cdn.net/assets/icons/favicon.ico 301 -/assets/icons/favicon.ico https://coryd-dev.b-cdn.net/assets/icons/favicon.ico 301 -/apple-touch-icon.png https://coryd-dev.b-cdn.net/assets/icons/apple-touch-icon.png 301 -/apple-touch-icon https://coryd-dev.b-cdn.net/assets/icons/apple-touch-icon.png 301 -/apple-touch-icon-precomposed.png https://coryd-dev.b-cdn.net/assets/icons/apple-touch-icon.png 301 -/assets/icons/apple-touch-icon.png https://coryd-dev.b-cdn.net/assets/icons/apple-touch-icon.png 301 -/assets/img/feed-icon.png https://coryd-dev.b-cdn.net/assets/icons/apple-touch-icon.png 301 -/assets/img/logo.webp https://coryd-dev.b-cdn.net/assets/avatar.webp 301 -/static/favicons/apple-touch-icon.png https://coryd-dev.b-cdn.net/assets/icons/apple-touch-icon.png 301 -/static/images/avatar.png https://coryd-dev.b-cdn.net/assets/avatar.png 301 -/static/images/avatar.webp https://coryd-dev.b-cdn.net/assets/avatar.webp 301 -/assets/img/favicon/favicon-32x32.png https://coryd-dev.b-cdn.net/assets/icons/favicon.ico 301 -/assets/img/favicon/favicon-16x16.png https://coryd-dev.b-cdn.net/assets/icons/favicon.ico 301 -/assets/img/logo.webp https://coryd-dev.b-cdn.net/assets/avatar.webp 301 +/favicon.ico https://cdn.coryd.dev/assets/icons/favicon.ico 301 +/assets/icons/favicon.ico https://cdn.coryd.dev/assets/icons/favicon.ico 301 +/apple-touch-icon.png https://cdn.coryd.dev/assets/icons/apple-touch-icon.png 301 +/apple-touch-icon https://cdn.coryd.dev/assets/icons/apple-touch-icon.png 301 +/apple-touch-icon-precomposed.png https://cdn.coryd.dev/assets/icons/apple-touch-icon.png 301 +/assets/icons/apple-touch-icon.png https://cdn.coryd.dev/assets/icons/apple-touch-icon.png 301 +/assets/img/feed-icon.png https://cdn.coryd.dev/assets/icons/apple-touch-icon.png 301 +/assets/img/logo.webp https://cdn.coryd.dev/assets/avatar.webp 301 +/static/favicons/apple-touch-icon.png https://cdn.coryd.dev/assets/icons/apple-touch-icon.png 301 +/static/images/avatar.png https://cdn.coryd.dev/assets/avatar.png 301 +/static/images/avatar.webp https://cdn.coryd.dev/assets/avatar.webp 301 +/assets/img/favicon/favicon-32x32.png https://cdn.coryd.dev/assets/icons/favicon.ico 301 +/assets/img/favicon/favicon-16x16.png https://cdn.coryd.dev/assets/icons/favicon.ico 301 +/assets/img/logo.webp https://cdn.coryd.dev/assets/avatar.webp 301 # feeds /rss https://feedpress.me/coryd 301 @@ -73,12 +73,12 @@ # general /articles/ / 301 /tags /search 301 -/tags/* /search 301 /referrals /save 301 /recent/movies /watching/recent/movies 301 /recent/shows /watching/recent/shows 301 /mastodon https://social.lol/@cory 301 /coffee https://www.buymeacoffee.com/cory 301 +/tags/* /search 301 /music/genre/* /music/genres/:splat 301 # mastodon diff --git a/src/_data/meta.js b/src/_data/meta.js index 933dea95..5dfc3973 100644 --- a/src/_data/meta.js +++ b/src/_data/meta.js @@ -10,7 +10,7 @@ export default async function () { "locale": "en_US", "lang": "en", "meta_data": { - "opengraph_default": "https://coryd-dev.b-cdn.net/assets/avatar.png" + "opengraph_default": "https://cdn.coryd.dev/assets/avatar.png" } } } diff --git a/src/_data/nav.js b/src/_data/nav.js index 60c04e67..df597248 100644 --- a/src/_data/nav.js +++ b/src/_data/nav.js @@ -3,7 +3,7 @@ export default async function () { footer: [ { name: 'Now' }, { name: 'Uses' }, - { name: 'Save' }, + { name: 'Referrals' }, { name: 'Blogroll' }, ], menu: [ diff --git a/src/_includes/base.liquid b/src/_includes/base.liquid index 33455d7e..7b1d091b 100644 --- a/src/_includes/base.liquid +++ b/src/_includes/base.liquid @@ -62,7 +62,7 @@ {%- when 'movie' -%} {%- assign ogImage = movie.backdrop -%} {%- when 'show' -%} - {%- assign ogImage = 'https://coryd-dev.b-cdn.net/shows/backdrops/backdrop-' | append: show.tmdb_id | append: '.jpg' -%} + {%- assign ogImage = 'https://cdn.coryd.dev/shows/backdrops/backdrop-' | append: show.tmdb_id | append: '.jpg' -%} {%- when 'genre' -%} {%- assign genreArtist = genre.artists | shuffleArray | first -%} {%- assign ogImage = genreArtist.image -%} @@ -86,9 +86,9 @@ - - - + + + diff --git a/src/_includes/partials/feeds/json.liquid b/src/_includes/partials/feeds/json.liquid index 6a3ae815..3e23bc7e 100644 --- a/src/_includes/partials/feeds/json.liquid +++ b/src/_includes/partials/feeds/json.liquid @@ -2,7 +2,7 @@ { "version": "https://jsonfeed.org/version/1", "title": "{{ title }}", - "icon": "https://coryd-dev.b-cdn.net/assets/avatar.png", + "icon": "https://cdn.coryd.dev/assets/avatar.png", "home_page_url": "{{ meta.url }}", "feed_url": "{{ permalink | absoluteUrl: meta.url }}", "items": [{% for entry in entries limit: 20 -%} diff --git a/src/_includes/partials/feeds/rss.liquid b/src/_includes/partials/feeds/rss.liquid index 41fc6c17..c9db1763 100644 --- a/src/_includes/partials/feeds/rss.liquid +++ b/src/_includes/partials/feeds/rss.liquid @@ -10,7 +10,7 @@ {{ title }} {{ permalink | absoluteUrl: meta.url }} - https://coryd-dev.b-cdn.net/assets/avatar.png + https://cdn.coryd.dev/assets/avatar.png 144 144 diff --git a/src/_includes/partials/media/grid.liquid b/src/_includes/partials/media/grid.liquid index b6b71dac..d8988ee4 100644 --- a/src/_includes/partials/media/grid.liquid +++ b/src/_includes/partials/media/grid.liquid @@ -19,16 +19,16 @@ {% if shape == 'square' %} {{ alt }} {{ alt }} {{ alt }} Made with 11ty Made with 11ty Made with 11ty Made with 11ty Made with 11ty Made with 11ty Made with 11ty Made with 11ty The Claw Webring {{ alt }} {{ alt }} {{ alt }} {{ alt }} {{ alt }} {{ alt }}{{ title }} diff --git a/workers/analytics/README.md b/workers/analytics/README.md new file mode 100644 index 00000000..e209d51f --- /dev/null +++ b/workers/analytics/README.md @@ -0,0 +1,5 @@ +# analytics worker + +```bash +wrangler deploy --env production +``` diff --git a/workers/analytics/index.js b/workers/analytics/index.js new file mode 100644 index 00000000..f7a95e65 --- /dev/null +++ b/workers/analytics/index.js @@ -0,0 +1,36 @@ +const ScriptName = '/js/script.js'; +const Endpoint = '/api/event'; + +const ScriptWithoutExtension = ScriptName.replace('.js', '') + +addEventListener('fetch', event => { + event.passThroughOnException(); + event.respondWith(handleRequest(event)); +}) + +async function handleRequest(event) { + const pathname = new URL(event.request.url).pathname + const [baseUri, ...extensions] = pathname.split('.') + + if (baseUri.endsWith(ScriptWithoutExtension)) { + return getScript(event, extensions) + } else if (pathname.endsWith(Endpoint)) { + return postData(event) + } + return new Response(null, { status: 404 }) +} + +async function getScript(event, extensions) { + let response = await caches.default.match(event.request); + if (!response) { + response = await fetch("https://plausible.io/js/plausible." + extensions.join(".")); + event.waitUntil(caches.default.put(event.request, response.clone())); + } + return response; +} + +async function postData(event) { + const request = new Request(event.request); + request.headers.delete('cookie'); + return await fetch("https://plausible.io/api/event", request); +} \ No newline at end of file diff --git a/workers/analytics/wrangler.toml b/workers/analytics/wrangler.toml new file mode 100644 index 00000000..6cdffb79 --- /dev/null +++ b/workers/analytics/wrangler.toml @@ -0,0 +1,13 @@ +name = "analytics-worker" +main = "./index.js" +compatibility_date = "2023-01-01" + +account_id = "" +workers_dev = true + +[env.production] +name = "analytics-worker-production" +routes = [ + { pattern = "coryd.dev/js/*", zone_id = "" }, + { pattern = "coryd.dev/api/event", zone_id = "" } +] \ No newline at end of file diff --git a/workers/contact/README.md b/workers/contact/README.md new file mode 100644 index 00000000..c6e87323 --- /dev/null +++ b/workers/contact/README.md @@ -0,0 +1,8 @@ +# contact worker + +```bash +wrangler deploy --env production + +wrangler secret put SUPABASE_URL --env production +wrangler secret put SUPABASE_KEY --env production +``` diff --git a/workers/contact/index.js b/workers/contact/index.js new file mode 100644 index 00000000..f2f8feb0 --- /dev/null +++ b/workers/contact/index.js @@ -0,0 +1,57 @@ +import { createClient } from '@supabase/supabase-js'; + +const RATE_LIMIT = 5; +const TIME_FRAME = 60 * 60 * 1000; + +const ipSubmissions = new Map(); + +export default { + async fetch(request, env) { + if (request.method === 'POST') { + const ip = request.headers.get('CF-Connecting-IP') || request.headers.get('X-Forwarded-For') || request.headers.get('Remote-Addr'); + const currentTime = Date.now(); + + if (!ipSubmissions.has(ip)) { + ipSubmissions.set(ip, []); + } + + const submissions = ipSubmissions.get(ip).filter(time => currentTime - time < TIME_FRAME); + + if (submissions.length >= RATE_LIMIT) { + return new Response('Rate limit exceeded', { status: 429 }); + } + + submissions.push(currentTime); + ipSubmissions.set(ip, submissions); + + try { + const formData = await request.formData(); + const name = formData.get('name'); + const email = formData.get('email'); + const message = formData.get('message'); + const hpName = formData.get('hp_name'); + + // check the honeypot field + if (hpName) return new Response('Spam detected', { status: 400 }); + + // validate input + if (!name || !email || !message) return new Response('Invalid input', { status: 400 }); + + const supabaseUrl = env.SUPABASE_URL; + const supabaseKey = env.SUPABASE_KEY; + const supabase = createClient(supabaseUrl, supabaseKey); + const { error } = await supabase.from('contacts').insert([ + { name, email, message, replied: false } + ]); + + if (error) throw error; + + return Response.redirect('https://coryd.dev/contact/success', 303); + } catch (error) { + return new Response(error.message, { status: 500 }); + } + } else { + return new Response('Method not allowed', { status: 405 }); + } + } +}; \ No newline at end of file diff --git a/workers/contact/package-lock.json b/workers/contact/package-lock.json new file mode 100644 index 00000000..d718a041 --- /dev/null +++ b/workers/contact/package-lock.json @@ -0,0 +1,159 @@ +{ + "name": "contact-worker", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "dependencies": { + "@supabase/supabase-js": "^2.43.4" + } + }, + "node_modules/@supabase/auth-js": { + "version": "2.64.2", + "resolved": "https://registry.npmjs.org/@supabase/auth-js/-/auth-js-2.64.2.tgz", + "integrity": "sha512-s+lkHEdGiczDrzXJ1YWt2y3bxRi+qIUnXcgkpLSrId7yjBeaXBFygNjTaoZLG02KNcYwbuZ9qkEIqmj2hF7svw==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14" + } + }, + "node_modules/@supabase/functions-js": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/@supabase/functions-js/-/functions-js-2.3.1.tgz", + "integrity": "sha512-QyzNle/rVzlOi4BbVqxLSH828VdGY1RElqGFAj+XeVypj6+PVtMlD21G8SDnsPQDtlqqTtoGRgdMlQZih5hTuw==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14" + } + }, + "node_modules/@supabase/node-fetch": { + "version": "2.6.15", + "resolved": "https://registry.npmjs.org/@supabase/node-fetch/-/node-fetch-2.6.15.tgz", + "integrity": "sha512-1ibVeYUacxWYi9i0cf5efil6adJ9WRyZBLivgjs+AUpewx1F3xPi7gLgaASI2SmIQxPoCEjAsLAzKPgMJVgOUQ==", + "license": "MIT", + "dependencies": { + "whatwg-url": "^5.0.0" + }, + "engines": { + "node": "4.x || >=6.0.0" + } + }, + "node_modules/@supabase/postgrest-js": { + "version": "1.15.2", + "resolved": "https://registry.npmjs.org/@supabase/postgrest-js/-/postgrest-js-1.15.2.tgz", + "integrity": "sha512-9/7pUmXExvGuEK1yZhVYXPZnLEkDTwxgMQHXLrN5BwPZZm4iUCL1YEyep/Z2lIZah8d8M433mVAUEGsihUj5KQ==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14" + } + }, + "node_modules/@supabase/realtime-js": { + "version": "2.9.5", + "resolved": "https://registry.npmjs.org/@supabase/realtime-js/-/realtime-js-2.9.5.tgz", + "integrity": "sha512-TEHlGwNGGmKPdeMtca1lFTYCedrhTAv3nZVoSjrKQ+wkMmaERuCe57zkC5KSWFzLYkb5FVHW8Hrr+PX1DDwplQ==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14", + "@types/phoenix": "^1.5.4", + "@types/ws": "^8.5.10", + "ws": "^8.14.2" + } + }, + "node_modules/@supabase/storage-js": { + "version": "2.5.5", + "resolved": "https://registry.npmjs.org/@supabase/storage-js/-/storage-js-2.5.5.tgz", + "integrity": "sha512-OpLoDRjFwClwc2cjTJZG8XviTiQH4Ik8sCiMK5v7et0MDu2QlXjCAW3ljxJB5+z/KazdMOTnySi+hysxWUPu3w==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14" + } + }, + "node_modules/@supabase/supabase-js": { + "version": "2.43.4", + "resolved": "https://registry.npmjs.org/@supabase/supabase-js/-/supabase-js-2.43.4.tgz", + "integrity": "sha512-/pLPaxiIsn5Vaz3s32HC6O/VNwfeddnzS0bZRpOW0AKcPuXroD8pT9G8mpiBlZfpKsMmq6k7tlhW7Sr1PAQ1lw==", + "license": "MIT", + "dependencies": { + "@supabase/auth-js": "2.64.2", + "@supabase/functions-js": "2.3.1", + "@supabase/node-fetch": "2.6.15", + "@supabase/postgrest-js": "1.15.2", + "@supabase/realtime-js": "2.9.5", + "@supabase/storage-js": "2.5.5" + } + }, + "node_modules/@types/node": { + "version": "20.14.2", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.14.2.tgz", + "integrity": "sha512-xyu6WAMVwv6AKFLB+e/7ySZVr/0zLCzOa7rSpq6jNwpqOrUbcACDWC+53d4n2QHOnDou0fbIsg8wZu/sxrnI4Q==", + "license": "MIT", + "dependencies": { + "undici-types": "~5.26.4" + } + }, + "node_modules/@types/phoenix": { + "version": "1.6.4", + "resolved": "https://registry.npmjs.org/@types/phoenix/-/phoenix-1.6.4.tgz", + "integrity": "sha512-B34A7uot1Cv0XtaHRYDATltAdKx0BvVKNgYNqE4WjtPUa4VQJM7kxeXcVKaH+KS+kCmZ+6w+QaUdcljiheiBJA==", + "license": "MIT" + }, + "node_modules/@types/ws": { + "version": "8.5.10", + "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.10.tgz", + "integrity": "sha512-vmQSUcfalpIq0R9q7uTo2lXs6eGIpt9wtnLdMv9LVpIjCA/+ufZRozlVoVelIYixx1ugCBKDhn89vnsEGOCx9A==", + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", + "license": "MIT" + }, + "node_modules/undici-types": { + "version": "5.26.5", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz", + "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==", + "license": "MIT" + }, + "node_modules/webidl-conversions": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==", + "license": "BSD-2-Clause" + }, + "node_modules/whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", + "license": "MIT", + "dependencies": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" + } + }, + "node_modules/ws": { + "version": "8.17.0", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.17.0.tgz", + "integrity": "sha512-uJq6108EgZMAl20KagGkzCKfMEjxmKvZHG7Tlq0Z6nOky7YF7aq4mOx6xK8TJ/i1LeK4Qus7INktacctDgY8Ow==", + "license": "MIT", + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": ">=5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + } + } +} diff --git a/workers/contact/package.json b/workers/contact/package.json new file mode 100644 index 00000000..2f6edacc --- /dev/null +++ b/workers/contact/package.json @@ -0,0 +1,5 @@ +{ + "dependencies": { + "@supabase/supabase-js": "^2.43.4" + } +} diff --git a/workers/contact/wrangler.toml b/workers/contact/wrangler.toml new file mode 100644 index 00000000..bad56b78 --- /dev/null +++ b/workers/contact/wrangler.toml @@ -0,0 +1,12 @@ +name = "contact-form-worker" +main = "./index.js" +compatibility_date = "2023-01-01" + +account_id = "" +workers_dev = true + +[env.production] +name = "contact-form-worker-production" +routes = [ + { pattern = "coryd.dev/api/contact", zone_id = "" } +] \ No newline at end of file diff --git a/workers/now-playing/README.md b/workers/now-playing/README.md new file mode 100644 index 00000000..de6e67c0 --- /dev/null +++ b/workers/now-playing/README.md @@ -0,0 +1,7 @@ +# now-playing worker + +```bash +wrangler deploy --env production +wrangler secret put SUPABASE_URL --env production +wrangler secret put SUPABASE_KEY --env production +``` diff --git a/workers/now-playing/index.js b/workers/now-playing/index.js new file mode 100644 index 00000000..bbc04a46 --- /dev/null +++ b/workers/now-playing/index.js @@ -0,0 +1,85 @@ +import { createClient } from '@supabase/supabase-js'; +import slugify from 'slugify'; + +const sanitizeMediaString = (str) => { + const sanitizedString = str.normalize('NFD').replace(/[\u0300-\u036f\u2010—\.\?\(\)\[\]\{\}]/g, '').replace(/\.{3}/g, ''); + + return slugify(sanitizedString, { + replacement: '-', + remove: /[#,&,+()$~%.'":*?<>{}]/g, + lower: true, + }); +}; + +const regionNames = new Intl.DisplayNames(['en'], { type: 'region' }); +const getCountryName = (countryCode) => regionNames.of(countryCode.trim()) || countryCode.trim(); +const parseCountryField = (countryField) => { + if (!countryField) return null; + + const delimiters = [',', '/', '&', 'and']; + let countries = [countryField]; + + delimiters.forEach(delimiter => { + countries = countries.flatMap(country => country.split(delimiter)); + }); + + return countries.map(getCountryName).join(', '); +}; + +const fetchGenreById = async (supabase, genreId) => { + const { data, error } = await supabase + .from('genres') + .select('emoji') + .eq('id', genreId) + .single(); + + if (error) { + console.error('Error fetching genre:', error); + return null; + } + + return data.emoji; +}; + +export default { + async fetch(request, env) { + const SUPABASE_URL = env.SUPABASE_URL; + const SUPABASE_KEY = env.SUPABASE_KEY; + const supabase = createClient(SUPABASE_URL, SUPABASE_KEY); + + const { data, error } = await supabase + .from('listens') + .select(` + track_name, + artist_name, + listened_at, + artists (mbid, genres, country, emoji) + `) + .order('listened_at', { ascending: false }) + .range(0, 1); + + const headers = { + "Content-Type": "application/json", + "Cache-Control": "public, s-maxage=360, stale-while-revalidate=1080", + }; + + if (error) { + console.error('Error fetching data:', error); + return new Response(JSON.stringify({ error: "Failed to fetch the latest track" }), { headers }); + } + + if (data.length === 0) { + return new Response(JSON.stringify({ message: "No recent tracks found" }), { headers }); + } + + const scrobbleData = data[0]; + const genreEmoji = await fetchGenreById(supabase, scrobbleData.artists.genres); + const emoji = scrobbleData.artists.emoji || genreEmoji; + + return new Response(JSON.stringify({ + content: `${emoji || '🎧'} ${scrobbleData.track_name} by ${ + scrobbleData.artist_name + }`, + }), { headers }); + } +}; \ No newline at end of file diff --git a/workers/now-playing/package-lock.json b/workers/now-playing/package-lock.json new file mode 100644 index 00000000..dfcd75e1 --- /dev/null +++ b/workers/now-playing/package-lock.json @@ -0,0 +1,169 @@ +{ + "name": "now-playing-worker", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "dependencies": { + "@supabase/supabase-js": "^2.43.4", + "slugify": "^1.6.6" + } + }, + "node_modules/@supabase/auth-js": { + "version": "2.64.2", + "resolved": "https://registry.npmjs.org/@supabase/auth-js/-/auth-js-2.64.2.tgz", + "integrity": "sha512-s+lkHEdGiczDrzXJ1YWt2y3bxRi+qIUnXcgkpLSrId7yjBeaXBFygNjTaoZLG02KNcYwbuZ9qkEIqmj2hF7svw==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14" + } + }, + "node_modules/@supabase/functions-js": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/@supabase/functions-js/-/functions-js-2.3.1.tgz", + "integrity": "sha512-QyzNle/rVzlOi4BbVqxLSH828VdGY1RElqGFAj+XeVypj6+PVtMlD21G8SDnsPQDtlqqTtoGRgdMlQZih5hTuw==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14" + } + }, + "node_modules/@supabase/node-fetch": { + "version": "2.6.15", + "resolved": "https://registry.npmjs.org/@supabase/node-fetch/-/node-fetch-2.6.15.tgz", + "integrity": "sha512-1ibVeYUacxWYi9i0cf5efil6adJ9WRyZBLivgjs+AUpewx1F3xPi7gLgaASI2SmIQxPoCEjAsLAzKPgMJVgOUQ==", + "license": "MIT", + "dependencies": { + "whatwg-url": "^5.0.0" + }, + "engines": { + "node": "4.x || >=6.0.0" + } + }, + "node_modules/@supabase/postgrest-js": { + "version": "1.15.2", + "resolved": "https://registry.npmjs.org/@supabase/postgrest-js/-/postgrest-js-1.15.2.tgz", + "integrity": "sha512-9/7pUmXExvGuEK1yZhVYXPZnLEkDTwxgMQHXLrN5BwPZZm4iUCL1YEyep/Z2lIZah8d8M433mVAUEGsihUj5KQ==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14" + } + }, + "node_modules/@supabase/realtime-js": { + "version": "2.9.5", + "resolved": "https://registry.npmjs.org/@supabase/realtime-js/-/realtime-js-2.9.5.tgz", + "integrity": "sha512-TEHlGwNGGmKPdeMtca1lFTYCedrhTAv3nZVoSjrKQ+wkMmaERuCe57zkC5KSWFzLYkb5FVHW8Hrr+PX1DDwplQ==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14", + "@types/phoenix": "^1.5.4", + "@types/ws": "^8.5.10", + "ws": "^8.14.2" + } + }, + "node_modules/@supabase/storage-js": { + "version": "2.5.5", + "resolved": "https://registry.npmjs.org/@supabase/storage-js/-/storage-js-2.5.5.tgz", + "integrity": "sha512-OpLoDRjFwClwc2cjTJZG8XviTiQH4Ik8sCiMK5v7et0MDu2QlXjCAW3ljxJB5+z/KazdMOTnySi+hysxWUPu3w==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14" + } + }, + "node_modules/@supabase/supabase-js": { + "version": "2.43.4", + "resolved": "https://registry.npmjs.org/@supabase/supabase-js/-/supabase-js-2.43.4.tgz", + "integrity": "sha512-/pLPaxiIsn5Vaz3s32HC6O/VNwfeddnzS0bZRpOW0AKcPuXroD8pT9G8mpiBlZfpKsMmq6k7tlhW7Sr1PAQ1lw==", + "license": "MIT", + "dependencies": { + "@supabase/auth-js": "2.64.2", + "@supabase/functions-js": "2.3.1", + "@supabase/node-fetch": "2.6.15", + "@supabase/postgrest-js": "1.15.2", + "@supabase/realtime-js": "2.9.5", + "@supabase/storage-js": "2.5.5" + } + }, + "node_modules/@types/node": { + "version": "20.14.2", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.14.2.tgz", + "integrity": "sha512-xyu6WAMVwv6AKFLB+e/7ySZVr/0zLCzOa7rSpq6jNwpqOrUbcACDWC+53d4n2QHOnDou0fbIsg8wZu/sxrnI4Q==", + "license": "MIT", + "dependencies": { + "undici-types": "~5.26.4" + } + }, + "node_modules/@types/phoenix": { + "version": "1.6.4", + "resolved": "https://registry.npmjs.org/@types/phoenix/-/phoenix-1.6.4.tgz", + "integrity": "sha512-B34A7uot1Cv0XtaHRYDATltAdKx0BvVKNgYNqE4WjtPUa4VQJM7kxeXcVKaH+KS+kCmZ+6w+QaUdcljiheiBJA==", + "license": "MIT" + }, + "node_modules/@types/ws": { + "version": "8.5.10", + "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.10.tgz", + "integrity": "sha512-vmQSUcfalpIq0R9q7uTo2lXs6eGIpt9wtnLdMv9LVpIjCA/+ufZRozlVoVelIYixx1ugCBKDhn89vnsEGOCx9A==", + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/slugify": { + "version": "1.6.6", + "resolved": "https://registry.npmjs.org/slugify/-/slugify-1.6.6.tgz", + "integrity": "sha512-h+z7HKHYXj6wJU+AnS/+IH8Uh9fdcX1Lrhg1/VMdf9PwoBQXFcXiAdsy2tSK0P6gKwJLXp02r90ahUCqHk9rrw==", + "license": "MIT", + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", + "license": "MIT" + }, + "node_modules/undici-types": { + "version": "5.26.5", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz", + "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==", + "license": "MIT" + }, + "node_modules/webidl-conversions": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==", + "license": "BSD-2-Clause" + }, + "node_modules/whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", + "license": "MIT", + "dependencies": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" + } + }, + "node_modules/ws": { + "version": "8.17.0", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.17.0.tgz", + "integrity": "sha512-uJq6108EgZMAl20KagGkzCKfMEjxmKvZHG7Tlq0Z6nOky7YF7aq4mOx6xK8TJ/i1LeK4Qus7INktacctDgY8Ow==", + "license": "MIT", + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": ">=5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + } + } +} diff --git a/workers/now-playing/package.json b/workers/now-playing/package.json new file mode 100644 index 00000000..86b04bcf --- /dev/null +++ b/workers/now-playing/package.json @@ -0,0 +1,6 @@ +{ + "dependencies": { + "@supabase/supabase-js": "^2.43.4", + "slugify": "^1.6.6" + } +} diff --git a/workers/now-playing/wrangler.toml b/workers/now-playing/wrangler.toml new file mode 100644 index 00000000..07d66a35 --- /dev/null +++ b/workers/now-playing/wrangler.toml @@ -0,0 +1,12 @@ +name = "now-playing-worker" +main = "./index.js" # Specify the entry point +compatibility_date = "2023-01-01" + +account_id = "" +workers_dev = true + +[env.production] +name = "now-playing-worker-production" +routes = [ + { pattern = "coryd.dev/api/now-playing", zone_id = "" } +] \ No newline at end of file diff --git a/workers/rebuild/README.md b/workers/rebuild/README.md new file mode 100644 index 00000000..4d33cdec --- /dev/null +++ b/workers/rebuild/README.md @@ -0,0 +1,7 @@ +# rebuild worker + +```bash +wrangler deploy --env production + +wrangler secret put DEPLOY_HOOK_URL --env production +``` diff --git a/workers/rebuild/index.js b/workers/rebuild/index.js new file mode 100644 index 00000000..89293986 --- /dev/null +++ b/workers/rebuild/index.js @@ -0,0 +1,17 @@ +export default { + async scheduled(event, env, ctx) { + const deployHookUrl = env.DEPLOY_HOOK_URL; + + const response = await fetch(deployHookUrl, { + method: 'POST', + }); + + if (!response.ok) { + const errorText = await response.text(); + console.error(`Error triggering deploy: ${response.statusText}`, errorText); + return; + } + + console.log('Deploy triggered successfully'); + } +}; \ No newline at end of file diff --git a/workers/rebuild/wrangler.toml b/workers/rebuild/wrangler.toml new file mode 100644 index 00000000..a0b580d7 --- /dev/null +++ b/workers/rebuild/wrangler.toml @@ -0,0 +1,10 @@ +name = "scheduled-rebuild-worker" +main = "./index.js" +compatibility_date = "2023-01-01" + +account_id = "" +workers_dev = true + +[env.production] +name = "scheduled-rebuild-worker-production" +triggers = {crons = ["0 * * * *"]} \ No newline at end of file diff --git a/workers/scrobble/README.md b/workers/scrobble/README.md new file mode 100644 index 00000000..550eaea5 --- /dev/null +++ b/workers/scrobble/README.md @@ -0,0 +1,8 @@ +# Scrobble worker + +```bash +wrangler deploy --env production +wrangler secret put SUPABASE_URL --env production +wrangler secret put SUPABASE_KEY --env production +wrangler secret put ACCOUNT_ID_PLEX --env production +``` diff --git a/workers/scrobble/index.js b/workers/scrobble/index.js new file mode 100644 index 00000000..f5bebecf --- /dev/null +++ b/workers/scrobble/index.js @@ -0,0 +1,129 @@ +import { createClient } from '@supabase/supabase-js'; +import { DateTime } from 'luxon'; +import slugify from 'slugify'; + +const sanitizeMediaString = (str) => { + const sanitizedString = str.normalize('NFD').replace(/[\u0300-\u036f\u2010\-\.\?\(\)\[\]\{\}]/g, '').replace(/\.{3}/g, ''); + return slugify(sanitizedString, { + replacement: '-', + remove: /[#,&,+()$~%.'":*?<>{}]/g, + lower: true, + }); +}; + +export default { + async fetch(request, env) { + const SUPABASE_URL = env.SUPABASE_URL; + const SUPABASE_KEY = env.SUPABASE_KEY; + const ACCOUNT_ID_PLEX = env.ACCOUNT_ID_PLEX; + const supabase = createClient(SUPABASE_URL, SUPABASE_KEY); + + const url = new URL(request.url); + const params = url.searchParams; + const id = params.get('id'); + + if (!id) { + return new Response(JSON.stringify({ status: 'Bad request' }), { headers: { "Content-Type": "application/json" } }); + } + if (id !== ACCOUNT_ID_PLEX) { + return new Response(JSON.stringify({ status: 'Forbidden' }), { headers: { "Content-Type": "application/json" } }); + } + + const contentType = request.headers.get("Content-Type") || ''; + if (!contentType.includes("multipart/form-data")) { + return new Response(JSON.stringify({ status: 'Bad request', message: 'Invalid Content-Type. Expected multipart/form-data.' }), { headers: { "Content-Type": "application/json" } }); + } + + try { + const data = await request.formData(); + const payload = JSON.parse(data.get('payload')); + + if (payload?.event === 'media.scrobble') { + const artist = payload['Metadata']['grandparentTitle']; + const album = payload['Metadata']['parentTitle']; + const track = payload['Metadata']['title']; + const listenedAt = Math.floor(DateTime.now().toSeconds()); + const artistKey = sanitizeMediaString(artist); + const albumKey = `${artistKey}-${sanitizeMediaString(album)}`; + + let { data: artistData, error: artistError } = await supabase + .from('artists') + .select('*') + .ilike('name_string', artist) + .single(); + + if (artistError && artistError.code === 'PGRST116') { + const { error: insertArtistError } = await supabase.from('artists').insert([ + { + mbid: null, + image: `/artists/${artistKey}.jpg`, + key: artistKey, + name: artist, + tentative: true + } + ]); + + if (insertArtistError) { + return new Response(JSON.stringify({ status: 'error', message: insertArtistError.message }), { headers: { "Content-Type": "application/json" } }); + } + + ({ data: artistData, error: artistError } = await supabase + .from('artists') + .select('*') + .ilike('name_string', artist) + .single()); + } else if (artistError) { + return new Response(JSON.stringify({ status: 'error', message: artistError.message }), { headers: { "Content-Type": "application/json" } }); + } + + let { data: albumData, error: albumError } = await supabase + .from('albums') + .select('*') + .ilike('key', albumKey) + .single(); + + if (albumError && albumError.code === 'PGRST116') { + const { error: insertAlbumError } = await supabase.from('albums').insert([ + { + mbid: null, + image: `/albums/${albumKey}.jpg`, + key: albumKey, + name: album, + tentative: true + } + ]); + + if (insertAlbumError) { + return new Response(JSON.stringify({ status: 'error', message: insertAlbumError.message }), { headers: { "Content-Type": "application/json" } }); + } + + ({ data: albumData, error: albumError } = await supabase + .from('albums') + .select('*') + .ilike('key', albumKey) + .single()); + } else if (albumError) { + return new Response(JSON.stringify({ status: 'error', message: albumError.message }), { headers: { "Content-Type": "application/json" } }); + } + + const { error: listenError } = await supabase.from('listens').insert([ + { + artist_name: artistData.name_string, + album_name: albumData.name, + track_name: track, + listened_at: listenedAt, + album_key: albumKey + } + ]); + + if (listenError) { + return new Response(JSON.stringify({ status: 'error', message: listenError.message }), { headers: { "Content-Type": "application/json" } }); + } + } + + return new Response(JSON.stringify({ status: 'success' }), { headers: { "Content-Type": "application/json" } }); + } catch (e) { + return new Response(JSON.stringify({ status: 'error', message: e.message }), { headers: { "Content-Type": "application/json" } }); + } + } +}; \ No newline at end of file diff --git a/workers/scrobble/package-lock.json b/workers/scrobble/package-lock.json new file mode 100644 index 00000000..e12fd0a3 --- /dev/null +++ b/workers/scrobble/package-lock.json @@ -0,0 +1,179 @@ +{ + "name": "scrobble-worker", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "dependencies": { + "@supabase/supabase-js": "^2.43.4", + "luxon": "^3.4.4", + "slugify": "^1.6.6" + } + }, + "node_modules/@supabase/auth-js": { + "version": "2.64.2", + "resolved": "https://registry.npmjs.org/@supabase/auth-js/-/auth-js-2.64.2.tgz", + "integrity": "sha512-s+lkHEdGiczDrzXJ1YWt2y3bxRi+qIUnXcgkpLSrId7yjBeaXBFygNjTaoZLG02KNcYwbuZ9qkEIqmj2hF7svw==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14" + } + }, + "node_modules/@supabase/functions-js": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/@supabase/functions-js/-/functions-js-2.3.1.tgz", + "integrity": "sha512-QyzNle/rVzlOi4BbVqxLSH828VdGY1RElqGFAj+XeVypj6+PVtMlD21G8SDnsPQDtlqqTtoGRgdMlQZih5hTuw==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14" + } + }, + "node_modules/@supabase/node-fetch": { + "version": "2.6.15", + "resolved": "https://registry.npmjs.org/@supabase/node-fetch/-/node-fetch-2.6.15.tgz", + "integrity": "sha512-1ibVeYUacxWYi9i0cf5efil6adJ9WRyZBLivgjs+AUpewx1F3xPi7gLgaASI2SmIQxPoCEjAsLAzKPgMJVgOUQ==", + "license": "MIT", + "dependencies": { + "whatwg-url": "^5.0.0" + }, + "engines": { + "node": "4.x || >=6.0.0" + } + }, + "node_modules/@supabase/postgrest-js": { + "version": "1.15.2", + "resolved": "https://registry.npmjs.org/@supabase/postgrest-js/-/postgrest-js-1.15.2.tgz", + "integrity": "sha512-9/7pUmXExvGuEK1yZhVYXPZnLEkDTwxgMQHXLrN5BwPZZm4iUCL1YEyep/Z2lIZah8d8M433mVAUEGsihUj5KQ==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14" + } + }, + "node_modules/@supabase/realtime-js": { + "version": "2.9.5", + "resolved": "https://registry.npmjs.org/@supabase/realtime-js/-/realtime-js-2.9.5.tgz", + "integrity": "sha512-TEHlGwNGGmKPdeMtca1lFTYCedrhTAv3nZVoSjrKQ+wkMmaERuCe57zkC5KSWFzLYkb5FVHW8Hrr+PX1DDwplQ==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14", + "@types/phoenix": "^1.5.4", + "@types/ws": "^8.5.10", + "ws": "^8.14.2" + } + }, + "node_modules/@supabase/storage-js": { + "version": "2.5.5", + "resolved": "https://registry.npmjs.org/@supabase/storage-js/-/storage-js-2.5.5.tgz", + "integrity": "sha512-OpLoDRjFwClwc2cjTJZG8XviTiQH4Ik8sCiMK5v7et0MDu2QlXjCAW3ljxJB5+z/KazdMOTnySi+hysxWUPu3w==", + "license": "MIT", + "dependencies": { + "@supabase/node-fetch": "^2.6.14" + } + }, + "node_modules/@supabase/supabase-js": { + "version": "2.43.4", + "resolved": "https://registry.npmjs.org/@supabase/supabase-js/-/supabase-js-2.43.4.tgz", + "integrity": "sha512-/pLPaxiIsn5Vaz3s32HC6O/VNwfeddnzS0bZRpOW0AKcPuXroD8pT9G8mpiBlZfpKsMmq6k7tlhW7Sr1PAQ1lw==", + "license": "MIT", + "dependencies": { + "@supabase/auth-js": "2.64.2", + "@supabase/functions-js": "2.3.1", + "@supabase/node-fetch": "2.6.15", + "@supabase/postgrest-js": "1.15.2", + "@supabase/realtime-js": "2.9.5", + "@supabase/storage-js": "2.5.5" + } + }, + "node_modules/@types/node": { + "version": "20.14.2", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.14.2.tgz", + "integrity": "sha512-xyu6WAMVwv6AKFLB+e/7ySZVr/0zLCzOa7rSpq6jNwpqOrUbcACDWC+53d4n2QHOnDou0fbIsg8wZu/sxrnI4Q==", + "license": "MIT", + "dependencies": { + "undici-types": "~5.26.4" + } + }, + "node_modules/@types/phoenix": { + "version": "1.6.4", + "resolved": "https://registry.npmjs.org/@types/phoenix/-/phoenix-1.6.4.tgz", + "integrity": "sha512-B34A7uot1Cv0XtaHRYDATltAdKx0BvVKNgYNqE4WjtPUa4VQJM7kxeXcVKaH+KS+kCmZ+6w+QaUdcljiheiBJA==", + "license": "MIT" + }, + "node_modules/@types/ws": { + "version": "8.5.10", + "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.10.tgz", + "integrity": "sha512-vmQSUcfalpIq0R9q7uTo2lXs6eGIpt9wtnLdMv9LVpIjCA/+ufZRozlVoVelIYixx1ugCBKDhn89vnsEGOCx9A==", + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/luxon": { + "version": "3.4.4", + "resolved": "https://registry.npmjs.org/luxon/-/luxon-3.4.4.tgz", + "integrity": "sha512-zobTr7akeGHnv7eBOXcRgMeCP6+uyYsczwmeRCauvpvaAltgNyTbLH/+VaEAPUeWBT+1GuNmz4wC/6jtQzbbVA==", + "license": "MIT", + "engines": { + "node": ">=12" + } + }, + "node_modules/slugify": { + "version": "1.6.6", + "resolved": "https://registry.npmjs.org/slugify/-/slugify-1.6.6.tgz", + "integrity": "sha512-h+z7HKHYXj6wJU+AnS/+IH8Uh9fdcX1Lrhg1/VMdf9PwoBQXFcXiAdsy2tSK0P6gKwJLXp02r90ahUCqHk9rrw==", + "license": "MIT", + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", + "license": "MIT" + }, + "node_modules/undici-types": { + "version": "5.26.5", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz", + "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==", + "license": "MIT" + }, + "node_modules/webidl-conversions": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==", + "license": "BSD-2-Clause" + }, + "node_modules/whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", + "license": "MIT", + "dependencies": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" + } + }, + "node_modules/ws": { + "version": "8.17.0", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.17.0.tgz", + "integrity": "sha512-uJq6108EgZMAl20KagGkzCKfMEjxmKvZHG7Tlq0Z6nOky7YF7aq4mOx6xK8TJ/i1LeK4Qus7INktacctDgY8Ow==", + "license": "MIT", + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": ">=5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + } + } +} diff --git a/workers/scrobble/package.json b/workers/scrobble/package.json new file mode 100644 index 00000000..5e88d1de --- /dev/null +++ b/workers/scrobble/package.json @@ -0,0 +1,7 @@ +{ + "dependencies": { + "@supabase/supabase-js": "^2.43.4", + "luxon": "^3.4.4", + "slugify": "^1.6.6" + } +} diff --git a/workers/scrobble/wrangler.toml b/workers/scrobble/wrangler.toml new file mode 100644 index 00000000..c338cfbc --- /dev/null +++ b/workers/scrobble/wrangler.toml @@ -0,0 +1,12 @@ +name = "scrobble-worker" +main = "./index.js" +compatibility_date = "2023-01-01" + +account_id = "" +workers_dev = true + +[env.production] +name = "scrobble-worker-production" +routes = [ + { pattern = "coryd.dev/api/scrobble*", zone_id = "" } +]