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).
-[](https://social.lol/@cory) [](https://www.buymeacoffee.com/cory) [](https://eleventy.dev)
+[](https://social.lol/@cory) [](https://www.buymeacoffee.com/cory) [](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' %}
{{ 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 = "" }
+]