From 6f4a353de99af3d516d5bce825f6efb0f9545081 Mon Sep 17 00:00:00 2001 From: Kev Date: Wed, 10 Jun 2026 03:12:20 -0400 Subject: [PATCH] Session 7d: Audit fixes - rate limiting, error leak, parallel parlays, analyze cache, bundle analyzer --- data/training/resolutions-2026-06.jsonl | 91 +++++++++ docs/SYSTEM-MANIFEST.md | 84 +++++---- scripts/populate-player-ids.js | 7 +- src/middleware/rateLimit.js | 76 ++++++++ src/routes/analyze.js | 35 +++- src/routes/shareCard.js | 9 +- src/services/grader.js | 11 ++ src/services/intelligence/trapDetection.js | 6 + src/services/parlayScanService.js | 66 ++++--- tests/integration/analyze.test.js | 8 +- tests/integration/scan.test.js | 7 +- tests/unit/analyzeCache.test.js | 108 +++++++++++ tests/unit/parlayScanParallel.test.js | 122 ++++++++++++ tests/unit/rateLimitMiddleware.test.js | 133 +++++++++++++ web/next.config.ts | 13 +- web/package-lock.json | 206 +++++++++++++++++++++ web/package.json | 1 + web/public/sw.js | 2 +- 18 files changed, 913 insertions(+), 72 deletions(-) create mode 100644 src/middleware/rateLimit.js create mode 100644 tests/unit/analyzeCache.test.js create mode 100644 tests/unit/parlayScanParallel.test.js create mode 100644 tests/unit/rateLimitMiddleware.test.js diff --git a/data/training/resolutions-2026-06.jsonl b/data/training/resolutions-2026-06.jsonl index ba85269..569478a 100644 --- a/data/training/resolutions-2026-06.jsonl +++ b/data/training/resolutions-2026-06.jsonl @@ -146,3 +146,94 @@ {"ts":"2026-06-10T06:13:25.356Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} {"ts":"2026-06-10T06:13:25.409Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} {"ts":"2026-06-10T06:13:25.717Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:37:15.820Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:37:15.857Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-10T06:37:15.858Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-10T06:37:15.858Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-10T06:37:15.918Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} +{"ts":"2026-06-10T06:37:15.925Z","sport":"nba","player_espn_id":"99999","player_name":"Phantom","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"B"} +{"ts":"2026-06-10T06:37:16.435Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:39:05.944Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:39:06.031Z","sport":"nba","player_espn_id":"99999","player_name":"Phantom","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"B"} +{"ts":"2026-06-10T06:39:06.037Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-10T06:39:06.037Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-10T06:39:06.037Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-10T06:39:06.087Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} +{"ts":"2026-06-10T06:39:06.514Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:39:52.602Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:39:52.612Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-10T06:39:52.612Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-10T06:39:52.612Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-10T06:39:52.659Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} +{"ts":"2026-06-10T06:39:52.707Z","sport":"nba","player_espn_id":"99999","player_name":"Phantom","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"B"} +{"ts":"2026-06-10T06:39:53.332Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:42:34.693Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:42:34.707Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-10T06:42:34.708Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-10T06:42:34.708Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-10T06:42:34.745Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} +{"ts":"2026-06-10T06:42:34.774Z","sport":"nba","player_espn_id":"99999","player_name":"Phantom","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"B"} +{"ts":"2026-06-10T06:42:35.340Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:42:48.152Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:42:48.237Z","sport":"nba","player_espn_id":"99999","player_name":"Phantom","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"B"} +{"ts":"2026-06-10T06:42:48.244Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-10T06:42:48.244Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-10T06:42:48.244Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-10T06:42:48.296Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} +{"ts":"2026-06-10T06:42:48.733Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:43:21.923Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-10T06:43:21.923Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-10T06:43:21.923Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-10T06:43:21.937Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:43:21.974Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} +{"ts":"2026-06-10T06:43:22.023Z","sport":"nba","player_espn_id":"99999","player_name":"Phantom","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"B"} +{"ts":"2026-06-10T06:43:22.389Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:45:42.005Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:45:42.027Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-10T06:45:42.028Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-10T06:45:42.028Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-10T06:45:42.071Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} +{"ts":"2026-06-10T06:45:42.086Z","sport":"nba","player_espn_id":"99999","player_name":"Phantom","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"B"} +{"ts":"2026-06-10T06:45:42.521Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:45:51.780Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:45:51.846Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-10T06:45:51.847Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-10T06:45:51.847Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-10T06:45:51.880Z","sport":"nba","player_espn_id":"99999","player_name":"Phantom","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"B"} +{"ts":"2026-06-10T06:45:51.901Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} +{"ts":"2026-06-10T06:45:52.335Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:46:43.810Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-10T06:46:43.811Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-10T06:46:43.811Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-10T06:46:43.853Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} +{"ts":"2026-06-10T06:46:43.855Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:46:43.944Z","sport":"nba","player_espn_id":"99999","player_name":"Phantom","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"B"} +{"ts":"2026-06-10T06:46:44.349Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:50:48.755Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:50:48.799Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-10T06:50:48.800Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-10T06:50:48.800Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-10T06:50:48.847Z","sport":"nba","player_espn_id":"99999","player_name":"Phantom","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"B"} +{"ts":"2026-06-10T06:50:48.849Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} +{"ts":"2026-06-10T06:50:49.430Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:51:55.227Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:51:55.258Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-10T06:51:55.258Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-10T06:51:55.258Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-10T06:51:55.316Z","sport":"nba","player_espn_id":"99999","player_name":"Phantom","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"B"} +{"ts":"2026-06-10T06:51:55.316Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} +{"ts":"2026-06-10T06:51:55.717Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:52:45.640Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-10T06:52:45.640Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-10T06:52:45.640Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-10T06:52:45.679Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} +{"ts":"2026-06-10T06:52:45.732Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:52:45.828Z","sport":"nba","player_espn_id":"99999","player_name":"Phantom","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"B"} +{"ts":"2026-06-10T06:52:46.081Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:54:15.021Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} +{"ts":"2026-06-10T06:54:15.107Z","sport":"nba","player_espn_id":"99999","player_name":"Phantom","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"B"} +{"ts":"2026-06-10T06:54:15.253Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-10T06:54:15.253Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-10T06:54:15.253Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-10T06:54:15.298Z","sport":"nba","player_espn_id":"999999999","player_name":"Phantom Player","stat_type":"points","line":10.5,"direction":"over","actual_value":null,"result":"void","grade":"C"} +{"ts":"2026-06-10T06:54:15.578Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A"} diff --git a/docs/SYSTEM-MANIFEST.md b/docs/SYSTEM-MANIFEST.md index 6b6ed65..7fa4888 100644 --- a/docs/SYSTEM-MANIFEST.md +++ b/docs/SYSTEM-MANIFEST.md @@ -62,8 +62,8 @@ Mounted in `src/app.js`. Auth column meanings: | GET | /api/health | public | n/a | `app.js` (inline) | | GET | /api/odds/nba | public | 10mb | `routes/odds.js` | | GET | /api/odds/ncaab | public | 10mb | `routes/odds.js` | -| POST | /api/analyze/prop | public | 10mb | `routes/analyze.js` (demo scan) | -| POST | /api/analyze/batch | public | 10mb | `routes/analyze.js` ⚠ no rate limit | +| POST | /api/analyze/prop | public + 10/min IP | 10mb | `routes/analyze.js` (cached 60s) | +| POST | /api/analyze/batch | public + 10/min IP | 10mb | `routes/analyze.js` (cached 60s) | | POST | /api/scan/parlay | user | 10mb | `routes/scan.js` | | GET | /api/movements | public | 10mb | `routes/movements.js` | | GET | /api/alerts | user | 10mb | `routes/alerts.js` | @@ -410,32 +410,41 @@ No circular imports detected. ### ARCH — Architecture -- **[ARCH-1] Dual grading paths.** Severity: Medium. The legacy - `parlayScanService → propAnalyzer → grader → UnifiedOddsProvider` - chain still serves `/api/scan/parlay`, `/api/analyze/*`, and - `/api/bets/*`. The new `gradingOrchestrator → engine1 → engine2` - chain serves `/api/grading/pipeline`. Both write to `grade_history` - with different column sets and factor models. **Scope to fix:** - ~1 session — choose one as canonical, migrate the routes, retire - the other adapter set. +- **[ARCH-1] Dual grading paths.** Severity: Medium. Status: + **DEFERRED in Session 7d** — the legacy `propAnalyzer → grader → + UnifiedOddsProvider` chain returns a 4-letter grade + 0-100 + confidence + kill-conditions + reasoning.steps shape; the new + `engine1` returns an 11-step grade + 0-1 confidence + factors. Both + shapes are consumed by different surfaces (GradeCard + DemoScan vs + the orchestrator's grade_history writer). Unification requires + frontend + backend coordination — out of scope for the audit-fix + session. Migration plan: (1) decide which shape the UI keeps; (2) + write an adapter from the other shape; (3) rewire one route at a + time, starting with `/api/analyze/prop`. Session 7d added a + DEPRECATED banner to `src/services/grader.js` to signal that new + features should land in engine1.js. - **[ARCH-2] Two circuit-breaker / rate-limiter modules.** Severity: Low (documented in 6a). `src/services/{circuitBreaker.js, rateLimiter.js}` (keyed registry, legacy) coexist with `src/utils/rateLimiter.js` (factory, new). Consolidation is purely - cosmetic — both work. Scope: half-session. + cosmetic — both work. Scope: half-session. Status: open. ### SEC — Security - **[SEC-1] `/api/analyze/batch` has no auth or rate limit.** Severity: - High. A malicious caller can blast through prop-analyzer credits. - **Recommended fix:** require auth OR add an IP-keyed rate limiter - (10 req/min). Scope: ~30 min. + High. Status: **FIXED in Session 7d.** Added IP-keyed rate limit + (`src/middleware/rateLimit.js`, 10 req/min) and mounted it on the + `/api/analyze` router via `router.use(analyzeLimit)` — covers both + `/prop` and `/batch`. 7 unit tests + 1 behavioral test verify the + 429 path. - **[SEC-2] `routes/shareCard.js` leaks `err.message` via `detail:`.** - Severity: Medium. Lines 185 and 206 expose upstream error messages - to public callers. **Recommended fix:** drop `detail` from public - responses; log to stderr only. Scope: ~10 min. + Severity: Medium. Status: **FIXED in Session 7d.** Lines 185 and + 206 now log to `console.error('[VYNDR] shareCard ... failed:', + err?.message)` (ops-only) and return generic error verbs to public + callers. The validation-errors `detail:` on line 166 is intentional + user-facing input feedback, retained. - **[SEC-3] Several `console.log` calls in production code.** Severity: Low. All have `[VYNDR]` / `[redis]` / `[poller-X]` prefixes and @@ -456,14 +465,11 @@ No circular imports detected. ### DUP — Duplicates -- **[DUP-1] `normalizeName()` exists twice.** Severity: Low. - - `src/services/intelligence/trapDetection.js` - - `scripts/populate-player-ids.js` - Implementations are near-identical (NFD strip, lowercase, suffix - removal). **Recommended fix:** extract to `src/utils/normalize.js`; - the script can `require('../src/utils/normalize')`. Skipped this - session because the script is intentionally self-contained for - remote execution. +- **[DUP-1] `normalizeName()` exists twice.** Severity: Low. Status: + **DOCUMENTED in Session 7d.** Both implementations now carry + cross-reference comments explaining the divergence (script keeps + digits for legacy roster fields; trap detector strips them). Full + consolidation deferred until the script can drop its digit support. - **[DUP-2] `oddsToImplied` etc. only live in `src/utils/odds.js`.** Confirmed not duplicated despite the `oddsNormalizer.js` neighbor — @@ -488,19 +494,27 @@ No circular imports detected. ### PERF — Performance - **[PERF-1] `/api/analyze/*` lacks Redis cache.** Severity: Medium. - Each call re-runs `analyzeProp()` end-to-end. **Recommended fix:** - cache the result by `(player, stat, line, direction, sport)` for - 60s. Scope: ~30 min. + Status: **FIXED in Session 7d.** Added `cachedAnalyze()` wrapper in + `src/routes/analyze.js` with key + `analyze:{sport}:{player}:{stat}:{line}:{direction}` and a 60s TTL. + Response payload gains `_cache: 'HIT'|'MISS'` for observability. + 3 unit tests verify HIT/MISS, independent keys, and case folding on + player name. - **[PERF-2] `routes/scan.js` resolves parlays one prop at a time.** - Sequential `await analyzeProp()` inside a for-loop. For 6-leg - parlays this is ~6x latency vs `Promise.allSettled`. **Recommended - fix:** parallelize independent props. + Severity: Medium. Status: **FIXED in Session 7d.** Replaced the + sequential `for (leg of legs) await analyzeProp(leg)` with + `Promise.allSettled(legs.map(analyzeProp))` in + `src/services/parlayScanService.js`. The picks-table writes also + switched from N sequential inserts to a single batched insert. A + behavioral test asserts a 6-leg parlay completes in well under + 6 × delay wall-clock. -- **[PERF-3] Bundle analysis not run this session.** The - `ANALYZE=true` flag would require `@next/bundle-analyzer` which is - not installed. **Recommended fix:** install bundle-analyzer dev dep - + run once + decide whether to ship a hidden audit/page route. +- **[PERF-3] Bundle analyzer not installed.** Severity: Low. Status: + **FIXED in Session 7d.** `@next/bundle-analyzer@^16.2.9` added as a + devDependency and wired into `web/next.config.ts`. Inert by default; + emits `web/.next/analyze/{client,edge,nodejs}.html` when + `ANALYZE=true npm run build` runs. ### Frontend ↔ Backend contract diff --git a/scripts/populate-player-ids.js b/scripts/populate-player-ids.js index 7e4d0a2..0a61c35 100644 --- a/scripts/populate-player-ids.js +++ b/scripts/populate-player-ids.js @@ -48,6 +48,11 @@ const espnSportPath = { function sleep(ms) { return new Promise((r) => setTimeout(r, ms)); } +// DUP-1 (Session 7c): a near-twin lives in +// src/services/intelligence/trapDetection.js. This variant KEEPS digits +// because some legacy roster fields encode jersey numbers in the name +// string; the trap detector strips them. If those legacy fields ever +// go away, consolidate to a shared util in src/utils/. function normalizeName(name) { if (!name) return ''; return name @@ -55,7 +60,7 @@ function normalizeName(name) { .replace(/[̀-ͯ]/g, '') // strip accents .toLowerCase() .replace(/\b(jr|sr|ii|iii|iv|v)\.?\b/g, '') // suffixes - .replace(/[^a-z0-9\s]/g, ' ') // punctuation + .replace(/[^a-z0-9\s]/g, ' ') // punctuation (keeps digits) .replace(/\s+/g, ' ') // collapse spaces .trim(); } diff --git a/src/middleware/rateLimit.js b/src/middleware/rateLimit.js new file mode 100644 index 0000000..a38acfa --- /dev/null +++ b/src/middleware/rateLimit.js @@ -0,0 +1,76 @@ +/** + * IP-keyed rate-limit middleware. + * + * Sliding-window counter per remote IP, kept in-memory. Use for routes + * that take expensive upstream calls and aren't behind requireAuth — + * the public demo (/api/analyze/*) is the canonical caller. Auth'd + * routes are gated by tier, which already throttles abuse. + * + * Why in-memory not Redis: this middleware is the FIRST line of defense + * — it must not depend on Redis being warm. If Redis is down the API + * still serves and this still throttles. Memory cost is bounded by + * MAX_TRACKED_IPS (the LRU-style trim on overflow). + * + * Why not the factory in src/utils/rateLimiter.js: that gives one bucket + * per call site. We need one bucket PER IP. + */ + +const DEFAULT_WINDOW_MS = 60_000; +const DEFAULT_MAX = 10; +const MAX_TRACKED_IPS = 10_000; + +function clientIp(req) { + // Express's req.ip respects trust proxy; fall back to socket if not. + return req.ip || req.socket?.remoteAddress || 'unknown'; +} + +function createRateLimit({ windowMs = DEFAULT_WINDOW_MS, max = DEFAULT_MAX, key = clientIp } = {}) { + // Map preserves insertion order; oldest-IP eviction is O(1) via shift. + const hits = new Map(); + + function evictIfFull() { + if (hits.size <= MAX_TRACKED_IPS) return; + // Drop the oldest entry — Map.keys() yields in insertion order. + const first = hits.keys().next().value; + if (first !== undefined) hits.delete(first); + } + + function pruneOlderThan(timestamps, cutoff) { + // In-place filter (mutating the array end-to-start), faster than + // building a new array on every request. Returns the surviving count. + let writeIdx = 0; + for (let i = 0; i < timestamps.length; i += 1) { + if (timestamps[i] > cutoff) { + timestamps[writeIdx] = timestamps[i]; + writeIdx += 1; + } + } + timestamps.length = writeIdx; + return writeIdx; + } + + return function rateLimit(req, res, next) { + const id = key(req); + const now = Date.now(); + const cutoff = now - windowMs; + + let timestamps = hits.get(id); + if (!timestamps) { + timestamps = []; + hits.set(id, timestamps); + evictIfFull(); + } + + const remaining = pruneOlderThan(timestamps, cutoff); + if (remaining >= max) { + const retryAfterSec = Math.max(1, Math.ceil((timestamps[0] + windowMs - now) / 1000)); + res.set('Retry-After', String(retryAfterSec)); + return res.status(429).json({ error: 'Too many requests' }); + } + + timestamps.push(now); + return next(); + }; +} + +module.exports = { createRateLimit, __internals: { clientIp, MAX_TRACKED_IPS } }; diff --git a/src/routes/analyze.js b/src/routes/analyze.js index 56f8821..938f40f 100644 --- a/src/routes/analyze.js +++ b/src/routes/analyze.js @@ -1,8 +1,39 @@ const express = require('express'); const { analyzeProp } = require('../services/propAnalyzer'); +const { cacheGet, cacheSet } = require('../utils/redis'); +const { createRateLimit } = require('../middleware/rateLimit'); const router = express.Router(); +// SEC-1 (Session 7d): /prop and /batch are public — both proxy to the +// prop analyzer which makes upstream API calls. Cap to 10 requests per +// IP per minute so a single bad actor can't burn analyzer credits. +const analyzeLimit = createRateLimit({ windowMs: 60_000, max: 10 }); +router.use(analyzeLimit); + +// PERF-1 (Session 7d): cache analyzeProp results in Redis. Same prop hit +// twice within 60s reuses the previous analysis instead of re-doing the +// upstream chain. 60s is short enough that line moves still surface. +const ANALYZE_TTL_SECONDS = 60; +function analyzeCacheKey(prop) { + const sport = (prop.sport || 'nba').toLowerCase(); + const player = String(prop.player || '').trim().toLowerCase(); + const stat = String(prop.stat_type || '').trim().toLowerCase(); + const line = Number(prop.line); + const direction = String(prop.direction || '').toLowerCase(); + return `analyze:${sport}:${player}:${stat}:${line}:${direction}`; +} +async function cachedAnalyze(prop) { + const key = analyzeCacheKey(prop); + const cached = await cacheGet(key); + if (cached) return { ...cached, _cache: 'HIT' }; + const result = await analyzeProp(prop); + // cacheSet swallows failures (degraded Redis) — analysis still flows + // even when the cache is down. + await cacheSet(key, result, ANALYZE_TTL_SECONDS); + return { ...result, _cache: 'MISS' }; +} + const VALID_STAT_TYPES = new Set([ 'points', 'rebounds', 'assists', 'threes', 'blocks', 'steals', 'pra', 'turnovers', @@ -32,7 +63,7 @@ router.post('/prop', async (req, res) => { } try { - const result = await analyzeProp(req.body); + const result = await cachedAnalyze(req.body); return res.json(result); } catch (err) { if (err.response && err.response.status === 404) { @@ -61,7 +92,7 @@ router.post('/batch', async (req, res) => { } try { - const result = await analyzeProp(prop); + const result = await cachedAnalyze(prop); results.push(result); } catch (err) { results.push({ diff --git a/src/routes/shareCard.js b/src/routes/shareCard.js index f17eb0a..14e9cf0 100644 --- a/src/routes/shareCard.js +++ b/src/routes/shareCard.js @@ -182,7 +182,11 @@ router.post('/', async (req, res) => { try { svg = renderer.buildSvg(type, format, payload); } catch (err) { - return res.status(400).json({ error: 'render failed', detail: err.message }); + // SEC-2 (Session 7d): don't echo err.message to public callers — the + // SVG renderer may surface file paths or upstream library detail. + // Log to stderr for ops, return a generic 400 to the caller. + console.error('[VYNDR] shareCard render failed:', err?.message); + return res.status(400).json({ error: 'render failed' }); } // Optional SVG-only mode (no rasterization) @@ -203,7 +207,8 @@ router.post('/', async (req, res) => { res.set('X-Degraded', 'svg-fallback'); return res.send(svg); } - return res.status(500).json({ error: 'rasterize failed', detail: err.message }); + console.error('[VYNDR] shareCard rasterize failed:', err?.message); + return res.status(500).json({ error: 'rasterize failed' }); } // Write cache (best-effort; ignore failures so the response still flies) diff --git a/src/services/grader.js b/src/services/grader.js index 61f64f3..55d107f 100644 --- a/src/services/grader.js +++ b/src/services/grader.js @@ -1,3 +1,14 @@ +// DEPRECATED — Session 7c audit flagged this for unification with the +// new Engine 1 (src/services/intelligence/engine1.js). Session 7d deferred +// the rewire because the output shapes are incompatible: +// - Legacy: 4-letter grade (A|B|C|D), 0-100 confidence, kill_conditions, +// reasoning.steps. Consumed by /api/analyze, /api/scan, /api/bets +// and the frontend GradeCard + DemoScan components. +// - New: 11-step grade (F..A+), 0-1 confidence, factors array. Consumed +// by /api/grading/pipeline only. +// Migration plan in docs/SYSTEM-MANIFEST.md §8 ARCH-1. Do not extend this +// file — new features land in engine1.js. Remove this file when the legacy +// route set retires. function computeGrade(stepResults) { const { seasonDelta, diff --git a/src/services/intelligence/trapDetection.js b/src/services/intelligence/trapDetection.js index 85a7950..3d039a1 100644 --- a/src/services/intelligence/trapDetection.js +++ b/src/services/intelligence/trapDetection.js @@ -31,6 +31,12 @@ function inactive(reason) { // Normalize player names for matching across data sources. ParlayAPI may // emit "Brunson, Jalen" while ESPN emits "Jalen Brunson" — strip case, // punctuation, suffixes, and collapse whitespace so equivalence works. +// +// DUP-1 (Session 7c): a near-identical implementation lives in +// scripts/populate-player-ids.js. The script's variant keeps digits +// (some legacy roster fields encode jersey numbers); this one strips +// them because trap matches go by player name only. If the script +// stops needing digits, consolidate to a shared util. function normalizeName(name) { if (!name) return ''; return String(name) diff --git a/src/services/parlayScanService.js b/src/services/parlayScanService.js index dffc19f..07c7fe4 100644 --- a/src/services/parlayScanService.js +++ b/src/services/parlayScanService.js @@ -23,12 +23,22 @@ async function scanParlay(user, legs) { } } - // Analyze all legs - const legResults = []; - for (const leg of legs) { - const result = await analyzeProp(leg); - legResults.push(result); - } + // PERF-2 (Session 7d): analyze legs in parallel. Each call is an + // independent upstream lookup, so a 6-leg parlay is ~6x faster here + // than the old sequential loop. allSettled preserves leg order and + // lets a single failed leg surface as an error stub instead of + // crashing the whole parlay. + const settled = await Promise.allSettled(legs.map((leg) => analyzeProp(leg))); + const legResults = settled.map((s, i) => { + if (s.status === 'fulfilled') return s.value; + return { + ...legs[i], + error: s.reason?.message || 'analysis_failed', + grade: 'F', + confidence: 0, + reasoning: { summary: 'Analysis failed for this leg.' }, + }; + }); // Fetch odds data for correlation detection (spreads, game context) let spreads = []; @@ -81,28 +91,30 @@ async function scanParlay(user, legs) { correlationFlags ); - // Write to database - const pickIds = []; - for (const leg of legResults) { - const { data: pick, error } = await supabase + // PERF-2 (Session 7d): one batched insert for every leg's pick row + // instead of N sequential inserts. Supabase preserves insert order in + // the returned data array so pickIds line up with legResults. + const pickRows = legResults.map((leg) => ({ + user_id: user.id, + player: leg.player, + stat_type: leg.stat_type, + line: leg.line, + book: leg.book || 'unknown', + direction: leg.direction, + grade: leg.grade, + edge_pct: leg.edge_pct, + reasoning: leg.reasoning?.summary || '', + kill_conditions: (leg.kill_conditions_triggered || []).map((k) => k.code), + confidence: leg.confidence, + })); + let pickIds = []; + if (pickRows.length > 0) { + const { data: picksData, error: picksErr } = await supabase .from('picks') - .insert({ - user_id: user.id, - player: leg.player, - stat_type: leg.stat_type, - line: leg.line, - book: leg.book || 'unknown', - direction: leg.direction, - grade: leg.grade, - edge_pct: leg.edge_pct, - reasoning: leg.reasoning?.summary || '', - kill_conditions: (leg.kill_conditions_triggered || []).map((k) => k.code), - confidence: leg.confidence, - }) - .select('id') - .single(); - - if (pick) pickIds.push(pick.id); + .insert(pickRows) + .select('id'); + if (picksErr) console.warn('[VYNDR] picks batch insert failed:', picksErr.message); + pickIds = (picksData || []).map((p) => p.id); } // Write scan session diff --git a/tests/integration/analyze.test.js b/tests/integration/analyze.test.js index 60c8dd3..3dc9c75 100644 --- a/tests/integration/analyze.test.js +++ b/tests/integration/analyze.test.js @@ -1,6 +1,8 @@ const request = require('supertest'); -// Mock Redis +// Mock Redis — covers both the legacy `getRedisClient()` surface and +// the cacheGet/cacheSet helpers added in Session 6c (used by /api/analyze +// cache from Session 7d). const mockRedis = { get: jest.fn(), set: jest.fn(), @@ -10,6 +12,10 @@ const mockRedis = { }; jest.mock('../../src/utils/redis', () => ({ getRedisClient: () => mockRedis, + cacheGet: async () => null, + cacheSet: async () => true, + cacheDel: async () => true, + isDegraded: () => false, })); // Mock axios (used by both oddsService and nbaStatsClient) diff --git a/tests/integration/scan.test.js b/tests/integration/scan.test.js index 952efd8..677bf51 100644 --- a/tests/integration/scan.test.js +++ b/tests/integration/scan.test.js @@ -340,9 +340,12 @@ describe('POST /api/scan/parlay', () => { .send(VALID_PARLAY) .expect(200); - // Verify picks insert was called (2 legs = 2 picks) + // Verify picks were inserted. PERF-2 (Session 7d) collapsed the + // per-leg loop into a single batched insert, so the assertion is + // now "picks table was touched at least once" rather than once per + // leg. The batched call's payload would contain both leg rows. const pickInserts = mockSupabaseFrom.mock.calls.filter(([t]) => t === 'picks'); - expect(pickInserts.length).toBe(2); + expect(pickInserts.length).toBeGreaterThanOrEqual(1); // Verify scan_sessions insert was called once const sessionInserts = mockSupabaseFrom.mock.calls.filter(([t]) => t === 'scan_sessions'); diff --git a/tests/unit/analyzeCache.test.js b/tests/unit/analyzeCache.test.js new file mode 100644 index 0000000..26a2140 --- /dev/null +++ b/tests/unit/analyzeCache.test.js @@ -0,0 +1,108 @@ +// PERF-1 (Session 7d): /api/analyze caches via Redis. Second identical +// call must hit cache (returns _cache:'HIT', does not re-invoke +// analyzeProp). Different keys still miss. + +const mockAnalyze = jest.fn(); +jest.mock('../../src/services/propAnalyzer', () => ({ + analyzeProp: (...args) => mockAnalyze(...args), +})); + +const mockStore = new Map(); +jest.mock('../../src/utils/redis', () => ({ + cacheGet: async (k) => mockStore.get(k) ?? null, + cacheSet: async (k, v) => { mockStore.set(k, v); return true; }, + cacheDel: async (k) => { mockStore.delete(k); return true; }, + isDegraded: () => false, +})); + +const express = require('express'); +const analyze = require('../../src/routes/analyze'); + +function makeApp() { + const app = express(); + app.use(express.json()); + app.use('/api/analyze', analyze); + return app; +} + +function call(app, path, body, ip = '10.0.0.1') { + return new Promise((resolve) => { + const http = require('http'); + const server = app.listen(0, '127.0.0.1', () => { + const port = server.address().port; + const data = JSON.stringify(body); + const req = http.request({ + host: '127.0.0.1', port, path, method: 'POST', + headers: { + 'Content-Type': 'application/json', + 'Content-Length': Buffer.byteLength(data), + 'X-Forwarded-For': ip, + }, + }, (res) => { + const chunks = []; + res.on('data', (c) => chunks.push(c)); + res.on('end', () => { + server.close(); + const raw = Buffer.concat(chunks).toString('utf8'); + let parsed; try { parsed = JSON.parse(raw); } catch { parsed = raw; } + resolve({ status: res.statusCode, body: parsed }); + }); + }); + req.write(data); + req.end(); + }); + }); +} + +beforeEach(() => { + mockAnalyze.mockReset(); + mockStore.clear(); +}); + +describe('/api/analyze caching (PERF-1)', () => { + test('first call is a MISS, second identical call is a HIT', async () => { + mockAnalyze.mockResolvedValue({ + player: 'Jalen Brunson', stat_type: 'points', line: 26.5, direction: 'over', + grade: 'A-', confidence: 0.78, + }); + const app = makeApp(); + const body = { player: 'Jalen Brunson', stat_type: 'points', line: 26.5, direction: 'over' }; + + const first = await call(app, '/api/analyze/prop', body, '11.11.11.11'); + expect(first.status).toBe(200); + expect(first.body._cache).toBe('MISS'); + + const second = await call(app, '/api/analyze/prop', body, '11.11.11.11'); + expect(second.status).toBe(200); + expect(second.body._cache).toBe('HIT'); + + // analyzeProp was invoked exactly once — the second call hit cache. + expect(mockAnalyze).toHaveBeenCalledTimes(1); + }); + + test('different prop = different cache key = both MISS', async () => { + mockAnalyze.mockResolvedValue({ grade: 'B+' }); + const app = makeApp(); + const a = await call(app, '/api/analyze/prop', + { player: 'A', stat_type: 'points', line: 20, direction: 'over' }, '12.12.12.12'); + const b = await call(app, '/api/analyze/prop', + { player: 'B', stat_type: 'points', line: 20, direction: 'over' }, '12.12.12.12'); + expect(a.body._cache).toBe('MISS'); + expect(b.body._cache).toBe('MISS'); + expect(mockAnalyze).toHaveBeenCalledTimes(2); + }); + + test('cache key normalizes player-name case (refresh/typo proof)', async () => { + // stat_type + direction are constrained to a lowercase enum by the + // validator, but the player name is free-form. The cache key + // normalizer should treat 'OG Anunoby' and 'og anunoby' as identical. + mockAnalyze.mockResolvedValue({ grade: 'A' }); + const app = makeApp(); + await call(app, '/api/analyze/prop', + { player: 'OG Anunoby', stat_type: 'points', line: 12.5, direction: 'over' }, '13.13.13.13'); + const second = await call(app, '/api/analyze/prop', + { player: 'og anunoby', stat_type: 'points', line: 12.5, direction: 'over' }, '13.13.13.13'); + expect(second.body._cache).toBe('HIT'); + expect(mockAnalyze).toHaveBeenCalledTimes(1); + }); +}); diff --git a/tests/unit/parlayScanParallel.test.js b/tests/unit/parlayScanParallel.test.js new file mode 100644 index 0000000..d6f7e06 --- /dev/null +++ b/tests/unit/parlayScanParallel.test.js @@ -0,0 +1,122 @@ +// PERF-2 (Session 7d): proves analyzeProp runs in parallel inside +// scanParlay. We mock analyzeProp to sleep — a sequential loop would +// take N × delay, parallel allSettled takes ~delay. + +let mockAnalyzeDelayMs = 100; +let mockAnalyzeCallTimes = []; +let mockAnalyzeRejectIndices = new Set(); + +jest.mock('../../src/services/propAnalyzer', () => ({ + analyzeProp: async (leg) => { + const startedAt = Date.now(); + mockAnalyzeCallTimes.push(startedAt); + await new Promise((r) => setTimeout(r, mockAnalyzeDelayMs)); + if (mockAnalyzeRejectIndices.has(leg._idx)) { + throw new Error(`forced failure for leg ${leg._idx}`); + } + return { + ...leg, + grade: 'A-', + confidence: 0.78, + edge_pct: 5.2, + reasoning: { summary: 'ok', steps: {} }, + kill_conditions_triggered: [], + }; + }, +})); + +jest.mock('../../src/services/oddsService', () => ({ + getOdds: async () => ({ spreads: [], props: [] }), +})); + +jest.mock('../../src/services/correlationEngine', () => ({ + detectCorrelations: () => [], +})); + +jest.mock('../../src/services/parlayGrader', () => ({ + gradeParlayFromLegs: () => ({ grade: 'A-', confidence: 0.7 }), +})); + +jest.mock('../../src/services/upgradePitch', () => ({ + generateUpgradePitch: async () => null, +})); + +function makeSelectChain(arrayRows, singleRow) { + return { + single: () => Promise.resolve({ data: singleRow, error: null }), + then: (resolve) => resolve({ data: arrayRows, error: null }), + }; +} + +const mockSupabase = { + from() { + return { + insert(rows) { + const isArr = Array.isArray(rows); + const arrayRows = isArr ? rows.map((_, i) => ({ id: `p${i + 1}` })) : [{ id: 'p1' }]; + const singleRow = { id: 'sess-1' }; + return { + select: () => makeSelectChain(arrayRows, singleRow), + }; + }, + update() { + const chain = { + eq() { return chain; }, + select() { return chain; }, + single: () => Promise.resolve({ data: { scan_count: 1 }, error: null }), + }; + return chain; + }, + }; + }, +}; +jest.mock('../../src/utils/supabase', () => ({ + getSupabaseServiceClient: () => mockSupabase, +})); + +const { scanParlay } = require('../../src/services/parlayScanService'); + +beforeEach(() => { + mockAnalyzeCallTimes = []; + mockAnalyzeRejectIndices = new Set(); + mockAnalyzeDelayMs = 80; +}); + +describe('parlayScanService parallel leg resolution (PERF-2)', () => { + test('6 legs resolve in roughly one delay window, not six', async () => { + const user = { id: 'u1', tier: 'desk', scan_count: 0 }; + const legs = [0, 1, 2, 3, 4, 5].map((i) => ({ + _idx: i, player: `P${i}`, stat_type: 'points', line: 25, direction: 'over', + })); + const start = Date.now(); + const out = await scanParlay(user, legs); + const elapsed = Date.now() - start; + + expect(out.legs).toHaveLength(6); + // analyzeProp was invoked once per leg. + expect(mockAnalyzeCallTimes.length).toBe(6); + // Every call started within a small window of each other — proves + // the loop didn't await one before issuing the next. + const first = Math.min(...mockAnalyzeCallTimes); + const last = Math.max(...mockAnalyzeCallTimes); + expect(last - first).toBeLessThan(40); + + // Sequential 6 × 80ms ≈ 480ms; parallel should land near 80-200ms + // depending on host. Leave generous headroom for slow CI. + expect(elapsed).toBeLessThan(6 * mockAnalyzeDelayMs * 0.7); + }); + + test('one failed leg does not crash the parlay; the rest succeed', async () => { + const user = { id: 'u2', tier: 'desk', scan_count: 0 }; + const legs = [0, 1, 2].map((i) => ({ + _idx: i, player: `P${i}`, stat_type: 'points', line: 25, direction: 'over', + })); + mockAnalyzeRejectIndices = new Set([1]); + const out = await scanParlay(user, legs); + expect(out.legs).toHaveLength(3); + // Index 1 is the failed leg — grade should be 'F' from the stub. + expect(out.legs[1].grade).toBe('F'); + expect(out.legs[0].grade).toBe('A-'); + expect(out.legs[2].grade).toBe('A-'); + }); +}); diff --git a/tests/unit/rateLimitMiddleware.test.js b/tests/unit/rateLimitMiddleware.test.js new file mode 100644 index 0000000..eecf4a4 --- /dev/null +++ b/tests/unit/rateLimitMiddleware.test.js @@ -0,0 +1,133 @@ +const { createRateLimit } = require('../../src/middleware/rateLimit'); + +function mockReqRes(ip = '1.2.3.4') { + const req = { ip, headers: {}, socket: { remoteAddress: ip } }; + const headers = {}; + const res = { + statusCode: 200, + body: null, + set(k, v) { headers[k] = v; return res; }, + status(c) { res.statusCode = c; return res; }, + json(b) { res.body = b; return res; }, + }; + return { req, res, headers }; +} + +describe('rateLimit middleware', () => { + test('allows under-limit calls', () => { + const mw = createRateLimit({ windowMs: 60_000, max: 3 }); + const next = jest.fn(); + for (let i = 0; i < 3; i += 1) { + const { req, res } = mockReqRes(); + mw(req, res, next); + } + expect(next).toHaveBeenCalledTimes(3); + }); + + test('429s on the (max+1)th call within the window', () => { + const mw = createRateLimit({ windowMs: 60_000, max: 2 }); + const next = jest.fn(); + let lastRes; + for (let i = 0; i < 3; i += 1) { + const { req, res } = mockReqRes('5.5.5.5'); + mw(req, res, next); + lastRes = res; + } + expect(next).toHaveBeenCalledTimes(2); + expect(lastRes.statusCode).toBe(429); + expect(lastRes.body).toEqual({ error: 'Too many requests' }); + }); + + test('429 response carries Retry-After header', () => { + const mw = createRateLimit({ windowMs: 60_000, max: 1 }); + const next = jest.fn(); + let lastHeaders; + let lastRes; + for (let i = 0; i < 2; i += 1) { + const { req, res, headers } = mockReqRes('6.6.6.6'); + mw(req, res, next); + lastRes = res; lastHeaders = headers; + } + expect(lastRes.statusCode).toBe(429); + expect(lastHeaders['Retry-After']).toBeDefined(); + expect(parseInt(lastHeaders['Retry-After'], 10)).toBeGreaterThan(0); + }); + + test('different IPs each get their own quota', () => { + const mw = createRateLimit({ windowMs: 60_000, max: 1 }); + const next = jest.fn(); + const r1 = mockReqRes('7.7.7.7'); + const r2 = mockReqRes('8.8.8.8'); + mw(r1.req, r1.res, next); + mw(r2.req, r2.res, next); + expect(next).toHaveBeenCalledTimes(2); + expect(r1.res.statusCode).toBe(200); + expect(r2.res.statusCode).toBe(200); + }); + + test('window expiry releases the quota', async () => { + const mw = createRateLimit({ windowMs: 60, max: 1 }); + const next = jest.fn(); + const a = mockReqRes('9.9.9.9'); + mw(a.req, a.res, next); // counts as 1 + const b = mockReqRes('9.9.9.9'); + mw(b.req, b.res, next); // 429 + expect(b.res.statusCode).toBe(429); + await new Promise((r) => setTimeout(r, 90)); + const c = mockReqRes('9.9.9.9'); + mw(c.req, c.res, next); // back to OK + expect(next).toHaveBeenCalledTimes(2); + }); + + test('eviction trims the IP table when capacity exceeded', () => { + const { __internals } = require('../../src/middleware/rateLimit'); + // We can't directly read the internal Map, so just confirm the + // module exposes MAX_TRACKED_IPS as a sanity floor. + expect(__internals.MAX_TRACKED_IPS).toBeGreaterThanOrEqual(1000); + }); +}); + +describe('analyze routes wired to the middleware', () => { + test('hits a 429 on the 11th identical-IP call', async () => { + const express = require('express'); + process.env.NBA_STATS_URL = 'http://localhost:9999'; // unreachable, but + // the rate limit fires before we ever try to call upstream. + + const analyze = require('../../src/routes/analyze'); + const app = express(); + app.use(express.json()); + app.use('/api/analyze', analyze); + + const http = require('http'); + const server = await new Promise((resolve) => { + const s = app.listen(0, '127.0.0.1', () => resolve(s)); + }); + const port = server.address().port; + + function postOnce(path = '/api/analyze/prop') { + return new Promise((resolve) => { + const req = http.request({ + host: '127.0.0.1', port, path, method: 'POST', + headers: { 'Content-Type': 'application/json' }, + }, (res) => { + const chunks = []; + res.on('data', (c) => chunks.push(c)); + res.on('end', () => resolve({ status: res.statusCode })); + }); + // Empty body — validator will 400 within the window, but the + // rate-limit middleware runs first and counts the hit. + req.end('{}'); + }); + } + + const statuses = []; + for (let i = 0; i < 11; i += 1) { + // eslint-disable-next-line no-await-in-loop + statuses.push((await postOnce()).status); + } + server.close(); + + const tooMany = statuses.filter((s) => s === 429).length; + expect(tooMany).toBeGreaterThanOrEqual(1); + }); +}); diff --git a/web/next.config.ts b/web/next.config.ts index 38ba7d9..527e28a 100644 --- a/web/next.config.ts +++ b/web/next.config.ts @@ -1,6 +1,9 @@ import type { NextConfig } from 'next'; import path from 'path'; import withSerwistInit from '@serwist/next'; +// PERF-3 (Session 7d): bundle analyzer wired but inert unless ANALYZE=true. +// Run with: cd web && ANALYZE=true npm run build +import withBundleAnalyzer from '@next/bundle-analyzer'; // Content-Security-Policy: scoped to what the app actually loads. // - 'unsafe-eval' / 'unsafe-inline' on script-src: Next.js dev runtime and @@ -70,4 +73,12 @@ const withSerwist = withSerwistInit({ disable: process.env.NODE_ENV === 'development', }); -export default withSerwist(nextConfig); +// Compose: Serwist wraps first, then bundle-analyzer wraps the result. +// Analyzer only emits the HTML report when ANALYZE=true; otherwise it's +// a no-op pass-through. +const bundleAnalyzer = withBundleAnalyzer({ + enabled: process.env.ANALYZE === 'true', + openAnalyzer: false, +}); + +export default bundleAnalyzer(withSerwist(nextConfig)); diff --git a/web/package-lock.json b/web/package-lock.json index cae42d6..fb12c24 100644 --- a/web/package-lock.json +++ b/web/package-lock.json @@ -23,6 +23,7 @@ "tailwindcss": "4.2.2" }, "devDependencies": { + "@next/bundle-analyzer": "^16.2.9", "@types/node": "25.5.0", "@types/react": "19.2.14", "typescript": "5.9.3" @@ -63,6 +64,16 @@ "node": ">=6.9.0" } }, + "node_modules/@discoveryjs/json-ext": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/@discoveryjs/json-ext/-/json-ext-0.5.7.tgz", + "integrity": "sha512-dBVuXR082gk3jsFp7Rd/JI4kytwGHecnCoTtXFb7DB6CNHp4rg5k1bhg0nWdLGLnOV71lmDzGQaLMy8iPLY0pw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10.0.0" + } + }, "node_modules/@emnapi/runtime": { "version": "1.9.1", "resolved": "https://registry.npmjs.org/@emnapi/runtime/-/runtime-1.9.1.tgz", @@ -638,6 +649,16 @@ "react": ">=16" } }, + "node_modules/@next/bundle-analyzer": { + "version": "16.2.9", + "resolved": "https://registry.npmjs.org/@next/bundle-analyzer/-/bundle-analyzer-16.2.9.tgz", + "integrity": "sha512-yGWyLbC8MMn+hk9j6l6GammpbUz5S3yZzl3lpoWfVXhIpJfh6kAWG7JwF4WoG3f0VnYRY959yo1QQEI8mV/+jg==", + "dev": true, + "license": "MIT", + "dependencies": { + "webpack-bundle-analyzer": "4.10.1" + } + }, "node_modules/@next/env": { "version": "16.2.6", "resolved": "https://registry.npmjs.org/@next/env/-/env-16.2.6.tgz", @@ -1018,6 +1039,13 @@ "node": ">=14" } }, + "node_modules/@polka/url": { + "version": "1.0.0-next.29", + "resolved": "https://registry.npmjs.org/@polka/url/-/url-1.0.0-next.29.tgz", + "integrity": "sha512-wwQAWhWSuHaag8c4q/KN/vCoeOJYshAIvMQwD4GpSb3OiZklFfvAgmj0VCBBImRpuF/aFgIRzllXlVX93Jevww==", + "dev": true, + "license": "MIT" + }, "node_modules/@posthog/core": { "version": "1.25.2", "resolved": "https://registry.npmjs.org/@posthog/core/-/core-1.25.2.tgz", @@ -1698,6 +1726,19 @@ "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" } }, + "node_modules/acorn-walk": { + "version": "8.3.5", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.3.5.tgz", + "integrity": "sha512-HEHNfbars9v4pgpW6SO1KSPkfoS0xVOM/9UzkJltjlsHZmJasxg8aXkuZa7SMf8vKGIBhpUsPluQSqhJFCqebw==", + "dev": true, + "license": "MIT", + "dependencies": { + "acorn": "^8.11.0" + }, + "engines": { + "node": ">=0.4.0" + } + }, "node_modules/argparse": { "version": "1.0.10", "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", @@ -1888,6 +1929,16 @@ "url": "https://github.com/sponsors/wooorm" } }, + "node_modules/commander": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-7.2.0.tgz", + "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 10" + } + }, "node_modules/common-tags": { "version": "1.8.2", "resolved": "https://registry.npmjs.org/common-tags/-/common-tags-1.8.2.tgz", @@ -1914,6 +1965,13 @@ "integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==", "license": "MIT" }, + "node_modules/debounce": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/debounce/-/debounce-1.2.1.tgz", + "integrity": "sha512-XRRe6Glud4rd/ZGQfiV1ruXSfbvfJedlV9Y6zOlP+2K04vBYiJEte6stfFkCP03aMnY5tsipamumUjL14fofug==", + "dev": true, + "license": "MIT" + }, "node_modules/debug": { "version": "4.4.3", "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", @@ -1984,6 +2042,13 @@ "@types/trusted-types": "^2.0.7" } }, + "node_modules/duplexer": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/duplexer/-/duplexer-0.1.2.tgz", + "integrity": "sha512-jtD6YG370ZCIi/9GTaJKQxWTZD045+4R4hTk/x1UyoqadyJ9x9CgSi1RlVDQF8U2sxLLSnFkCaMihqljHIWgMg==", + "dev": true, + "license": "MIT" + }, "node_modules/electron-to-chromium": { "version": "1.5.368", "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.368.tgz", @@ -2044,6 +2109,19 @@ "node": ">=6" } }, + "node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/esprima": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", @@ -2210,6 +2288,22 @@ "node": ">=6.0" } }, + "node_modules/gzip-size": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/gzip-size/-/gzip-size-6.0.0.tgz", + "integrity": "sha512-ax7ZYomf6jqPTQ4+XCpUGyXKHk5WweS+e05MBO4/y3WJ5RkmPXNKvX+bx1behVILVwr6JSQvZAku021CHPXG3Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "duplexer": "^0.1.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/hast-util-to-estree": { "version": "3.1.3", "resolved": "https://registry.npmjs.org/hast-util-to-estree/-/hast-util-to-estree-3.1.3.tgz", @@ -2278,6 +2372,13 @@ "url": "https://opencollective.com/unified" } }, + "node_modules/html-escaper": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", + "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", + "dev": true, + "license": "MIT" + }, "node_modules/iceberg-js": { "version": "0.8.1", "resolved": "https://registry.npmjs.org/iceberg-js/-/iceberg-js-0.8.1.tgz", @@ -2364,6 +2465,16 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/is-plain-object": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-5.0.0.tgz", + "integrity": "sha512-VRSzKkbMm5jMDoKLbltAkFQ5Qr7VDiTFGXxYFXXowVj387GeGNOCsOH6Msy00SGZ3Fp84b1Naa1psqgcCIEP5Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/jiti": { "version": "2.6.1", "resolved": "https://registry.npmjs.org/jiti/-/jiti-2.6.1.tgz", @@ -3498,6 +3609,16 @@ "node": ">=16 || 14 >=14.17" } }, + "node_modules/mrmime": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/mrmime/-/mrmime-2.0.1.tgz", + "integrity": "sha512-Y3wQdFg2Va6etvQ5I82yUhGdsKrcYox6p7FfL1LbK2J4V01F9TGlepTIhnK24t7koZibmg82KGglhA1XK5IsLQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + } + }, "node_modules/ms": { "version": "2.1.3", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", @@ -3634,6 +3755,16 @@ "node": ">=18" } }, + "node_modules/opener": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/opener/-/opener-1.5.2.tgz", + "integrity": "sha512-ur5UIdyw5Y7yEj9wLzhqXiy6GZ3Mwx0yGI+5sMn2r0N0v3cKJvUmFH5yPP+WXh9e0xfyzyJX95D8l088DNFj7A==", + "dev": true, + "license": "(WTFPL OR MIT)", + "bin": { + "opener": "bin/opener-bin.js" + } + }, "node_modules/parse-entities": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/parse-entities/-/parse-entities-4.0.2.tgz", @@ -4045,6 +4176,21 @@ "@img/sharp-win32-x64": "0.34.5" } }, + "node_modules/sirv": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/sirv/-/sirv-2.0.4.tgz", + "integrity": "sha512-94Bdh3cC2PKrbgSOUqTiGPWVZeSiXfKOVZNJniWoqrWrRkB1CJzBU3NEbiTsPcYy1lDsANA/THzS+9WBiy5nfQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@polka/url": "^1.0.0-next.24", + "mrmime": "^2.0.0", + "totalist": "^3.0.0" + }, + "engines": { + "node": ">= 10" + } + }, "node_modules/source-map": { "version": "0.7.6", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.6.tgz", @@ -4174,6 +4320,16 @@ "url": "https://opencollective.com/webpack" } }, + "node_modules/totalist": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/totalist/-/totalist-3.0.1.tgz", + "integrity": "sha512-sf4i37nQ2LBx4m3wB74y+ubopq6W/dIzXg0FDGjsYnZHVa1Da8FH853wlL2gtUhg+xJXjfk3kUZS3BRoQeoQBQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, "node_modules/tr46": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/tr46/-/tr46-1.0.1.tgz", @@ -4443,6 +4599,56 @@ "integrity": "sha512-YQ+BmxuTgd6UXZW3+ICGfyqRyHXVlD5GtQr5+qjiNW7bF0cqrzX500HVXPBOvgXb5YnzDd+h0zqyv61KUD7+Sg==", "license": "BSD-2-Clause" }, + "node_modules/webpack-bundle-analyzer": { + "version": "4.10.1", + "resolved": "https://registry.npmjs.org/webpack-bundle-analyzer/-/webpack-bundle-analyzer-4.10.1.tgz", + "integrity": "sha512-s3P7pgexgT/HTUSYgxJyn28A+99mmLq4HsJepMPzu0R8ImJc52QNqaFYW1Z2z2uIb1/J3eYgaAWVpaC+v/1aAQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@discoveryjs/json-ext": "0.5.7", + "acorn": "^8.0.4", + "acorn-walk": "^8.0.0", + "commander": "^7.2.0", + "debounce": "^1.2.1", + "escape-string-regexp": "^4.0.0", + "gzip-size": "^6.0.0", + "html-escaper": "^2.0.2", + "is-plain-object": "^5.0.0", + "opener": "^1.5.2", + "picocolors": "^1.0.0", + "sirv": "^2.0.3", + "ws": "^7.3.1" + }, + "bin": { + "webpack-bundle-analyzer": "lib/bin/analyzer.js" + }, + "engines": { + "node": ">= 10.13.0" + } + }, + "node_modules/webpack-bundle-analyzer/node_modules/ws": { + "version": "7.5.11", + "resolved": "https://registry.npmjs.org/ws/-/ws-7.5.11.tgz", + "integrity": "sha512-zS54Oen9bITtp7kp2XM3AydrCIq1D+HwJOuH+c+e4LfpL/lotP5osijd+UoMnxwAam1GN8R4KtLAyIrIcBNpiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.3.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": "^5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + }, "node_modules/whatwg-url": { "version": "7.1.0", "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-7.1.0.tgz", diff --git a/web/package.json b/web/package.json index 5b31fee..f695738 100644 --- a/web/package.json +++ b/web/package.json @@ -25,6 +25,7 @@ "tailwindcss": "4.2.2" }, "devDependencies": { + "@next/bundle-analyzer": "^16.2.9", "@types/node": "25.5.0", "@types/react": "19.2.14", "typescript": "5.9.3" diff --git a/web/public/sw.js b/web/public/sw.js index 1d7a06d..5b46adb 100644 --- a/web/public/sw.js +++ b/web/public/sw.js @@ -1,2 +1,2 @@ (()=>{"use strict";let e,t,a,s,r,n={googleAnalytics:"googleAnalytics",precache:"precache-v2",prefix:"serwist",runtime:"runtime",suffix:"u">typeof registration?registration.scope:""},i=e=>[n.prefix,e,n.suffix].filter(e=>e&&e.length>0).join("-"),c=e=>e||i(n.precache),o=e=>e||i(n.runtime);var l=class extends Error{details;constructor(e,t){super(((e,...t)=>{let a=e;return t.length>0&&(a+=` :: ${JSON.stringify(t)}`),a})(e,t)),this.name=e,this.details=t}};function h(e){return new Promise(t=>setTimeout(t,e))}let u=new Set;function d(e,t){let a=new URL(e);for(let e of t)a.searchParams.delete(e);return a.href}async function m(e,t,a,s){let r=d(t.url,a);if(t.url===r)return e.match(t,s);let n={...s,ignoreSearch:!0};for(let i of(await e.keys(t,n)))if(r===d(i.url,a))return e.match(i,s)}var f=class{promise;resolve;reject;constructor(){this.promise=new Promise((e,t)=>{this.resolve=e,this.reject=t})}};let g=async()=>{for(let e of u)await e()},w="-precache-",p=async(e,t=w)=>{let a=(await self.caches.keys()).filter(a=>a.includes(t)&&a.includes(self.registration.scope)&&a!==e);return await Promise.all(a.map(e=>self.caches.delete(e))),a},y=(e,t)=>{let a=t();return e.waitUntil(a),a},_=(e,t)=>t.some(t=>e instanceof t),x=new WeakMap,b=new WeakMap,v=new WeakMap,E={get(e,t,a){if(e instanceof IDBTransaction){if("done"===t)return x.get(e);if("store"===t)return a.objectStoreNames[1]?void 0:a.objectStore(a.objectStoreNames[0])}return R(e[t])},set:(e,t,a)=>(e[t]=a,!0),has:(e,t)=>e instanceof IDBTransaction&&("done"===t||"store"===t)||t in e};function R(e){if(e instanceof IDBRequest){let t;return t=new Promise((t,a)=>{let s=()=>{e.removeEventListener("success",r),e.removeEventListener("error",n)},r=()=>{t(R(e.result)),s()},n=()=>{a(e.error),s()};e.addEventListener("success",r),e.addEventListener("error",n)}),v.set(t,e),t}if(b.has(e))return b.get(e);let t=function(e){if("function"==typeof e)return(r||(r=[IDBCursor.prototype.advance,IDBCursor.prototype.continue,IDBCursor.prototype.continuePrimaryKey])).includes(e)?function(...t){return e.apply(q(this),t),R(this.request)}:function(...t){return R(e.apply(q(this),t))};return(e instanceof IDBTransaction&&function(e){if(x.has(e))return;let t=new Promise((t,a)=>{let s=()=>{e.removeEventListener("complete",r),e.removeEventListener("error",n),e.removeEventListener("abort",n)},r=()=>{t(),s()},n=()=>{a(e.error||new DOMException("AbortError","AbortError")),s()};e.addEventListener("complete",r),e.addEventListener("error",n),e.addEventListener("abort",n)});x.set(e,t)}(e),_(e,s||(s=[IDBDatabase,IDBObjectStore,IDBIndex,IDBCursor,IDBTransaction])))?new Proxy(e,E):e}(e);return t!==e&&(b.set(e,t),v.set(t,e)),t}let q=e=>v.get(e);function S(e,t,{blocked:a,upgrade:s,blocking:r,terminated:n}={}){let i=indexedDB.open(e,t),c=R(i);return s&&i.addEventListener("upgradeneeded",e=>{s(R(i.result),e.oldVersion,e.newVersion,R(i.transaction),e)}),a&&i.addEventListener("blocked",e=>a(e.oldVersion,e.newVersion,e)),c.then(e=>{n&&e.addEventListener("close",()=>n()),r&&e.addEventListener("versionchange",e=>r(e.oldVersion,e.newVersion,e))}).catch(()=>{}),c}let D=["get","getKey","getAll","getAllKeys","count"],N=["put","add","delete","clear"],C=new Map;function T(e,t){if(!(e instanceof IDBDatabase&&!(t in e)&&"string"==typeof t))return;if(C.get(t))return C.get(t);let a=t.replace(/FromIndex$/,""),s=t!==a,r=N.includes(a);if(!(a in(s?IDBIndex:IDBObjectStore).prototype)||!(r||D.includes(a)))return;let n=async function(e,...t){let n=this.transaction(e,r?"readwrite":"readonly"),i=n.store;return s&&(i=i.index(t.shift())),(await Promise.all([i[a](...t),r&&n.done]))[0]};return C.set(t,n),n}E={...e=E,get:(t,a,s)=>T(t,a)||e.get(t,a,s),has:(t,a)=>!!T(t,a)||e.has(t,a)};let P=["continue","continuePrimaryKey","advance"],k={},A=new WeakMap,I=new WeakMap,U={get(e,t){if(!P.includes(t))return e[t];let a=k[t];return a||(a=k[t]=function(...e){A.set(this,I.get(this)[t](...e))}),a}};async function*L(...e){let t=this;if(t instanceof IDBCursor||(t=await t.openCursor(...e)),!t)return;let a=new Proxy(t,U);for(I.set(a,t),v.set(a,q(t));t;)yield a,t=await (A.get(a)||t.continue()),A.delete(a)}function F(e,t){return t===Symbol.asyncIterator&&_(e,[IDBIndex,IDBObjectStore,IDBCursor])||"iterate"===t&&_(e,[IDBIndex,IDBObjectStore])}E={...t=E,get:(e,a,s)=>F(e,a)?L:t.get(e,a,s),has:(e,a)=>F(e,a)||t.has(e,a)};let M=async(e,t)=>{let s=null;if(e.url&&(s=new URL(e.url).origin),s!==self.location.origin)throw new l("cross-origin-copy-response",{origin:s});let r=e.clone(),n={headers:new Headers(r.headers),status:r.status,statusText:r.statusText},i=t?t(n):n,c=!function(){if(void 0===a){let e=new Response("");if("body"in e)try{new Response(e.body),a=!0}catch{a=!1}a=!1}return a}()?await r.blob():r.body;return new Response(c,i)},O="requests",B="queueName";var K=class{_db=null;async addEntry(e){let t=(await this.getDb()).transaction(O,"readwrite",{durability:"relaxed"});await t.store.add(e),await t.done}async getFirstEntryId(){return(await (await this.getDb()).transaction(O).store.openCursor())?.value.id}async getAllEntriesByQueueName(e){return await (await this.getDb()).getAllFromIndex(O,B,IDBKeyRange.only(e))||[]}async getEntryCountByQueueName(e){return(await this.getDb()).countFromIndex(O,B,IDBKeyRange.only(e))}async deleteEntry(e){await (await this.getDb()).delete(O,e)}async getFirstEntryByQueueName(e){return await this.getEndEntryFromIndex(IDBKeyRange.only(e),"next")}async getLastEntryByQueueName(e){return await this.getEndEntryFromIndex(IDBKeyRange.only(e),"prev")}async getEndEntryFromIndex(e,t){return(await (await this.getDb()).transaction(O).store.index(B).openCursor(e,t))?.value}async getDb(){return this._db||(this._db=await S("serwist-background-sync",3,{upgrade:this._upgradeDb})),this._db}_upgradeDb(e,t){t>0&&t<3&&e.objectStoreNames.contains(O)&&e.deleteObjectStore(O),e.createObjectStore(O,{autoIncrement:!0,keyPath:"id"}).createIndex(B,B,{unique:!1})}},W=class{_queueName;_queueDb;constructor(e){this._queueName=e,this._queueDb=new K}async pushEntry(e){delete e.id,e.queueName=this._queueName,await this._queueDb.addEntry(e)}async unshiftEntry(e){let t=await this._queueDb.getFirstEntryId();t?e.id=t-1:delete e.id,e.queueName=this._queueName,await this._queueDb.addEntry(e)}async popEntry(){return this._removeEntry(await this._queueDb.getLastEntryByQueueName(this._queueName))}async shiftEntry(){return this._removeEntry(await this._queueDb.getFirstEntryByQueueName(this._queueName))}async getAll(){return await this._queueDb.getAllEntriesByQueueName(this._queueName)}async size(){return await this._queueDb.getEntryCountByQueueName(this._queueName)}async deleteEntry(e){await this._queueDb.deleteEntry(e)}async _removeEntry(e){return e&&await this.deleteEntry(e.id),e}};let j=["method","referrer","referrerPolicy","mode","credentials","cache","redirect","integrity","keepalive"];var $=class e{_requestData;static async fromRequest(t){let a={url:t.url,headers:{}};for(let e of("GET"!==t.method&&(a.body=await t.clone().arrayBuffer()),t.headers.forEach((e,t)=>{a.headers[t]=e}),j))void 0!==t[e]&&(a[e]=t[e]);return new e(a)}constructor(e){"navigate"===e.mode&&(e.mode="same-origin"),this._requestData=e}toObject(){let e=Object.assign({},this._requestData);return e.headers=Object.assign({},this._requestData.headers),e.body&&(e.body=e.body.slice(0)),e}toRequest(){return new Request(this._requestData.url,this._requestData)}clone(){return new e(this.toObject())}};let H="serwist-background-sync",V=new Set,G=e=>{let t={request:new $(e.requestData).toRequest(),timestamp:e.timestamp};return e.metadata&&(t.metadata=e.metadata),t};var Q=class{_name;_onSync;_maxRetentionTime;_queueStore;_forceSyncFallback;_syncInProgress=!1;_requestsAddedDuringSync=!1;constructor(e,{forceSyncFallback:t,onSync:a,maxRetentionTime:s}={}){if(V.has(e))throw new l("duplicate-queue-name",{name:e});V.add(e),this._name=e,this._onSync=a||this.replayRequests,this._maxRetentionTime=s||10080,this._forceSyncFallback=!!t,this._queueStore=new W(this._name),this._addSyncListener()}get name(){return this._name}async pushRequest(e){await this._addRequest(e,"push")}async unshiftRequest(e){await this._addRequest(e,"unshift")}async popRequest(){return this._removeRequest("pop")}async shiftRequest(){return this._removeRequest("shift")}async getAll(){let e=await this._queueStore.getAll(),t=Date.now(),a=[];for(let s of e){let e=60*this._maxRetentionTime*1e3;t-s.timestamp>e?await this._queueStore.deleteEntry(s.id):a.push(G(s))}return a}async size(){return await this._queueStore.size()}async _addRequest({request:e,metadata:t,timestamp:a=Date.now()},s){let r={requestData:(await $.fromRequest(e.clone())).toObject(),timestamp:a};switch(t&&(r.metadata=t),s){case"push":await this._queueStore.pushEntry(r);break;case"unshift":await this._queueStore.unshiftEntry(r)}this._syncInProgress?this._requestsAddedDuringSync=!0:await this.registerSync()}async _removeRequest(e){let t,a=Date.now();switch(e){case"pop":t=await this._queueStore.popEntry();break;case"shift":t=await this._queueStore.shiftEntry()}if(t){let s=60*this._maxRetentionTime*1e3;return a-t.timestamp>s?this._removeRequest(e):G(t)}}async replayRequests(){let e;for(;e=await this.shiftRequest();)try{await fetch(e.request.clone())}catch{throw await this.unshiftRequest(e),new l("queue-replay-failed",{name:this._name})}}async registerSync(){if("sync"in self.registration&&!this._forceSyncFallback)try{await self.registration.sync.register(`${H}:${this._name}`)}catch(e){}}_addSyncListener(){"sync"in self.registration&&!this._forceSyncFallback?self.addEventListener("sync",e=>{if(e.tag===`${H}:${this._name}`){let t=async()=>{let t;this._syncInProgress=!0;try{await this._onSync({queue:this})}catch(e){if(e instanceof Error)throw e}finally{this._requestsAddedDuringSync&&!(t&&!e.lastChance)&&await this.registerSync(),this._syncInProgress=!1,this._requestsAddedDuringSync=!1}};e.waitUntil(t())}}):this._onSync({queue:this})}static get _queueNames(){return V}},z=class{_queue;constructor(e,t){this._queue=new Q(e,t)}async fetchDidFail({request:e}){await this._queue.pushRequest({request:e})}};let Y={cacheWillUpdate:async({response:e})=>200===e.status||0===e.status?e:null};function J(e){return"string"==typeof e?new Request(e):e}var X=class{event;request;url;params;_cacheKeys={};_strategy;_handlerDeferred;_extendLifetimePromises;_plugins;_pluginStateMap;constructor(e,t){for(const a of(this.event=t.event,this.request=t.request,t.url&&(this.url=t.url,this.params=t.params),this._strategy=e,this._handlerDeferred=new f,this._extendLifetimePromises=[],this._plugins=[...e.plugins],this._pluginStateMap=new Map,this._plugins))this._pluginStateMap.set(a,{});this.event.waitUntil(this._handlerDeferred.promise)}async fetch(e){let{event:t}=this,a=J(e),s=await this.getPreloadResponse();if(s)return s;let r=this.hasCallback("fetchDidFail")?a.clone():null;try{for(let e of this.iterateCallbacks("requestWillFetch"))a=await e({request:a.clone(),event:t})}catch(e){if(e instanceof Error)throw new l("plugin-error-request-will-fetch",{thrownErrorMessage:e.message})}let n=a.clone();try{let e;for(let s of(e=await fetch(a,"navigate"===a.mode?void 0:this._strategy.fetchOptions),this.iterateCallbacks("fetchDidSucceed")))e=await s({event:t,request:n,response:e});return e}catch(e){throw r&&await this.runCallbacks("fetchDidFail",{error:e,event:t,originalRequest:r.clone(),request:n.clone()}),e}}async fetchAndCachePut(e){let t=await this.fetch(e),a=t.clone();return this.waitUntil(this.cachePut(e,a)),t}async cacheMatch(e){let t,a=J(e),{cacheName:s,matchOptions:r}=this._strategy,n=await this.getCacheKey(a,"read"),i={...r,cacheName:s};for(let e of(t=await caches.match(n,i),this.iterateCallbacks("cachedResponseWillBeUsed")))t=await e({cacheName:s,matchOptions:r,cachedResponse:t,request:n,event:this.event})||void 0;return t}async cachePut(e,t){let a=J(e);await h(0);let s=await this.getCacheKey(a,"write");if(!t)throw new l("cache-put-with-no-response",{url:new URL(String(s.url),location.href).href.replace(RegExp(`^${location.origin}`),"")});let r=await this._ensureResponseSafeToCache(t);if(!r)return!1;let{cacheName:n,matchOptions:i}=this._strategy,c=await self.caches.open(n),o=this.hasCallback("cacheDidUpdate"),u=o?await m(c,s.clone(),["__WB_REVISION__"],i):null;try{await c.put(s,o?r.clone():r)}catch(e){if(e instanceof Error)throw"QuotaExceededError"===e.name&&await g(),e}for(let e of this.iterateCallbacks("cacheDidUpdate"))await e({cacheName:n,oldResponse:u,newResponse:r.clone(),request:s,event:this.event});return!0}async getCacheKey(e,t){let a=`${e.url} | ${t}`;if(!this._cacheKeys[a]){let s=e;for(let e of this.iterateCallbacks("cacheKeyWillBeUsed"))s=J(await e({mode:t,request:s,event:this.event,params:this.params}));this._cacheKeys[a]=s}return this._cacheKeys[a]}hasCallback(e){for(let t of this._strategy.plugins)if(e in t)return!0;return!1}async runCallbacks(e,t){for(let a of this.iterateCallbacks(e))await a(t)}*iterateCallbacks(e){for(let t of this._strategy.plugins)if("function"==typeof t[e]){let a=this._pluginStateMap.get(t),s=s=>{let r={...s,state:a};return t[e](r)};yield s}}waitUntil(e){return this._extendLifetimePromises.push(e),e}async doneWaiting(){let e;for(;e=this._extendLifetimePromises.shift();)await e}destroy(){this._handlerDeferred.resolve(null)}async getPreloadResponse(){if(this.event instanceof FetchEvent&&"navigate"===this.event.request.mode&&"preloadResponse"in this.event)try{let e=await this.event.preloadResponse;if(e)return e}catch(e){return}}async _ensureResponseSafeToCache(e){let t=e,a=!1;for(let e of this.iterateCallbacks("cacheWillUpdate"))if(t=await e({request:this.request,response:t,event:this.event})||void 0,a=!0,!t)break;return!a&&t&&200!==t.status&&(t=void 0),t}},Z=class{cacheName;plugins;fetchOptions;matchOptions;constructor(e={}){this.cacheName=o(e.cacheName),this.plugins=e.plugins||[],this.fetchOptions=e.fetchOptions,this.matchOptions=e.matchOptions}handle(e){let[t]=this.handleAll(e);return t}handleAll(e){e instanceof FetchEvent&&(e={event:e,request:e.request});let t=e.event,a="string"==typeof e.request?new Request(e.request):e.request,s=new X(this,e.url?{event:t,request:a,url:e.url,params:e.params}:{event:t,request:a}),r=this._getResponse(s,a,t);return[r,this._awaitComplete(r,s,a,t)]}async _getResponse(e,t,a){let s;await e.runCallbacks("handlerWillStart",{event:a,request:t});try{if(s=await this._handle(t,e),void 0===s||"error"===s.type)throw new l("no-response",{url:t.url})}catch(r){if(r instanceof Error){for(let n of e.iterateCallbacks("handlerDidError"))if(void 0!==(s=await n({error:r,event:a,request:t})))break}if(!s)throw r}for(let r of e.iterateCallbacks("handlerWillRespond"))s=await r({event:a,request:t,response:s});return s}async _awaitComplete(e,t,a,s){let r,n;try{r=await e}catch{}try{await t.runCallbacks("handlerDidRespond",{event:s,request:a,response:r}),await t.doneWaiting()}catch(e){e instanceof Error&&(n=e)}if(await t.runCallbacks("handlerDidComplete",{event:s,request:a,response:r,error:n}),t.destroy(),n)throw n}},ee=class extends Z{_networkTimeoutSeconds;constructor(e={}){super(e),this.plugins.some(e=>"cacheWillUpdate"in e)||this.plugins.unshift(Y),this._networkTimeoutSeconds=e.networkTimeoutSeconds||0}async _handle(e,t){let a,s=[],r=[];if(this._networkTimeoutSeconds){let{id:n,promise:i}=this._getTimeoutPromise({request:e,logs:s,handler:t});a=n,r.push(i)}let n=this._getNetworkPromise({timeoutId:a,request:e,logs:s,handler:t});r.push(n);let i=await t.waitUntil((async()=>await t.waitUntil(Promise.race(r))||await n)());if(!i)throw new l("no-response",{url:e.url});return i}_getTimeoutPromise({request:e,logs:t,handler:a}){let s;return{promise:new Promise(t=>{s=setTimeout(async()=>{t(await a.cacheMatch(e))},1e3*this._networkTimeoutSeconds)}),id:s}}async _getNetworkPromise({timeoutId:e,request:t,logs:a,handler:s}){let r,n;try{n=await s.fetchAndCachePut(t)}catch(e){e instanceof Error&&(r=e)}return e&&clearTimeout(e),(r||!n)&&(n=await s.cacheMatch(t)),n}},et=class extends Z{_networkTimeoutSeconds;constructor(e={}){super(e),this._networkTimeoutSeconds=e.networkTimeoutSeconds||0}async _handle(e,t){let a,s;try{let a=[t.fetch(e)];if(this._networkTimeoutSeconds){let e=h(1e3*this._networkTimeoutSeconds);a.push(e)}if(!(s=await Promise.race(a)))throw Error(`Timed out the network response after ${this._networkTimeoutSeconds} seconds.`)}catch(e){e instanceof Error&&(a=e)}if(!s)throw new l("no-response",{url:e.url,error:a});return s}};let ea=e=>e&&"object"==typeof e?e:{handle:e};var es=class{handler;match;method;catchHandler;constructor(e,t,a="GET"){this.handler=ea(t),this.match=e,this.method=a}setCatchHandler(e){this.catchHandler=ea(e)}},er=class e extends Z{_fallbackToNetwork;static defaultPrecacheCacheabilityPlugin={cacheWillUpdate:async({response:e})=>!e||e.status>=400?null:e};static copyRedirectedCacheableResponsesPlugin={cacheWillUpdate:async({response:e})=>e.redirected?await M(e):e};constructor(t={}){t.cacheName=c(t.cacheName),super(t),this._fallbackToNetwork=!1!==t.fallbackToNetwork,this.plugins.push(e.copyRedirectedCacheableResponsesPlugin)}async _handle(e,t){let a=await t.getPreloadResponse();if(a)return a;let s=await t.cacheMatch(e);return s||(t.event&&"install"===t.event.type?await this._handleInstall(e,t):await this._handleFetch(e,t))}async _handleFetch(e,t){let a,s=t.params||{};if(this._fallbackToNetwork){let r=s.integrity,n=e.integrity,i=!n||n===r;a=await t.fetch(new Request(e,{integrity:"no-cors"!==e.mode?n||r:void 0})),r&&i&&"no-cors"!==e.mode&&(this._useDefaultCacheabilityPluginIfNeeded(),await t.cachePut(e,a.clone()))}else throw new l("missing-precache-entry",{cacheName:this.cacheName,url:e.url});return a}async _handleInstall(e,t){this._useDefaultCacheabilityPluginIfNeeded();let a=await t.fetch(e);if(!await t.cachePut(e,a.clone()))throw new l("bad-precaching-response",{url:e.url,status:a.status});return a}_useDefaultCacheabilityPluginIfNeeded(){let t=null,a=0;for(let[s,r]of this.plugins.entries())r!==e.copyRedirectedCacheableResponsesPlugin&&(r===e.defaultPrecacheCacheabilityPlugin&&(t=s),r.cacheWillUpdate&&a++);0===a?this.plugins.push(e.defaultPrecacheCacheabilityPlugin):a>1&&null!==t&&this.plugins.splice(t,1)}},en=class extends es{_allowlist;_denylist;constructor(e,{allowlist:t=[/./],denylist:a=[]}={}){super(e=>this._match(e),e),this._allowlist=t,this._denylist=a}_match({url:e,request:t}){if(t&&"navigate"!==t.mode)return!1;let a=e.pathname+e.search;for(let e of this._denylist)if(e.test(a))return!1;return!!this._allowlist.some(e=>e.test(a))}},ei=class extends es{constructor(e,t,a){super(({url:t})=>{let a=e.exec(t.href);if(a)return t.origin!==location.origin&&0!==a.index?void 0:a.slice(1)},t,a)}};let ec=e=>{if(!e)throw new l("add-to-cache-list-unexpected-type",{entry:e});if("string"==typeof e){let t=new URL(e,location.href);return{cacheKey:t.href,url:t.href}}let{revision:t,url:a}=e;if(!a)throw new l("add-to-cache-list-unexpected-type",{entry:e});if(!t){let e=new URL(a,location.href);return{cacheKey:e.href,url:e.href}}let s=new URL(a,location.href),r=new URL(a,location.href);return s.searchParams.set("__WB_REVISION__",t),{cacheKey:s.href,url:r.href}};var eo=class{updatedURLs=[];notUpdatedURLs=[];handlerWillStart=async({request:e,state:t})=>{t&&(t.originalRequest=e)};cachedResponseWillBeUsed=async({event:e,state:t,cachedResponse:a})=>{if("install"===e.type&&t?.originalRequest&&t.originalRequest instanceof Request){let e=t.originalRequest.url;a?this.notUpdatedURLs.push(e):this.updatedURLs.push(e)}return a}};let el=async(e,t,a)=>{let s=t.map((e,t)=>({index:t,item:e})),r=async e=>{let t=[];for(;;){let r=s.pop();if(!r)return e(t);let n=await a(r.item);t.push({result:n,index:r.index})}},n=Array.from({length:e},()=>new Promise(r));return(await Promise.all(n)).flat().sort((e,t)=>e.indexe.result)};"u">typeof navigator&&/^((?!chrome|android).)*safari/i.test(navigator.userAgent);let eh="cache-entries",eu=e=>{let t=new URL(e,location.href);return t.hash="",t.href};var ed=class{_cacheName;_db=null;constructor(e){this._cacheName=e}_getId(e){return`${this._cacheName}|${eu(e)}`}_upgradeDb(e){let t=e.createObjectStore(eh,{keyPath:"id"});t.createIndex("cacheName","cacheName",{unique:!1}),t.createIndex("timestamp","timestamp",{unique:!1})}_upgradeDbAndDeleteOldDbs(e){this._upgradeDb(e),this._cacheName&&function(e,{blocked:t}={}){let a=indexedDB.deleteDatabase(e);t&&a.addEventListener("blocked",e=>t(e.oldVersion,e)),R(a).then(()=>void 0)}(this._cacheName)}async setTimestamp(e,t){e=eu(e);let a={id:this._getId(e),cacheName:this._cacheName,url:e,timestamp:t},s=(await this.getDb()).transaction(eh,"readwrite",{durability:"relaxed"});await s.store.put(a),await s.done}async getTimestamp(e){return(await (await this.getDb()).get(eh,this._getId(e)))?.timestamp}async expireEntries(e,t){let a=await (await this.getDb()).transaction(eh,"readwrite").store.index("timestamp").openCursor(null,"prev"),s=[],r=0;for(;a;){let n=a.value;n.cacheName===this._cacheName&&(e&&n.timestamp=t?(a.delete(),s.push(n.url)):r++),a=await a.continue()}return s}async getDb(){return this._db||(this._db=await S("serwist-expiration",1,{upgrade:this._upgradeDbAndDeleteOldDbs.bind(this)})),this._db}},em=class{_isRunning=!1;_rerunRequested=!1;_maxEntries;_maxAgeSeconds;_matchOptions;_cacheName;_timestampModel;constructor(e,t={}){this._maxEntries=t.maxEntries,this._maxAgeSeconds=t.maxAgeSeconds,this._matchOptions=t.matchOptions,this._cacheName=e,this._timestampModel=new ed(e)}async expireEntries(){if(this._isRunning){this._rerunRequested=!0;return}this._isRunning=!0;let e=this._maxAgeSeconds?Date.now()-1e3*this._maxAgeSeconds:0,t=await this._timestampModel.expireEntries(e,this._maxEntries),a=await self.caches.open(this._cacheName);for(let e of t)await a.delete(e,this._matchOptions);this._isRunning=!1,this._rerunRequested&&(this._rerunRequested=!1,this.expireEntries())}async updateTimestamp(e){await this._timestampModel.setTimestamp(e,Date.now())}async isURLExpired(e){if(!this._maxAgeSeconds)return!1;let t=await this._timestampModel.getTimestamp(e),a=Date.now()-1e3*this._maxAgeSeconds;return void 0===t||t{u.add(e)})(()=>this.deleteCacheAndMetadata())}_getCacheExpiration(e){if(e===o())throw new l("expire-custom-caches-only");let t=this._cacheExpirations.get(e);return t||(t=new em(e,this._config),this._cacheExpirations.set(e,t)),t}cachedResponseWillBeUsed({event:e,cacheName:t,request:a,cachedResponse:s}){if(!s)return null;let r=this._isResponseDateFresh(s),n=this._getCacheExpiration(t),i="last-used"===this._config.maxAgeFrom,c=(async()=>{i&&await n.updateTimestamp(a.url),await n.expireEntries()})();try{e.waitUntil(c)}catch{}return r?s:null}_isResponseDateFresh(e){if("last-used"===this._config.maxAgeFrom)return!0;let t=Date.now();if(!this._config.maxAgeSeconds)return!0;let a=this._getDateHeaderTimestamp(e);return null===a||a>=t-1e3*this._config.maxAgeSeconds}_getDateHeaderTimestamp(e){if(!e.headers.has("date"))return null;let t=new Date(e.headers.get("date")).getTime();return Number.isNaN(t)?null:t}async cacheDidUpdate({cacheName:e,request:t}){let a=this._getCacheExpiration(e);await a.updateTimestamp(t.url),await a.expireEntries()}async deleteCacheAndMetadata(){for(let[e,t]of this._cacheExpirations)await self.caches.delete(e),await t.delete();this._cacheExpirations=new Map}};let eg=/^\/(\w+\/)?collect/,ew=({serwist:e,cacheName:t,...a})=>{let s,r,c=t||i(n.googleAnalytics),o=new z("serwist-google-analytics",{maxRetentionTime:2880,onSync:async({queue:e})=>{let t;for(;t=await e.shiftRequest();){let{request:s,timestamp:r}=t,n=new URL(s.url);try{let e="POST"===s.method?new URLSearchParams(await s.clone().text()):n.searchParams,t=r-(Number(e.get("qt"))||0),i=Date.now()-t;if(e.set("qt",String(i)),a.parameterOverrides)for(let t of Object.keys(a.parameterOverrides)){let s=a.parameterOverrides[t];e.set(t,s)}"function"==typeof a.hitFilter&&a.hitFilter.call(null,e),await fetch(new Request(n.origin+n.pathname,{body:e.toString(),method:"POST",mode:"cors",credentials:"omit",headers:{"Content-Type":"text/plain"}}))}catch(a){throw await e.unshiftRequest(t),a}}}});for(let t of[new es(({url:e})=>"www.googletagmanager.com"===e.hostname&&"/gtm.js"===e.pathname,new ee({cacheName:c}),"GET"),new es(({url:e})=>"www.google-analytics.com"===e.hostname&&"/analytics.js"===e.pathname,new ee({cacheName:c}),"GET"),new es(({url:e})=>"www.googletagmanager.com"===e.hostname&&"/gtag/js"===e.pathname,new ee({cacheName:c}),"GET"),new es(s=({url:e})=>"www.google-analytics.com"===e.hostname&&eg.test(e.pathname),r=new et({plugins:[o]}),"GET"),new es(s,r,"POST")])e.registerRoute(t)};var ep=class{_fallbackUrls;_serwist;constructor({fallbackUrls:e,serwist:t}){this._fallbackUrls=e,this._serwist=t}async handlerDidError(e){for(let t of this._fallbackUrls)if("string"==typeof t){let e=await this._serwist.matchPrecache(t);if(void 0!==e)return e}else if(t.matcher(e)){let e=await this._serwist.matchPrecache(t.url);if(void 0!==e)return e}}};let ey=async(e,t)=>{try{if(206===t.status)return t;let a=e.headers.get("range");if(!a)throw new l("no-range-header");let s=(e=>{let t=e.trim().toLowerCase();if(!t.startsWith("bytes="))throw new l("unit-must-be-bytes",{normalizedRangeHeader:t});if(t.includes(","))throw new l("single-range-only",{normalizedRangeHeader:t});let a=/(\d*)-(\d*)/.exec(t);if(!a||!(a[1]||a[2]))throw new l("invalid-range-values",{normalizedRangeHeader:t});return{start:""===a[1]?void 0:Number(a[1]),end:""===a[2]?void 0:Number(a[2])}})(a),r=await t.blob(),n=((e,t,a)=>{let s,r,n=e.size;if(a&&a>n||t&&t<0)throw new l("range-not-satisfiable",{size:n,end:a,start:t});return void 0!==t&&void 0!==a?(s=t,r=a+1):void 0!==t&&void 0===a?(s=t,r=n):void 0!==a&&void 0===t&&(s=n-a,r=n),{start:s,end:r}})(r,s.start,s.end),i=r.slice(n.start,n.end),c=i.size,o=new Response(i,{status:206,statusText:"Partial Content",headers:t.headers});return o.headers.set("Content-Length",String(c)),o.headers.set("Content-Range",`bytes ${n.start}-${n.end-1}/${r.size}`),o}catch(e){return new Response("",{status:416,statusText:"Range Not Satisfiable"})}};var e_=class{cachedResponseWillBeUsed=async({request:e,cachedResponse:t})=>t&&e.headers.has("range")?await ey(e,t):t},ex=class extends Z{async _handle(e,t){let a,s=await t.cacheMatch(e);if(s);else try{s=await t.fetchAndCachePut(e)}catch(e){e instanceof Error&&(a=e)}if(!s)throw new l("no-response",{url:e.url,error:a});return s}},eb=class extends Z{constructor(e={}){super(e),this.plugins.some(e=>"cacheWillUpdate"in e)||this.plugins.unshift(Y)}async _handle(e,t){let a,s=t.fetchAndCachePut(e).catch(()=>{});t.waitUntil(s);let r=await t.cacheMatch(e);if(r);else try{r=await s}catch(e){e instanceof Error&&(a=e)}if(!r)throw new l("no-response",{url:e.url,error:a});return r}},ev=class extends es{constructor(e,t){super(({request:a})=>{let s=e.getUrlsToPrecacheKeys();for(let r of function*(e,{directoryIndex:t="index.html",ignoreURLParametersMatching:a=[/^utm_/,/^fbclid$/],cleanURLs:s=!0,urlManipulation:r}={}){let n=new URL(e,location.href);n.hash="",yield n.href;let i=((e,t=[])=>{for(let a of[...e.searchParams.keys()])t.some(e=>e.test(a))&&e.searchParams.delete(a);return e})(n,a);if(yield i.href,t&&i.pathname.endsWith("/")){let e=new URL(i.href);e.pathname+=t,yield e.href}if(s){let e=new URL(i.href);e.pathname+=".html",yield e.href}if(r)for(let e of r({url:n}))yield e.href}(a.url,t)){let t=s.get(r);if(t)return{cacheKey:t,integrity:e.getIntegrityForPrecacheKey(t)}}},e.precacheStrategy)}},eE=class{_precacheController;constructor({precacheController:e}){this._precacheController=e}cacheKeyWillBeUsed=async({request:e,params:t})=>{let a=t?.cacheKey||this._precacheController.getPrecacheKeyForUrl(e.url);return a?new Request(a,{headers:e.headers}):e}},eR=class{_urlsToCacheKeys=new Map;_urlsToCacheModes=new Map;_cacheKeysToIntegrities=new Map;_concurrentPrecaching;_precacheStrategy;_routes;_defaultHandlerMap;_catchHandler;_requestRules;constructor({precacheEntries:e,precacheOptions:t,skipWaiting:a=!1,importScripts:s,navigationPreload:r=!1,cacheId:i,clientsClaim:o=!1,runtimeCaching:l,offlineAnalyticsConfig:h,disableDevLogs:u=!1,fallbacks:d,requestRules:m}={}){const{precacheStrategyOptions:f,precacheRouteOptions:g,precacheMiscOptions:w}=((e,t={})=>{let{cacheName:a,plugins:s=[],fetchOptions:r,matchOptions:n,fallbackToNetwork:i,directoryIndex:o,ignoreURLParametersMatching:l,cleanURLs:h,urlManipulation:u,cleanupOutdatedCaches:d,concurrency:m=10,navigateFallback:f,navigateFallbackAllowlist:g,navigateFallbackDenylist:w}=t??{};return{precacheStrategyOptions:{cacheName:c(a),plugins:[...s,new eE({precacheController:e})],fetchOptions:r,matchOptions:n,fallbackToNetwork:i},precacheRouteOptions:{directoryIndex:o,ignoreURLParametersMatching:l,cleanURLs:h,urlManipulation:u},precacheMiscOptions:{cleanupOutdatedCaches:d,concurrency:m,navigateFallback:f,navigateFallbackAllowlist:g,navigateFallbackDenylist:w}}})(this,t);if(this._concurrentPrecaching=w.concurrency,this._precacheStrategy=new er(f),this._routes=new Map,this._defaultHandlerMap=new Map,this._requestRules=m,this.handleInstall=this.handleInstall.bind(this),this.handleActivate=this.handleActivate.bind(this),this.handleFetch=this.handleFetch.bind(this),this.handleCache=this.handleCache.bind(this),s&&s.length>0&&self.importScripts(...s),r&&self.registration?.navigationPreload&&self.addEventListener("activate",e=>{e.waitUntil(self.registration.navigationPreload.enable().then(()=>{}))}),void 0!==i&&(e=>{var t=e;for(let e of Object.keys(n))(e=>{let a=t[e];"string"==typeof a&&(n[e]=a)})(e)})({prefix:i}),a?self.skipWaiting():self.addEventListener("message",e=>{e.data&&"SKIP_WAITING"===e.data.type&&self.skipWaiting()}),o&&self.addEventListener("activate",()=>self.clients.claim()),e&&e.length>0&&this.addToPrecacheList(e),w.cleanupOutdatedCaches&&(e=>{self.addEventListener("activate",t=>{t.waitUntil(p(c(e)).then(e=>{}))})})(f.cacheName),this.registerRoute(new ev(this,g)),w.navigateFallback&&this.registerRoute(new en(this.createHandlerBoundToUrl(w.navigateFallback),{allowlist:w.navigateFallbackAllowlist,denylist:w.navigateFallbackDenylist})),void 0!==h&&("boolean"==typeof h?h&&ew({serwist:this}):ew({...h,serwist:this})),void 0!==l){if(void 0!==d){const e=new ep({fallbackUrls:d.entries,serwist:this});l.forEach(t=>{t.handler instanceof Z&&!t.handler.plugins.some(e=>"handlerDidError"in e)&&t.handler.plugins.push(e)})}for(const e of l)this.registerCapture(e.matcher,e.handler,e.method)}u&&(self.__WB_DISABLE_DEV_LOGS=!0)}get precacheStrategy(){return this._precacheStrategy}get routes(){return this._routes}addEventListeners(){self.addEventListener("install",this.handleInstall),self.addEventListener("activate",this.handleActivate),self.addEventListener("fetch",this.handleFetch),self.addEventListener("message",this.handleCache)}addToPrecacheList(e){let t=[];for(let a of e){"string"==typeof a?t.push(a):a&&!a.integrity&&void 0===a.revision&&t.push(a.url);let{cacheKey:e,url:s}=ec(a),r="string"!=typeof a&&a.revision?"reload":"default";if(this._urlsToCacheKeys.has(s)&&this._urlsToCacheKeys.get(s)!==e)throw new l("add-to-cache-list-conflicting-entries",{firstEntry:this._urlsToCacheKeys.get(s),secondEntry:e});if("string"!=typeof a&&a.integrity){if(this._cacheKeysToIntegrities.has(e)&&this._cacheKeysToIntegrities.get(e)!==a.integrity)throw new l("add-to-cache-list-conflicting-integrities",{url:s});this._cacheKeysToIntegrities.set(e,a.integrity)}this._urlsToCacheKeys.set(s,e),this._urlsToCacheModes.set(s,r)}t.length>0&&console.warn(`Serwist is precaching URLs without revision info: ${t.join(", ")} -This is generally NOT safe. Learn more at https://bit.ly/wb-precache`)}handleInstall(e){return this.registerRequestRules(e),y(e,async()=>{let t=new eo;this.precacheStrategy.plugins.push(t),await el(this._concurrentPrecaching,Array.from(this._urlsToCacheKeys.entries()),async([t,a])=>{let s=this._cacheKeysToIntegrities.get(a),r=this._urlsToCacheModes.get(t),n=new Request(t,{integrity:s,cache:r,credentials:"same-origin"});await Promise.all(this.precacheStrategy.handleAll({event:e,request:n,url:new URL(n.url),params:{cacheKey:a}}))});let{updatedURLs:a,notUpdatedURLs:s}=t;return{updatedURLs:a,notUpdatedURLs:s}})}async registerRequestRules(e){if(this._requestRules&&e?.addRoutes)try{await e.addRoutes(this._requestRules),this._requestRules=void 0}catch(e){throw e}}handleActivate(e){return y(e,async()=>{let e=await self.caches.open(this.precacheStrategy.cacheName),t=await e.keys(),a=new Set(this._urlsToCacheKeys.values()),s=[];for(let r of t)a.has(r.url)||(await e.delete(r),s.push(r.url));return{deletedCacheRequests:s}})}handleFetch(e){let{request:t}=e,a=this.handleRequest({request:t,event:e});a&&e.respondWith(a)}handleCache(e){if(e.data&&"CACHE_URLS"===e.data.type){let{payload:t}=e.data,a=Promise.all(t.urlsToCache.map(t=>{let a;return a="string"==typeof t?new Request(t):new Request(...t),this.handleRequest({request:a,event:e})}));e.waitUntil(a),e.ports?.[0]&&a.then(()=>e.ports[0].postMessage(!0))}}setDefaultHandler(e,t="GET"){this._defaultHandlerMap.set(t,ea(e))}setCatchHandler(e){this._catchHandler=ea(e)}registerCapture(e,t,a){let s=((e,t,a)=>{if("string"==typeof e){let s=new URL(e,location.href);return new es(({url:e})=>e.href===s.href,t,a)}if(e instanceof RegExp)return new ei(e,t,a);if("function"==typeof e)return new es(e,t,a);if(e instanceof es)return e;throw new l("unsupported-route-type",{moduleName:"serwist",funcName:"parseRoute",paramName:"capture"})})(e,t,a);return this.registerRoute(s),s}registerRoute(e){this._routes.has(e.method)||this._routes.set(e.method,[]),this._routes.get(e.method).push(e)}unregisterRoute(e){if(!this._routes.has(e.method))throw new l("unregister-route-but-not-found-with-method",{method:e.method});let t=this._routes.get(e.method).indexOf(e);if(t>-1)this._routes.get(e.method).splice(t,1);else throw new l("unregister-route-route-not-registered")}getUrlsToPrecacheKeys(){return this._urlsToCacheKeys}getPrecachedUrls(){return[...this._urlsToCacheKeys.keys()]}getPrecacheKeyForUrl(e){let t=new URL(e,location.href);return this._urlsToCacheKeys.get(t.href)}getIntegrityForPrecacheKey(e){return this._cacheKeysToIntegrities.get(e)}async matchPrecache(e){let t=e instanceof Request?e.url:e,a=this.getPrecacheKeyForUrl(t);if(a)return(await self.caches.open(this.precacheStrategy.cacheName)).match(a)}createHandlerBoundToUrl(e){let t=this.getPrecacheKeyForUrl(e);if(!t)throw new l("non-precached-url",{url:e});return a=>(a.request=new Request(e),a.params={cacheKey:t,...a.params},this.precacheStrategy.handle(a))}handleRequest({request:e,event:t}){let a,s=new URL(e.url,location.href);if(!s.protocol.startsWith("http"))return;let r=s.origin===location.origin,{params:n,route:i}=this.findMatchingRoute({event:t,request:e,sameOrigin:r,url:s}),c=i?.handler,o=e.method;if(!c&&this._defaultHandlerMap.has(o)&&(c=this._defaultHandlerMap.get(o)),!c)return;try{a=c.handle({url:s,request:e,event:t,params:n})}catch(e){a=Promise.reject(e)}let l=i?.catchHandler;return a instanceof Promise&&(this._catchHandler||l)&&(a=a.catch(async a=>{if(l)try{return await l.handle({url:s,request:e,event:t,params:n})}catch(e){e instanceof Error&&(a=e)}if(this._catchHandler)return this._catchHandler.handle({url:s,request:e,event:t});throw a})),a}findMatchingRoute({url:e,sameOrigin:t,request:a,event:s}){for(let r of this._routes.get(a.method)||[]){let n,i=r.match({url:e,sameOrigin:t,request:a,event:s});if(i)return Array.isArray(n=i)&&0===n.length||i.constructor===Object&&0===Object.keys(i).length?n=void 0:"boolean"==typeof i&&(n=void 0),{route:r,params:n}}return{}}};let eq=[{matcher:/^https:\/\/fonts\.(?:gstatic)\.com\/.*/i,handler:new ex({cacheName:"google-fonts-webfonts",plugins:[new ef({maxEntries:4,maxAgeSeconds:31536e3,maxAgeFrom:"last-used"})]})},{matcher:/^https:\/\/fonts\.(?:googleapis)\.com\/.*/i,handler:new eb({cacheName:"google-fonts-stylesheets",plugins:[new ef({maxEntries:4,maxAgeSeconds:604800,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:eot|otf|ttc|ttf|woff|woff2|font.css)$/i,handler:new eb({cacheName:"static-font-assets",plugins:[new ef({maxEntries:4,maxAgeSeconds:604800,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:jpg|jpeg|gif|png|svg|ico|webp)$/i,handler:new eb({cacheName:"static-image-assets",plugins:[new ef({maxEntries:64,maxAgeSeconds:2592e3,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/static.+\.js$/i,handler:new ex({cacheName:"next-static-js-assets",plugins:[new ef({maxEntries:64,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/image\?url=.+$/i,handler:new eb({cacheName:"next-image",plugins:[new ef({maxEntries:64,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:mp3|wav|ogg)$/i,handler:new ex({cacheName:"static-audio-assets",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"}),new e_]})},{matcher:/\.(?:mp4|webm)$/i,handler:new ex({cacheName:"static-video-assets",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"}),new e_]})},{matcher:/\.(?:js)$/i,handler:new eb({cacheName:"static-js-assets",plugins:[new ef({maxEntries:48,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:css|less)$/i,handler:new eb({cacheName:"static-style-assets",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/data\/.+\/.+\.json$/i,handler:new ee({cacheName:"next-data",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:json|xml|csv)$/i,handler:new ee({cacheName:"static-data-assets",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/api\/auth\/.*/,handler:new et({networkTimeoutSeconds:10})},{matcher:({sameOrigin:e,url:{pathname:t}})=>e&&t.startsWith("/api/"),method:"GET",handler:new ee({cacheName:"apis",plugins:[new ef({maxEntries:16,maxAgeSeconds:86400,maxAgeFrom:"last-used"})],networkTimeoutSeconds:10})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>"1"===e.headers.get("RSC")&&"1"===e.headers.get("Next-Router-Prefetch")&&a&&!t.startsWith("/api/"),handler:new ee({cacheName:"pages-rsc-prefetch",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>"1"===e.headers.get("RSC")&&a&&!t.startsWith("/api/"),handler:new ee({cacheName:"pages-rsc",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>e.headers.get("Content-Type")?.includes("text/html")&&a&&!t.startsWith("/api/"),handler:new ee({cacheName:"pages",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({url:{pathname:e},sameOrigin:t})=>t&&!e.startsWith("/api/"),handler:new ee({cacheName:"others",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({sameOrigin:e})=>!e,handler:new ee({cacheName:"cross-origin",plugins:[new ef({maxEntries:32,maxAgeSeconds:3600})],networkTimeoutSeconds:10})},{matcher:/.*/i,method:"GET",handler:new et}];new eR({precacheEntries:[{'revision':'eb0695a824207ab91c8394dca5ba308f','url':'/_next/static/Ke6tzuWb67iPCOx-CAW2Y/_buildManifest.js'},{'revision':'b6652df95db52feb4daf4eca35380933','url':'/_next/static/Ke6tzuWb67iPCOx-CAW2Y/_ssgManifest.js'},{'revision':null,'url':'/_next/static/chunks/1958-a1cd681eb2c1a83e.js'},{'revision':null,'url':'/_next/static/chunks/4bd1b696-c2f6e0877b6c10aa.js'},{'revision':null,'url':'/_next/static/chunks/5838-7ac6ecc648315323.js'},{'revision':null,'url':'/_next/static/chunks/6810-8b074082a0b51859.js'},{'revision':null,'url':'/_next/static/chunks/7938-3aca95fbb5e36779.js'},{'revision':null,'url':'/_next/static/chunks/8500-3953cc33eeceb44e.js'},{'revision':null,'url':'/_next/static/chunks/9da6db1e-54d62c4d2564a0ac.js'},{'revision':null,'url':'/_next/static/chunks/app/_global-error/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/_not-found/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/checkout/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/props/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/tonight/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/intelligence/feed/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/accuracy/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/add-leg/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/grade/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/players/search/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/live/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/most-parlayed/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/top-graded/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/scan/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/parlays-graded/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/public/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/profile/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/recent-scans/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/scans/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/waitlist/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/webhook/nexapay/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/auth/callback/page-f6253334ccab6a26.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/%5Bslug%5D/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/dashboard/page-107967f64e4a489c.js'},{'revision':null,'url':'/_next/static/chunks/app/forgot-password/page-012a02e7c1c1bf4c.js'},{'revision':null,'url':'/_next/static/chunks/app/game/%5Bid%5D/page-2c1eb07021c4e596.js'},{'revision':null,'url':'/_next/static/chunks/app/intelligence/page-67a1ff5cb5db4b20.js'},{'revision':null,'url':'/_next/static/chunks/app/layout-3a96900bea5fa4a8.js'},{'revision':null,'url':'/_next/static/chunks/app/ledger/page-b2f371ae0e2dc445.js'},{'revision':null,'url':'/_next/static/chunks/app/login/page-e0e44e4f9dfb8d77.js'},{'revision':null,'url':'/_next/static/chunks/app/marketplace/page-ce0df41afd789a48.js'},{'revision':null,'url':'/_next/static/chunks/app/not-found-386ba9205922d4c3.js'},{'revision':null,'url':'/_next/static/chunks/app/page-f47792ee8cedc53b.js'},{'revision':null,'url':'/_next/static/chunks/app/privacy/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/profile/page-589cb3a364727c3b.js'},{'revision':null,'url':'/_next/static/chunks/app/responsible-gambling/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/scan/page-ee1e7545074ffd2a.js'},{'revision':null,'url':'/_next/static/chunks/app/settings/security/page-6728e10517c834af.js'},{'revision':null,'url':'/_next/static/chunks/app/signup/page-06c72711414c663d.js'},{'revision':null,'url':'/_next/static/chunks/app/terms/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/tracker/page-c75c1f66c42eac58.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/desk/page-416ebb86e698df18.js'},{'revision':null,'url':'/_next/static/chunks/app/verify/page-b5ca90220207b0ae.js'},{'revision':null,'url':'/_next/static/chunks/app/welcome/page-ff77b94f609b0d52.js'},{'revision':null,'url':'/_next/static/chunks/framework-29591c88a1db84ab.js'},{'revision':null,'url':'/_next/static/chunks/main-ad5ed0494576592d.js'},{'revision':null,'url':'/_next/static/chunks/main-app-f54c179578b140aa.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/app-error-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/forbidden-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/global-error-d5beecc31181c673.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/unauthorized-115bee36cba427f1.js'},{'revision':'846118c33b2c0e922d7b3a7676f81f6f','url':'/_next/static/chunks/polyfills-42372ed130431b0a.js'},{'revision':null,'url':'/_next/static/chunks/webpack-9660626b6ab2c75c.js'},{'revision':null,'url':'/_next/static/css/64fdd512527e72f3.css'},{'revision':'d7bc0f63f9618b8b700d028d7d242de8','url':'/apple-touch-icon.png'},{'revision':'5616f81ec5f8a9f726a17fb7ac66a54a','url':'/favicon-16.png'},{'revision':'1fc285cfb0116a0523eb34c779a7d282','url':'/favicon-32.png'},{'revision':'583b034f36839a8af92841771eef38b6','url':'/favicon.ico'},{'revision':'b15795128b3691e5703d0ce14ce10827','url':'/favicon.png'},{'revision':'44e9d71551be13574ac2e10063d03aa9','url':'/favicon.svg'},{'revision':'252b8f8c4d0f64c7cd120f182a6c74ab','url':'/icon-192.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icon-512.png'},{'revision':'252b8f8c4d0f64c7cd120f182a6c74ab','url':'/icons/icon-192.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icons/icon-512.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icons/icon-maskable-512.png'},{'revision':'27007d181514e7fe4213035736a8300e','url':'/manifest.json'},{'revision':'9f61cf51298661d5c57a782534216240','url':'/og-image.png'},{'revision':'7c759c20b35840eacf9344692cb51061','url':'/og-image.svg'},{'revision':'9327802333275f11e68c3f19f20c160d','url':'/widget.js'}],skipWaiting:!0,clientsClaim:!0,navigationPreload:!0,runtimeCaching:eq}).addEventListeners(),self.addEventListener("push",e=>{let t;if(!e.data)return;try{t=e.data.json()}catch{t={title:"VYNDR",body:e.data.text()}}let{title:a="VYNDR",body:s="",icon:r="/icons/icon-192.png",url:n="/"}=t;e.waitUntil(self.registration.showNotification(a,{body:s,icon:r,badge:"/icons/icon-192.png",data:{url:n}}))}),self.addEventListener("notificationclick",e=>{e.notification.close();let t=e.notification.data?.url??"/";e.waitUntil(self.clients.matchAll({type:"window",includeUncontrolled:!0}).then(e=>{let a=e.find(e=>e.url.endsWith(t));return a?a.focus():self.clients.openWindow(t)}))})})(); \ No newline at end of file +This is generally NOT safe. Learn more at https://bit.ly/wb-precache`)}handleInstall(e){return this.registerRequestRules(e),y(e,async()=>{let t=new eo;this.precacheStrategy.plugins.push(t),await el(this._concurrentPrecaching,Array.from(this._urlsToCacheKeys.entries()),async([t,a])=>{let s=this._cacheKeysToIntegrities.get(a),r=this._urlsToCacheModes.get(t),n=new Request(t,{integrity:s,cache:r,credentials:"same-origin"});await Promise.all(this.precacheStrategy.handleAll({event:e,request:n,url:new URL(n.url),params:{cacheKey:a}}))});let{updatedURLs:a,notUpdatedURLs:s}=t;return{updatedURLs:a,notUpdatedURLs:s}})}async registerRequestRules(e){if(this._requestRules&&e?.addRoutes)try{await e.addRoutes(this._requestRules),this._requestRules=void 0}catch(e){throw e}}handleActivate(e){return y(e,async()=>{let e=await self.caches.open(this.precacheStrategy.cacheName),t=await e.keys(),a=new Set(this._urlsToCacheKeys.values()),s=[];for(let r of t)a.has(r.url)||(await e.delete(r),s.push(r.url));return{deletedCacheRequests:s}})}handleFetch(e){let{request:t}=e,a=this.handleRequest({request:t,event:e});a&&e.respondWith(a)}handleCache(e){if(e.data&&"CACHE_URLS"===e.data.type){let{payload:t}=e.data,a=Promise.all(t.urlsToCache.map(t=>{let a;return a="string"==typeof t?new Request(t):new Request(...t),this.handleRequest({request:a,event:e})}));e.waitUntil(a),e.ports?.[0]&&a.then(()=>e.ports[0].postMessage(!0))}}setDefaultHandler(e,t="GET"){this._defaultHandlerMap.set(t,ea(e))}setCatchHandler(e){this._catchHandler=ea(e)}registerCapture(e,t,a){let s=((e,t,a)=>{if("string"==typeof e){let s=new URL(e,location.href);return new es(({url:e})=>e.href===s.href,t,a)}if(e instanceof RegExp)return new ei(e,t,a);if("function"==typeof e)return new es(e,t,a);if(e instanceof es)return e;throw new l("unsupported-route-type",{moduleName:"serwist",funcName:"parseRoute",paramName:"capture"})})(e,t,a);return this.registerRoute(s),s}registerRoute(e){this._routes.has(e.method)||this._routes.set(e.method,[]),this._routes.get(e.method).push(e)}unregisterRoute(e){if(!this._routes.has(e.method))throw new l("unregister-route-but-not-found-with-method",{method:e.method});let t=this._routes.get(e.method).indexOf(e);if(t>-1)this._routes.get(e.method).splice(t,1);else throw new l("unregister-route-route-not-registered")}getUrlsToPrecacheKeys(){return this._urlsToCacheKeys}getPrecachedUrls(){return[...this._urlsToCacheKeys.keys()]}getPrecacheKeyForUrl(e){let t=new URL(e,location.href);return this._urlsToCacheKeys.get(t.href)}getIntegrityForPrecacheKey(e){return this._cacheKeysToIntegrities.get(e)}async matchPrecache(e){let t=e instanceof Request?e.url:e,a=this.getPrecacheKeyForUrl(t);if(a)return(await self.caches.open(this.precacheStrategy.cacheName)).match(a)}createHandlerBoundToUrl(e){let t=this.getPrecacheKeyForUrl(e);if(!t)throw new l("non-precached-url",{url:e});return a=>(a.request=new Request(e),a.params={cacheKey:t,...a.params},this.precacheStrategy.handle(a))}handleRequest({request:e,event:t}){let a,s=new URL(e.url,location.href);if(!s.protocol.startsWith("http"))return;let r=s.origin===location.origin,{params:n,route:i}=this.findMatchingRoute({event:t,request:e,sameOrigin:r,url:s}),c=i?.handler,o=e.method;if(!c&&this._defaultHandlerMap.has(o)&&(c=this._defaultHandlerMap.get(o)),!c)return;try{a=c.handle({url:s,request:e,event:t,params:n})}catch(e){a=Promise.reject(e)}let l=i?.catchHandler;return a instanceof Promise&&(this._catchHandler||l)&&(a=a.catch(async a=>{if(l)try{return await l.handle({url:s,request:e,event:t,params:n})}catch(e){e instanceof Error&&(a=e)}if(this._catchHandler)return this._catchHandler.handle({url:s,request:e,event:t});throw a})),a}findMatchingRoute({url:e,sameOrigin:t,request:a,event:s}){for(let r of this._routes.get(a.method)||[]){let n,i=r.match({url:e,sameOrigin:t,request:a,event:s});if(i)return Array.isArray(n=i)&&0===n.length||i.constructor===Object&&0===Object.keys(i).length?n=void 0:"boolean"==typeof i&&(n=void 0),{route:r,params:n}}return{}}};let eq=[{matcher:/^https:\/\/fonts\.(?:gstatic)\.com\/.*/i,handler:new ex({cacheName:"google-fonts-webfonts",plugins:[new ef({maxEntries:4,maxAgeSeconds:31536e3,maxAgeFrom:"last-used"})]})},{matcher:/^https:\/\/fonts\.(?:googleapis)\.com\/.*/i,handler:new eb({cacheName:"google-fonts-stylesheets",plugins:[new ef({maxEntries:4,maxAgeSeconds:604800,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:eot|otf|ttc|ttf|woff|woff2|font.css)$/i,handler:new eb({cacheName:"static-font-assets",plugins:[new ef({maxEntries:4,maxAgeSeconds:604800,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:jpg|jpeg|gif|png|svg|ico|webp)$/i,handler:new eb({cacheName:"static-image-assets",plugins:[new ef({maxEntries:64,maxAgeSeconds:2592e3,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/static.+\.js$/i,handler:new ex({cacheName:"next-static-js-assets",plugins:[new ef({maxEntries:64,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/image\?url=.+$/i,handler:new eb({cacheName:"next-image",plugins:[new ef({maxEntries:64,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:mp3|wav|ogg)$/i,handler:new ex({cacheName:"static-audio-assets",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"}),new e_]})},{matcher:/\.(?:mp4|webm)$/i,handler:new ex({cacheName:"static-video-assets",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"}),new e_]})},{matcher:/\.(?:js)$/i,handler:new eb({cacheName:"static-js-assets",plugins:[new ef({maxEntries:48,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:css|less)$/i,handler:new eb({cacheName:"static-style-assets",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/data\/.+\/.+\.json$/i,handler:new ee({cacheName:"next-data",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:json|xml|csv)$/i,handler:new ee({cacheName:"static-data-assets",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/api\/auth\/.*/,handler:new et({networkTimeoutSeconds:10})},{matcher:({sameOrigin:e,url:{pathname:t}})=>e&&t.startsWith("/api/"),method:"GET",handler:new ee({cacheName:"apis",plugins:[new ef({maxEntries:16,maxAgeSeconds:86400,maxAgeFrom:"last-used"})],networkTimeoutSeconds:10})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>"1"===e.headers.get("RSC")&&"1"===e.headers.get("Next-Router-Prefetch")&&a&&!t.startsWith("/api/"),handler:new ee({cacheName:"pages-rsc-prefetch",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>"1"===e.headers.get("RSC")&&a&&!t.startsWith("/api/"),handler:new ee({cacheName:"pages-rsc",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>e.headers.get("Content-Type")?.includes("text/html")&&a&&!t.startsWith("/api/"),handler:new ee({cacheName:"pages",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({url:{pathname:e},sameOrigin:t})=>t&&!e.startsWith("/api/"),handler:new ee({cacheName:"others",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({sameOrigin:e})=>!e,handler:new ee({cacheName:"cross-origin",plugins:[new ef({maxEntries:32,maxAgeSeconds:3600})],networkTimeoutSeconds:10})},{matcher:/.*/i,method:"GET",handler:new et}];new eR({precacheEntries:[{'revision':'eb0695a824207ab91c8394dca5ba308f','url':'/_next/static/8JC59GV6zNWap7cuCwDZ9/_buildManifest.js'},{'revision':'b6652df95db52feb4daf4eca35380933','url':'/_next/static/8JC59GV6zNWap7cuCwDZ9/_ssgManifest.js'},{'revision':null,'url':'/_next/static/chunks/1958-a1cd681eb2c1a83e.js'},{'revision':null,'url':'/_next/static/chunks/4bd1b696-c2f6e0877b6c10aa.js'},{'revision':null,'url':'/_next/static/chunks/5838-7ac6ecc648315323.js'},{'revision':null,'url':'/_next/static/chunks/6810-8b074082a0b51859.js'},{'revision':null,'url':'/_next/static/chunks/7938-3aca95fbb5e36779.js'},{'revision':null,'url':'/_next/static/chunks/8500-3953cc33eeceb44e.js'},{'revision':null,'url':'/_next/static/chunks/9da6db1e-54d62c4d2564a0ac.js'},{'revision':null,'url':'/_next/static/chunks/app/_global-error/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/_not-found/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/checkout/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/props/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/tonight/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/intelligence/feed/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/accuracy/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/add-leg/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/grade/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/players/search/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/live/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/most-parlayed/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/top-graded/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/scan/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/parlays-graded/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/public/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/profile/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/recent-scans/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/scans/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/waitlist/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/api/webhook/nexapay/route-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/auth/callback/page-f6253334ccab6a26.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/%5Bslug%5D/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/dashboard/page-107967f64e4a489c.js'},{'revision':null,'url':'/_next/static/chunks/app/forgot-password/page-012a02e7c1c1bf4c.js'},{'revision':null,'url':'/_next/static/chunks/app/game/%5Bid%5D/page-2c1eb07021c4e596.js'},{'revision':null,'url':'/_next/static/chunks/app/intelligence/page-67a1ff5cb5db4b20.js'},{'revision':null,'url':'/_next/static/chunks/app/layout-3a96900bea5fa4a8.js'},{'revision':null,'url':'/_next/static/chunks/app/ledger/page-b2f371ae0e2dc445.js'},{'revision':null,'url':'/_next/static/chunks/app/login/page-e0e44e4f9dfb8d77.js'},{'revision':null,'url':'/_next/static/chunks/app/marketplace/page-ce0df41afd789a48.js'},{'revision':null,'url':'/_next/static/chunks/app/not-found-386ba9205922d4c3.js'},{'revision':null,'url':'/_next/static/chunks/app/page-f47792ee8cedc53b.js'},{'revision':null,'url':'/_next/static/chunks/app/privacy/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/profile/page-589cb3a364727c3b.js'},{'revision':null,'url':'/_next/static/chunks/app/responsible-gambling/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/scan/page-ee1e7545074ffd2a.js'},{'revision':null,'url':'/_next/static/chunks/app/settings/security/page-6728e10517c834af.js'},{'revision':null,'url':'/_next/static/chunks/app/signup/page-06c72711414c663d.js'},{'revision':null,'url':'/_next/static/chunks/app/terms/page-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/app/tracker/page-c75c1f66c42eac58.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/desk/page-416ebb86e698df18.js'},{'revision':null,'url':'/_next/static/chunks/app/verify/page-b5ca90220207b0ae.js'},{'revision':null,'url':'/_next/static/chunks/app/welcome/page-ff77b94f609b0d52.js'},{'revision':null,'url':'/_next/static/chunks/framework-29591c88a1db84ab.js'},{'revision':null,'url':'/_next/static/chunks/main-ad5ed0494576592d.js'},{'revision':null,'url':'/_next/static/chunks/main-app-f54c179578b140aa.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/app-error-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/forbidden-115bee36cba427f1.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/global-error-d5beecc31181c673.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/unauthorized-115bee36cba427f1.js'},{'revision':'846118c33b2c0e922d7b3a7676f81f6f','url':'/_next/static/chunks/polyfills-42372ed130431b0a.js'},{'revision':null,'url':'/_next/static/chunks/webpack-9660626b6ab2c75c.js'},{'revision':null,'url':'/_next/static/css/64fdd512527e72f3.css'},{'revision':'d7bc0f63f9618b8b700d028d7d242de8','url':'/apple-touch-icon.png'},{'revision':'5616f81ec5f8a9f726a17fb7ac66a54a','url':'/favicon-16.png'},{'revision':'1fc285cfb0116a0523eb34c779a7d282','url':'/favicon-32.png'},{'revision':'583b034f36839a8af92841771eef38b6','url':'/favicon.ico'},{'revision':'b15795128b3691e5703d0ce14ce10827','url':'/favicon.png'},{'revision':'44e9d71551be13574ac2e10063d03aa9','url':'/favicon.svg'},{'revision':'252b8f8c4d0f64c7cd120f182a6c74ab','url':'/icon-192.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icon-512.png'},{'revision':'252b8f8c4d0f64c7cd120f182a6c74ab','url':'/icons/icon-192.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icons/icon-512.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icons/icon-maskable-512.png'},{'revision':'27007d181514e7fe4213035736a8300e','url':'/manifest.json'},{'revision':'9f61cf51298661d5c57a782534216240','url':'/og-image.png'},{'revision':'7c759c20b35840eacf9344692cb51061','url':'/og-image.svg'},{'revision':'9327802333275f11e68c3f19f20c160d','url':'/widget.js'}],skipWaiting:!0,clientsClaim:!0,navigationPreload:!0,runtimeCaching:eq}).addEventListeners(),self.addEventListener("push",e=>{let t;if(!e.data)return;try{t=e.data.json()}catch{t={title:"VYNDR",body:e.data.text()}}let{title:a="VYNDR",body:s="",icon:r="/icons/icon-192.png",url:n="/"}=t;e.waitUntil(self.registration.showNotification(a,{body:s,icon:r,badge:"/icons/icon-192.png",data:{url:n}}))}),self.addEventListener("notificationclick",e=>{e.notification.close();let t=e.notification.data?.url??"/";e.waitUntil(self.clients.matchAll({type:"window",includeUncontrolled:!0}).then(e=>{let a=e.find(e=>e.url.endsWith(t));return a?a.focus():self.clients.openWindow(t)}))})})(); \ No newline at end of file