diff --git a/BUILD-STATE.md b/BUILD-STATE.md
index 0bf8e88..062ecf6 100755
--- a/BUILD-STATE.md
+++ b/BUILD-STATE.md
@@ -4,7 +4,79 @@
2026-06-12
## Current Phase
-SHIP BUILD v23.0 — All-Day Intelligence Layer: schedule, game lines, streaks, hot lists, stat filtering (Session 23)
+SHIP BUILD v24.0 — Connect Everything: wired the all-day intelligence layer into the live UI + killed stale copy (Session 24)
+
+## Session 24 (2026-06-12) — SHIPPED
+
+Connected Session 23's backend to what users actually see. The Ferrari
+engine got wheels. Frontend-heavy: the Slate now fetches every free/cheap
+layer, the site shows content even with odds-api at 0 credits, and every
+piece of stale copy is gone.
+
+Backend 1567 → **1571 tests** (+4), 125 suites, zero regressions.
+Web build clean.
+
+### PHASE 1 — Slate wired to ALL sources
+- `fetchSlate` now fetches odds + schedule (ESPN) + gamelines (Tank01)
+ per sport in parallel. `mergeSlate()` makes the SCHEDULE the foundation
+ (always shows), overlays odds props (matched by nickname token) and
+ Tank01 lines (matched by team abbreviation). Unmatched odds games are
+ appended so props are never dropped. Schedule empty → odds-only fallback.
+- `GameCard` extended with optional `status`/`score` (LIVE/FINAL badge +
+ score) and `gameLines` (book-by-book ML / spread / total strip).
+- Odds-down-but-schedule-up → soft inline notice, NOT a wall-of-error.
+
+### PHASE 2 — Stat filter pills
+- Pills hidden on the ALL tab (filtering by "points" across mixed sports
+ is meaningless). Sport-specific categories on a single-sport tab.
+- Switching sport resets `activeStat` to 'all' (stale filter would blank
+ the panels).
+
+### PHASE 3 — Copy
+- Hero badge "NBA · MLB · WNBA" → "EVERY SPORT · EVERY PROP"; subhead
+ de-listed the three leagues. Features "Three sports, one engine" →
+ "Every sport, one engine". FAQ updated. LivePropsStrip "TONIGHT'S
+ GRADES LOAD AT 5 PM ET" → "LIVE GRADES APPEAR HERE AS BOOKS POST LINES".
+ Removed the developer-facing "odds endpoint not configured yet" footer.
+ No BetonBLK references existed.
+
+### PHASE 4 — Nav for paid users
+- Paid (analyst/desk) users see "Account" where free/anon see "Pricing".
+- `/account` page created → redirects to `/profile` (canonical plan +
+ subscription-management surface; no duplicate UI).
+
+### PHASE 5 — Cache population
+- `src/startupPrefetch.js` — non-blocking, crash-safe Tank01 cache warm
+ scheduled 5s after boot (`server.js`). Skips when RAPID_API_KEY unset;
+ prefetch failure never crashes the server. Bounded by prefetch's budget.
+
+### PHASE 6 — Language switcher
+- Removed ` ` from the Nav (no translations behind it).
+ i18n infrastructure (LocaleContext, useT, react-i18next, the
+ LocaleSwitcher component file) kept for when translations land.
+
+### PHASE 7 — Empty states
+- Dashboard falls back to the free ESPN schedule when the odds slate is
+ empty, so it shows today's matchups instead of "NO SLATE". "NO SLATE"
+ now appears only when BOTH odds and schedule are genuinely empty.
+- "Tonight's slate is loaded. 0 games across 3 sports." → honest,
+ sport-aware count (or "Your ledger starts here." when zero).
+
+### Files created
+- `src/startupPrefetch.js`
+- `web/src/app/account/page.tsx`
+- `tests/unit/startupPrefetch.test.js`
+
+### Files modified
+- `web/src/components/Slate.tsx` (parallel fetch + merge, notice, pills)
+- `web/src/components/GameCard.tsx` (status/score/game-lines layers)
+- `web/src/components/Nav.tsx` (paid→Account, locale switcher removed)
+- `web/src/components/Hero.tsx`, `Features.tsx`, `FAQ.tsx`,
+ `LivePropsStrip.tsx` (copy)
+- `web/src/app/dashboard/page.tsx` (schedule fallback + copy)
+- `src/server.js` (startup prefetch hook)
+
+---
## Session 23 (2026-06-12) — SHIPPED
diff --git a/data/training/resolutions-2026-06.jsonl b/data/training/resolutions-2026-06.jsonl
index 9d3f842..471e64b 100644
--- a/data/training/resolutions-2026-06.jsonl
+++ b/data/training/resolutions-2026-06.jsonl
@@ -675,3 +675,24 @@
{"ts":"2026-06-12T14:57:50.503Z","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-12T14:57:50.614Z","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-12T14:57:50.736Z","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-12T19:12:28.843Z","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-12T19:12:28.945Z","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-12T19:12:29.072Z","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-12T19:12:29.692Z","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-12T19:12:29.692Z","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-12T19:12:29.692Z","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-12T19:12:29.733Z","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-12T19:38:07.379Z","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-12T19:38:07.479Z","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-12T19:38:07.627Z","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-12T19:38:07.628Z","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-12T19:38:07.628Z","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-12T19:38:07.689Z","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-12T19:38:07.697Z","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-12T19:39:18.273Z","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-12T19:39:18.533Z","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-12T19:39:18.626Z","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-12T19:39:18.768Z","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-12T19:39:18.769Z","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-12T19:39:18.769Z","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-12T19:39:18.874Z","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"}
diff --git a/src/server.js b/src/server.js
index 301a20f..bb6240c 100644
--- a/src/server.js
+++ b/src/server.js
@@ -4,6 +4,9 @@ const app = require('./app');
// otherwise only manifests when the gateway tries to fall over and
// finds no chain.
const { getConfiguredProviders, listProviderIds } = require('./config/providers');
+// Session 24 — warm the Tank01 cache after boot so streaks / hot lists /
+// game lines have data on the first page load. Non-blocking; see module.
+const { scheduleStartupPrefetch } = require('./startupPrefetch');
// Default 3001 — Next.js owns 3000 locally and in production. The poller,
// internal cron, and BASE_URL conventions all assume 3001 for the Express
@@ -18,4 +21,8 @@ app.listen(PORT, () => {
if (missing.length) {
console.warn(`[VYNDR] providers missing keys: ${missing.join(', ')}`);
}
+
+ // Session 24 — fire-and-forget cache warm. 5s delay so Redis is ready.
+ // Skips itself when RAPID_API_KEY is unset; never blocks or crashes boot.
+ scheduleStartupPrefetch();
});
diff --git a/src/startupPrefetch.js b/src/startupPrefetch.js
new file mode 100644
index 0000000..f6ce0f0
--- /dev/null
+++ b/src/startupPrefetch.js
@@ -0,0 +1,64 @@
+/**
+ * Startup prefetch (Session 24).
+ *
+ * Warms the Tank01 game-log + game-lines cache shortly after the server
+ * boots, so the streaks / hot-list panels and the game-lines strip have
+ * data on the FIRST page load instead of self-hiding until a user
+ * happens to trigger a fetch.
+ *
+ * Hard rules:
+ * - NON-BLOCKING. Server readiness must never wait on this. The caller
+ * schedules it and moves on.
+ * - NEVER crashes the process. Every failure is swallowed + logged.
+ * - Skips entirely when RAPID_API_KEY is absent (nothing to warm, and
+ * no point spinning the prefetch's no-op passes).
+ *
+ * The prefetch module owns its own quota budget (`--max`), so a runaway
+ * can't blow the RapidAPI monthly cap.
+ */
+
+const prefetch = require('../scripts/tank01-prefetch');
+
+const DEFAULTS = Object.freeze({
+ sports: ['nba', 'mlb'],
+ maxRequests: 40, // conservative — bounded by the prefetch's own budget
+});
+
+/**
+ * Run the prefetch once. Awaitable for tests; callers in server.js do NOT
+ * await it. Returns the prefetch summary, or null when skipped/failed.
+ */
+async function runStartupPrefetch(opts = {}) {
+ const sports = opts.sports || DEFAULTS.sports;
+ const maxRequests = opts.maxRequests || DEFAULTS.maxRequests;
+
+ if (!process.env.RAPID_API_KEY) {
+ console.log('[VYNDR] startup prefetch skipped — RAPID_API_KEY not set');
+ return null;
+ }
+
+ try {
+ const argv = ['node', 'startup-prefetch', `--sports=${sports.join(',')}`, `--max=${maxRequests}`];
+ const summary = await prefetch.main(argv);
+ console.log('[VYNDR] startup prefetch complete');
+ return summary;
+ } catch (err) {
+ // Swallow — a flaky RapidAPI must never take the API server down.
+ console.warn('[VYNDR] startup prefetch failed:', err.message);
+ return null;
+ }
+}
+
+/**
+ * Schedule the prefetch to run after `delayMs` (default 5s so Redis is
+ * ready). Returns the timer handle (unref'd so it never keeps the event
+ * loop alive on its own). Fully fire-and-forget.
+ */
+function scheduleStartupPrefetch(opts = {}) {
+ const delayMs = opts.delayMs != null ? opts.delayMs : 5000;
+ const timer = setTimeout(() => { void runStartupPrefetch(opts); }, delayMs);
+ if (typeof timer.unref === 'function') timer.unref();
+ return timer;
+}
+
+module.exports = { runStartupPrefetch, scheduleStartupPrefetch, DEFAULTS };
diff --git a/tests/unit/startupPrefetch.test.js b/tests/unit/startupPrefetch.test.js
new file mode 100644
index 0000000..90b46a5
--- /dev/null
+++ b/tests/unit/startupPrefetch.test.js
@@ -0,0 +1,48 @@
+// Unit: startup prefetch (Session 24). Must be non-blocking and crash-safe.
+
+jest.mock('../../scripts/tank01-prefetch', () => ({
+ main: jest.fn(),
+}));
+const prefetch = require('../../scripts/tank01-prefetch');
+const { runStartupPrefetch, scheduleStartupPrefetch } = require('../../src/startupPrefetch');
+
+const savedKey = process.env.RAPID_API_KEY;
+afterAll(() => {
+ if (savedKey === undefined) delete process.env.RAPID_API_KEY;
+ else process.env.RAPID_API_KEY = savedKey;
+});
+
+beforeEach(() => jest.clearAllMocks());
+
+describe('runStartupPrefetch', () => {
+ test('skips (returns null) when RAPID_API_KEY is unset', async () => {
+ delete process.env.RAPID_API_KEY;
+ const result = await runStartupPrefetch();
+ expect(result).toBeNull();
+ expect(prefetch.main).not.toHaveBeenCalled();
+ });
+
+ test('runs the prefetch with sports + max budget when key is set', async () => {
+ process.env.RAPID_API_KEY = 'test-key';
+ prefetch.main.mockResolvedValue({ requestsSpent: 12 });
+ const result = await runStartupPrefetch({ sports: ['nba', 'mlb'], maxRequests: 40 });
+ expect(result).toEqual({ requestsSpent: 12 });
+ const argv = prefetch.main.mock.calls[0][0];
+ expect(argv).toEqual(expect.arrayContaining(['--sports=nba,mlb', '--max=40']));
+ });
+
+ test('prefetch failure resolves to null — never throws (server stays up)', async () => {
+ process.env.RAPID_API_KEY = 'test-key';
+ prefetch.main.mockRejectedValue(new Error('rapidapi 503'));
+ await expect(runStartupPrefetch()).resolves.toBeNull();
+ });
+});
+
+describe('scheduleStartupPrefetch', () => {
+ test('returns an unref-able timer and does not run synchronously', () => {
+ process.env.RAPID_API_KEY = 'test-key';
+ const timer = scheduleStartupPrefetch({ delayMs: 10_000 });
+ expect(prefetch.main).not.toHaveBeenCalled(); // deferred, not immediate
+ clearTimeout(timer);
+ });
+});
diff --git a/web/public/sw.js b/web/public/sw.js
index 2d003ad..555562f 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':'1aa013c5b633c35fbffd48069b861e0c','url':'/_next/static/cfUWKtHLsAhi-PsEa8rB4/_buildManifest.js'},{'revision':'b6652df95db52feb4daf4eca35380933','url':'/_next/static/cfUWKtHLsAhi-PsEa8rB4/_ssgManifest.js'},{'revision':null,'url':'/_next/static/chunks/1896-359ed2e68a8c4f1d.js'},{'revision':null,'url':'/_next/static/chunks/1958-56cbbb8dba15fd2f.js'},{'revision':null,'url':'/_next/static/chunks/2346-d63ba3de1d09a41e.js'},{'revision':null,'url':'/_next/static/chunks/4bd1b696-e356ca5ba0218e27.js'},{'revision':null,'url':'/_next/static/chunks/52774a7f.f2ab7a9b4b8ba576.js'},{'revision':null,'url':'/_next/static/chunks/5838-7e0aa455e9f24259.js'},{'revision':null,'url':'/_next/static/chunks/6810-65ea0987ee90a78b.js'},{'revision':null,'url':'/_next/static/chunks/7602.2868ff821d53ee09.js'},{'revision':null,'url':'/_next/static/chunks/7918-2500511be32ff738.js'},{'revision':null,'url':'/_next/static/chunks/8500-f62a38ff68ab7f42.js'},{'revision':null,'url':'/_next/static/chunks/9da6db1e-9623f3245f088d02.js'},{'revision':null,'url':'/_next/static/chunks/app/_global-error/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/_not-found/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/admin/page-7653c5a94021e122.js'},{'revision':null,'url':'/_next/static/chunks/app/api/admin/stats/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/checkout/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/props/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/tonight/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/hero-prop/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/intelligence/feed/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/accuracy/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/mlb/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/nba/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/soccer/%5Bleague%5D/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/wnba/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/add-leg/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/grade/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/players/search/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/live/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/most-parlayed/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/top-graded/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/scan/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/parlays-graded/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/public/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/profile/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/recent-scans/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/scans/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/waitlist/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/webhook/nexapay/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/welcome-email/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/auth/callback/page-0cb8d3790d034de0.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/%5Bslug%5D/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/dashboard/page-1b62da9adc97d4c5.js'},{'revision':null,'url':'/_next/static/chunks/app/forgot-password/page-a251f52b4206e8d1.js'},{'revision':null,'url':'/_next/static/chunks/app/game/%5Bid%5D/page-34d758f67cb52da7.js'},{'revision':null,'url':'/_next/static/chunks/app/intelligence/page-5f6c106fd6c53851.js'},{'revision':null,'url':'/_next/static/chunks/app/layout-2d5d636ce98fef68.js'},{'revision':null,'url':'/_next/static/chunks/app/ledger/page-3bfc2406ebeb6d2e.js'},{'revision':null,'url':'/_next/static/chunks/app/login/page-69c1ebd6ff5c12d8.js'},{'revision':null,'url':'/_next/static/chunks/app/marketplace/page-cbada49ae96a7527.js'},{'revision':null,'url':'/_next/static/chunks/app/not-found-1fd99a98dab3c771.js'},{'revision':null,'url':'/_next/static/chunks/app/page-2894b8877e22e9ec.js'},{'revision':null,'url':'/_next/static/chunks/app/pricing/page-3f26af475b3f8452.js'},{'revision':null,'url':'/_next/static/chunks/app/privacy/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/profile/page-44681f894156db65.js'},{'revision':null,'url':'/_next/static/chunks/app/responsible-gambling/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/scan/page-fc810e8b5e4f992b.js'},{'revision':null,'url':'/_next/static/chunks/app/settings/security/page-59df87aff0c62ec8.js'},{'revision':null,'url':'/_next/static/chunks/app/signup/page-33e502bb04569f60.js'},{'revision':null,'url':'/_next/static/chunks/app/soccer/page-9ba3b46b0b1dfa59.js'},{'revision':null,'url':'/_next/static/chunks/app/terms/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/tracker/page-d6937e1faa494c49.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/cancel/page-1fd99a98dab3c771.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/desk/page-0ed948b44f9c64d7.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/success/page-b6baded9f44ae951.js'},{'revision':null,'url':'/_next/static/chunks/app/verify/page-cdeec62ab3a5d8ff.js'},{'revision':null,'url':'/_next/static/chunks/app/welcome/page-e3ab6d5822dee6dc.js'},{'revision':null,'url':'/_next/static/chunks/framework-95da69ac6843d788.js'},{'revision':null,'url':'/_next/static/chunks/main-a354262253293123.js'},{'revision':null,'url':'/_next/static/chunks/main-app-4231d5f500f33972.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/app-error-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/forbidden-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/global-error-756d787dba63aa4d.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/unauthorized-cdfd9ff3adbcee2e.js'},{'revision':'846118c33b2c0e922d7b3a7676f81f6f','url':'/_next/static/chunks/polyfills-42372ed130431b0a.js'},{'revision':null,'url':'/_next/static/chunks/webpack-a41d593db6ea66d4.js'},{'revision':null,'url':'/_next/static/css/f795112d016cc138.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':'bf670b1fafa1b3127bd50abe86d723b9','url':'/images/player-silhouette.svg'},{'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':null,'url':'/_next/static/chunks/1896-359ed2e68a8c4f1d.js'},{'revision':null,'url':'/_next/static/chunks/1958-56cbbb8dba15fd2f.js'},{'revision':null,'url':'/_next/static/chunks/2346-d63ba3de1d09a41e.js'},{'revision':null,'url':'/_next/static/chunks/4bd1b696-e356ca5ba0218e27.js'},{'revision':null,'url':'/_next/static/chunks/52774a7f.f2ab7a9b4b8ba576.js'},{'revision':null,'url':'/_next/static/chunks/5838-7e0aa455e9f24259.js'},{'revision':null,'url':'/_next/static/chunks/6810-65ea0987ee90a78b.js'},{'revision':null,'url':'/_next/static/chunks/7602.2868ff821d53ee09.js'},{'revision':null,'url':'/_next/static/chunks/7918-9cfe56e3ee27ed27.js'},{'revision':null,'url':'/_next/static/chunks/8500-f62a38ff68ab7f42.js'},{'revision':null,'url':'/_next/static/chunks/9da6db1e-9623f3245f088d02.js'},{'revision':null,'url':'/_next/static/chunks/app/_global-error/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/_not-found/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/account/page-f5e2a6d5e729cc87.js'},{'revision':null,'url':'/_next/static/chunks/app/admin/page-7653c5a94021e122.js'},{'revision':null,'url':'/_next/static/chunks/app/api/admin/stats/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/checkout/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/props/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/tonight/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/hero-prop/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/intelligence/feed/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/accuracy/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/mlb/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/nba/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/soccer/%5Bleague%5D/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/wnba/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/add-leg/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/grade/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/players/search/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/live/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/most-parlayed/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/top-graded/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/scan/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/parlays-graded/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/public/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/profile/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/recent-scans/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/scans/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/waitlist/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/webhook/nexapay/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/api/welcome-email/route-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/auth/callback/page-0cb8d3790d034de0.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/%5Bslug%5D/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/dashboard/page-be8c44c2d3ecd74d.js'},{'revision':null,'url':'/_next/static/chunks/app/forgot-password/page-a251f52b4206e8d1.js'},{'revision':null,'url':'/_next/static/chunks/app/game/%5Bid%5D/page-34d758f67cb52da7.js'},{'revision':null,'url':'/_next/static/chunks/app/intelligence/page-5f6c106fd6c53851.js'},{'revision':null,'url':'/_next/static/chunks/app/layout-c72e0130bc50baac.js'},{'revision':null,'url':'/_next/static/chunks/app/ledger/page-3bfc2406ebeb6d2e.js'},{'revision':null,'url':'/_next/static/chunks/app/login/page-69c1ebd6ff5c12d8.js'},{'revision':null,'url':'/_next/static/chunks/app/marketplace/page-cbada49ae96a7527.js'},{'revision':null,'url':'/_next/static/chunks/app/not-found-1fd99a98dab3c771.js'},{'revision':null,'url':'/_next/static/chunks/app/page-e854e880465d9c9e.js'},{'revision':null,'url':'/_next/static/chunks/app/pricing/page-3f26af475b3f8452.js'},{'revision':null,'url':'/_next/static/chunks/app/privacy/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/profile/page-44681f894156db65.js'},{'revision':null,'url':'/_next/static/chunks/app/responsible-gambling/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/scan/page-fc810e8b5e4f992b.js'},{'revision':null,'url':'/_next/static/chunks/app/settings/security/page-59df87aff0c62ec8.js'},{'revision':null,'url':'/_next/static/chunks/app/signup/page-33e502bb04569f60.js'},{'revision':null,'url':'/_next/static/chunks/app/soccer/page-9ba3b46b0b1dfa59.js'},{'revision':null,'url':'/_next/static/chunks/app/terms/page-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/app/tracker/page-d6937e1faa494c49.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/cancel/page-1fd99a98dab3c771.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/desk/page-0ed948b44f9c64d7.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/success/page-b6baded9f44ae951.js'},{'revision':null,'url':'/_next/static/chunks/app/verify/page-cdeec62ab3a5d8ff.js'},{'revision':null,'url':'/_next/static/chunks/app/welcome/page-e3ab6d5822dee6dc.js'},{'revision':null,'url':'/_next/static/chunks/framework-95da69ac6843d788.js'},{'revision':null,'url':'/_next/static/chunks/main-a354262253293123.js'},{'revision':null,'url':'/_next/static/chunks/main-app-4231d5f500f33972.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/app-error-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/forbidden-cdfd9ff3adbcee2e.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/global-error-756d787dba63aa4d.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/unauthorized-cdfd9ff3adbcee2e.js'},{'revision':'846118c33b2c0e922d7b3a7676f81f6f','url':'/_next/static/chunks/polyfills-42372ed130431b0a.js'},{'revision':null,'url':'/_next/static/chunks/webpack-a41d593db6ea66d4.js'},{'revision':null,'url':'/_next/static/css/f795112d016cc138.css'},{'revision':'2471bb8484e5d20d74f7e2e07e0b8f17','url':'/_next/static/qM5oQbx60U91cObpI9jnn/_buildManifest.js'},{'revision':'b6652df95db52feb4daf4eca35380933','url':'/_next/static/qM5oQbx60U91cObpI9jnn/_ssgManifest.js'},{'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':'bf670b1fafa1b3127bd50abe86d723b9','url':'/images/player-silhouette.svg'},{'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
diff --git a/web/src/app/account/page.tsx b/web/src/app/account/page.tsx
new file mode 100644
index 0000000..cabc6f6
--- /dev/null
+++ b/web/src/app/account/page.tsx
@@ -0,0 +1,31 @@
+'use client';
+
+import { useEffect } from 'react';
+import { useRouter } from 'next/navigation';
+
+/**
+ * /account (Session 24).
+ *
+ * Paid users see "Account" in the nav instead of "Pricing". Rather than
+ * duplicate the subscription UI, this route forwards to /profile, which
+ * already renders the current plan, usage, founder pricing, and the
+ * cancel/manage-subscription controls. Keeping one canonical surface
+ * avoids two screens drifting out of sync.
+ */
+export default function AccountPage() {
+ const router = useRouter();
+ useEffect(() => {
+ router.replace('/profile');
+ }, [router]);
+
+ return (
+
+
+ Opening your account…
+
+
+ );
+}
diff --git a/web/src/app/dashboard/page.tsx b/web/src/app/dashboard/page.tsx
index 41f6ac3..0d84100 100644
--- a/web/src/app/dashboard/page.tsx
+++ b/web/src/app/dashboard/page.tsx
@@ -24,6 +24,16 @@ interface Game {
injury_note?: string;
}
+// Session 24 — shape returned by the free ESPN schedule endpoint
+// (/api/schedule/:sport), used as a fallback when the odds slate is empty.
+interface ScheduleApiGame {
+ id: string;
+ homeTeam?: { name?: string | null; abbreviation?: string | null };
+ awayTeam?: { name?: string | null; abbreviation?: string | null };
+ gameTime?: string | null;
+ status?: 'pre' | 'in' | 'post' | null;
+}
+
interface TopGrade {
player: string;
stat: string;
@@ -88,9 +98,30 @@ export default function DashboardPage() {
Promise.all([
fetch(`/api/games/tonight?sport=${sport}`).then((r) => r.json()).catch(() => ({ games: [] })),
fetch(`/api/props/top-graded?sport=${sport}`).then((r) => r.json()).catch(() => ({ props: [] })),
- ]).then(([gamesData, gradesData]) => {
+ ]).then(async ([gamesData, gradesData]) => {
if (cancelled) return;
- setGames(Array.isArray(gamesData?.games) ? gamesData.games : []);
+ let list: Game[] = Array.isArray(gamesData?.games) ? gamesData.games : [];
+
+ // Session 24 — when the odds-backed slate is empty (off-day or
+ // odds-api quota exhausted), fall back to the FREE ESPN schedule so
+ // the dashboard still shows today's matchups instead of "NO SLATE".
+ if (list.length === 0) {
+ try {
+ const sched = await fetch(`/api/schedule/${sport.toLowerCase()}`).then((r) => r.json());
+ const schedGames = Array.isArray(sched?.games) ? sched.games : [];
+ list = schedGames.map((sg: ScheduleApiGame) => ({
+ id: sg.id,
+ away: sg.awayTeam?.name || sg.awayTeam?.abbreviation || 'Away',
+ home: sg.homeTeam?.name || sg.homeTeam?.abbreviation || 'Home',
+ start_time: sg.gameTime || '',
+ sport,
+ status: sg.status === 'in' ? 'live' : sg.status === 'post' ? 'final' : 'scheduled',
+ }));
+ } catch { /* schedule unavailable too — leave list empty */ }
+ }
+
+ if (cancelled) return;
+ setGames(list);
setTopGrades(Array.isArray(gradesData?.props) ? gradesData.props.slice(0, 10) : []);
});
@@ -255,7 +286,7 @@ export default function DashboardPage() {
{/* Tonight's games */}
-
+
{games === null ? (
) : games.length === 0 ? (
@@ -378,7 +409,9 @@ export default function DashboardPage() {
WELCOME TO THE LEDGER
- Tonight's slate is loaded. {games?.length ?? 0} {games?.length === 1 ? 'game' : 'games'} across 3 sports.
+ {(games?.length ?? 0) > 0
+ ? `Today's slate is loaded. ${games?.length} ${games?.length === 1 ? 'game' : 'games'} on the ${sport} board.`
+ : 'Your ledger starts here.'}
Pick a game and read your first prop — it's on us.
diff --git a/web/src/components/FAQ.tsx b/web/src/components/FAQ.tsx
index 3cb6477..6dd3dce 100644
--- a/web/src/components/FAQ.tsx
+++ b/web/src/components/FAQ.tsx
@@ -13,7 +13,7 @@ const FAQS = [
},
{
q: 'What sports do you cover?',
- a: 'NBA, MLB, and WNBA at launch. NFL is targeted for September 2026. Each sport has its own calibrated weights and sport-specific factor models.',
+ a: 'NBA, MLB, WNBA, and soccer today. NFL is targeted for September 2026, with more sports rolling out through 2026. Each sport has its own calibrated weights and sport-specific factor models.',
},
{
q: 'Can I cancel anytime?',
diff --git a/web/src/components/Features.tsx b/web/src/components/Features.tsx
index 5492337..b46fc22 100644
--- a/web/src/components/Features.tsx
+++ b/web/src/components/Features.tsx
@@ -31,8 +31,8 @@ const FEATURES = [
},
{
icon: '◯',
- title: 'Three sports, one engine',
- body: 'NBA. MLB. WNBA. Unified intelligence layer with sport-specific calibration. NFL coming September 2026.',
+ title: 'Every sport, one engine',
+ body: 'NBA. MLB. WNBA. Soccer. NFL coming September 2026 — more rolling out through 2026. A unified intelligence layer with sport-specific calibration.',
},
{
icon: '⌦',
diff --git a/web/src/components/GameCard.tsx b/web/src/components/GameCard.tsx
index a73ad26..7c40d65 100644
--- a/web/src/components/GameCard.tsx
+++ b/web/src/components/GameCard.tsx
@@ -34,6 +34,24 @@ const SPORT_ACCENT: Record = {
soccer: '#00D4A0',
};
+// Session 24 — game-level book-by-book lines from Tank01. One row per
+// sportsbook (bet365 / betmgm / caesars …). All fields optional/null —
+// books publish lines independently, so we render only what exists.
+export interface GameLineBook {
+ homeML?: string | null;
+ awayML?: string | null;
+ total?: string | null;
+ homeSpread?: string | null;
+ awaySpread?: string | null;
+}
+export interface GameLines {
+ homeTeam?: string | null;
+ awayTeam?: string | null;
+ books: Record;
+}
+
+export type GameStatus = 'pre' | 'in' | 'post';
+
export interface GameCardProps {
sport: SlateSport;
homeTeam: string;
@@ -49,6 +67,19 @@ export interface GameCardProps {
onGrade: (prop: PropRowProp) => void;
onUpgrade?: () => void;
defaultVisible?: number; // how many props to show before "+ N more"
+ // Session 24 — all-day intelligence layers, all optional. A game card
+ // shows whatever exists: schedule status/score (ESPN), game lines
+ // (Tank01), and props (odds-api) — nothing replaces anything else.
+ status?: GameStatus;
+ score?: { home: number; away: number } | null;
+ gameLines?: GameLines | null;
+}
+
+// Map ESPN status → a compact, human badge. 'in' is live; 'post' is final.
+function statusBadge(status?: GameStatus, score?: { home: number; away: number } | null) {
+ if (status === 'in') return { label: 'LIVE', color: '#FF4D4D', score };
+ if (status === 'post') return { label: 'FINAL', color: '#6B6B7B', score };
+ return null; // 'pre' or unknown → no badge; the tip-off time carries it
}
function formatTime(iso?: string) {
@@ -142,8 +173,11 @@ export default function GameCard(props: GameCardProps) {
props: propList, gradedProps, loadingKey, errorByKey,
tier = 'free', onGrade, onUpgrade,
defaultVisible = 4,
+ status, score, gameLines,
} = props;
const [expanded, setExpanded] = useState(false);
+ const badge = statusBadge(status, score);
+ const bookRows = gameLines?.books ? Object.entries(gameLines.books) : [];
// Session 19 — visibility budget now applies to PLAYERS, not raw
// props. Showing the first 4 prop rows that all belonged to the
@@ -244,21 +278,76 @@ export default function GameCard(props: GameCardProps) {
{[formatTime(gameTime), venue, context].filter(Boolean).join(' · ') || ' '}
-
- {propList.length} prop{propList.length === 1 ? '' : 's'}
+
+ {/* Session 24 — live/final status + score (ESPN schedule). */}
+ {badge && (
+
+ {badge.label}{badge.score ? ` · ${badge.score.away}–${badge.score.home}` : ''}
+
+ )}
+
+ {propList.length} prop{propList.length === 1 ? '' : 's'}
+
+ {/* Session 24 — game lines strip (Tank01). Book-by-book moneyline,
+ spread, total. Renders only when lines exist; never blocks the
+ card. The brand edge is props, but lines give immediate action
+ even when props haven't been published. */}
+ {bookRows.length > 0 && (
+
+
+ Game Lines
+
+ {bookRows.map(([book, line]) => (
+
+ {book}
+ {line.awayML && {teamAbbr(awayTeam, sport)} {line.awayML} }
+ {line.homeML && {teamAbbr(homeTeam, sport)} {line.homeML} }
+ {line.total && O/U {line.total} }
+
+ ))}
+
+ )}
+
{propList.length === 0 ? (
- NBA · MLB · WNBA
+ EVERY SPORT · EVERY PROP
- Grade your NBA, MLB, and WNBA props with intelligence the books don't want you to have.
+ Grade your props across every sport with intelligence the books don't want you to have.
Forty-plus factors. Kill conditions. Alt-line ladders. The honest ledger.
diff --git a/web/src/components/LivePropsStrip.tsx b/web/src/components/LivePropsStrip.tsx
index 9fd5e79..a8b6b06 100644
--- a/web/src/components/LivePropsStrip.tsx
+++ b/web/src/components/LivePropsStrip.tsx
@@ -80,7 +80,7 @@ export default function LivePropsStrip() {
letterSpacing: '0.08em',
}}
>
- TONIGHT'S GRADES LOAD AT 5 PM ET
+ LIVE GRADES APPEAR HERE AS BOOKS POST LINES
);
diff --git a/web/src/components/Nav.tsx b/web/src/components/Nav.tsx
index 3abd06c..1d3aa64 100644
--- a/web/src/components/Nav.tsx
+++ b/web/src/components/Nav.tsx
@@ -5,7 +5,10 @@ import { usePathname } from 'next/navigation';
import { useAuth } from '@/contexts/AuthContext';
import Wordmark from '@/components/Wordmark';
import NotificationBell from '@/components/NotificationBell';
-import LocaleSwitcher from '@/components/LocaleSwitcher';
+// Session 24 — LocaleSwitcher removed from the nav. The i18n
+// infrastructure (react-i18next, LocaleContext, useT) stays in place,
+// but a visible language toggle with no translations behind it is
+// worse than none. Re-add the switcher when translations land.
import { useT } from '@/contexts/LocaleContext';
export default function Nav() {
@@ -27,10 +30,16 @@ export default function Nav() {
// /dashboard IS the scan surface (click [Read] on any prop). The
// /scan page still exists as a fallback for custom props and is
// reachable from the slate's "Scan manually" empty-state CTA.
+ // Session 24 — paid users (analyst / desk) get "Account" where free
+ // users and signed-out visitors see "Pricing". A subscriber shouldn't
+ // be pitched a plan they already pay for.
+ const isPaid = !!user && tier !== 'free';
const NAV_LINKS = [
{ label: t('nav.tracker'), href: '/tracker' },
{ label: t('nav.ledger'), href: '/ledger' },
- { label: t('nav.pricing'), href: '/pricing' },
+ isPaid
+ ? { label: 'Account', href: '/account' }
+ : { label: t('nav.pricing'), href: '/pricing' },
{ label: 'Blog', href: '/blog' },
];
@@ -124,7 +133,6 @@ export default function Nav() {
)}
-
setMenuOpen((o) => !o)}
aria-haspopup="menu"
@@ -198,7 +206,6 @@ export default function Nav() {
) : (
-
{t('nav.login')}
diff --git a/web/src/components/Slate.tsx b/web/src/components/Slate.tsx
index 387f52a..6ed416f 100644
--- a/web/src/components/Slate.tsx
+++ b/web/src/components/Slate.tsx
@@ -2,7 +2,7 @@
import { useCallback, useEffect, useMemo, useState } from 'react';
import { useRouter } from 'next/navigation';
-import GameCard, { SlateSport } from '@/components/GameCard';
+import GameCard, { SlateSport, GameLines } from '@/components/GameCard';
import { PropRowProp, PropRowResult, propRowKey, Tier } from '@/components/PropRow';
import { useAuth } from '@/contexts/AuthContext';
// Session 23 — all-day intelligence layer. The stat filter is the
@@ -134,6 +134,98 @@ interface SlateGame {
venue?: string;
context?: string;
props: PropRowProp[];
+ // Session 24 — schedule + game-lines layers overlaid onto each game.
+ status?: 'pre' | 'in' | 'post';
+ score?: { home: number; away: number } | null;
+ gameLines?: GameLines | null;
+}
+
+// ---- Session 24: schedule + game-lines response shapes ----
+interface ScheduleTeam { name?: string | null; abbreviation?: string | null }
+interface ScheduleGame {
+ id?: string;
+ homeTeam?: ScheduleTeam;
+ awayTeam?: ScheduleTeam;
+ gameTime?: string | null;
+ status?: 'pre' | 'in' | 'post' | null;
+ score?: { home: number; away: number } | null;
+ venue?: string | null;
+ broadcast?: string | null;
+}
+interface ScheduleResponse { games?: ScheduleGame[] }
+interface GameLinesResponse { games?: Record
}
+
+// Nickname token (last word) — the most stable cross-source identifier
+// between ESPN full names and odds-api full names ("San Antonio Spurs"
+// ↔ "spurs"). Falls back to the whole normalized string.
+function nickToken(name?: string | null): string {
+ const w = String(name || '').trim().split(/\s+/);
+ const last = w[w.length - 1] || '';
+ return last.toLowerCase().replace(/[^a-z]/g, '');
+}
+
+// Match an odds-derived game to a schedule game by both nicknames.
+function gamesMatch(scheduleHome: string, scheduleAway: string, oddsHome: string, oddsAway: string): boolean {
+ const sh = nickToken(scheduleHome), sa = nickToken(scheduleAway);
+ const oh = nickToken(oddsHome), oa = nickToken(oddsAway);
+ if (!sh || !sa || !oh || !oa) return false;
+ return (sh === oh && sa === oa) || (sh === oa && sa === oh);
+}
+
+// Find the Tank01 game-lines entry for a schedule game by team
+// abbreviation (ESPN + Tank01 both use standard team abbreviations).
+function findGameLines(home?: ScheduleTeam, away?: ScheduleTeam, lines?: Record): GameLines | null {
+ if (!lines) return null;
+ const h = (home?.abbreviation || '').toUpperCase();
+ const a = (away?.abbreviation || '').toUpperCase();
+ if (!h && !a) return null;
+ for (const entry of Object.values(lines)) {
+ const eh = String(entry.homeTeam || '').toUpperCase();
+ const ea = String(entry.awayTeam || '').toUpperCase();
+ if ((eh === h && ea === a) || (eh === a && ea === h)) return entry;
+ }
+ return null;
+}
+
+/**
+ * Session 24 — merge the three free/cheap layers into one game list.
+ * Schedule is the FOUNDATION (always shows from ESPN); odds props and
+ * Tank01 lines overlay onto matching games. Unmatched odds games are
+ * appended so we never drop props. When schedule is empty, the odds
+ * games become the base (odds-only fallback).
+ */
+function mergeSlate(
+ sport: SlateSport,
+ scheduleGames: ScheduleGame[],
+ oddsGames: SlateGame[],
+ lines?: Record,
+): SlateGame[] {
+ const base: SlateGame[] = scheduleGames.map((sg) => ({
+ sport,
+ homeTeam: sg.homeTeam?.name || '',
+ awayTeam: sg.awayTeam?.name || '',
+ gameTime: sg.gameTime || undefined,
+ venue: sg.venue || undefined,
+ status: sg.status || undefined,
+ score: sg.score || undefined,
+ props: [],
+ gameLines: findGameLines(sg.homeTeam, sg.awayTeam, lines),
+ }));
+
+ const unmatched: SlateGame[] = [];
+ for (const og of oddsGames) {
+ const target = base.find((b) => gamesMatch(b.homeTeam, b.awayTeam, og.homeTeam, og.awayTeam));
+ if (target) target.props.push(...og.props);
+ else unmatched.push(og);
+ }
+
+ const merged = [...base, ...unmatched];
+ // Stable order: scheduled tip-off time, unknowns last.
+ return merged.sort((a, b) => {
+ const ta = a.gameTime ? Date.parse(a.gameTime) : Number.MAX_SAFE_INTEGER;
+ const tb = b.gameTime ? Date.parse(b.gameTime) : Number.MAX_SAFE_INTEGER;
+ return ta - tb;
+ });
}
function groupByGame(rawProps: RawProp[], sport: SlateSport): SlateGame[] {
@@ -194,7 +286,9 @@ export default function Slate({ initialTab = 'all', tier = 'free' }: SlateProps)
const [games, setGames] = useState([]);
const [loading, setLoading] = useState(false);
const [fetchError, setFetchError] = useState(null);
- const [unsupportedSports, setUnsupportedSports] = useState([]);
+ // Session 24 — when odds are unavailable but the schedule still has
+ // games, this becomes a soft inline notice instead of a wall-of-error.
+ const [oddsNotice, setOddsNotice] = useState(false);
// Grade state — Map keyed by propRowKey.
const [gradedProps, setGradedProps] = useState>(() => new Map());
@@ -204,19 +298,24 @@ export default function Slate({ initialTab = 'all', tier = 'free' }: SlateProps)
// Search filter (Phase 3.4 — kept here so the Slate owns its own filtering).
const [searchQuery, setSearchQuery] = useState('');
- // Fetch + group. Promise.allSettled so one sport failing doesn't blank the slate.
+ // Session 24 — fetch ALL free/cheap layers per sport in parallel:
+ // odds (odds-api props) · schedule (ESPN) · gamelines (Tank01)
+ // Schedule is the foundation — games render even when odds are
+ // empty/503. Odds + lines overlay on top. The slate is never empty
+ // just because one provider is down.
const fetchSlate = useCallback(async (active: SlateTab) => {
setLoading(true);
setFetchError(null);
+ setOddsNotice(false);
- const sportsToFetch: Array<{ sport: SlateSport; urls: string[] }> = [];
- const unsupported: SlateSport[] = [];
+ // Sports that carry a schedule/streaks feed (ESPN-backed). Soccer
+ // has no schedule endpoint, so it stays odds-only.
+ const SCHEDULE_SPORTS = new Set(['nba', 'wnba', 'mlb']);
+
+ const sportsToFetch: SlateSport[] = [];
const consider = (s: Exclude) => {
- const urls = FETCH_URLS[s];
- if (urls === null) unsupported.push(s as SlateSport);
- else sportsToFetch.push({ sport: s as SlateSport, urls });
+ if (FETCH_URLS[s] !== null) sportsToFetch.push(s as SlateSport);
};
-
if (active === 'all') {
consider('nba'); consider('wnba'); consider('mlb'); consider('soccer');
} else {
@@ -225,64 +324,66 @@ export default function Slate({ initialTab = 'all', tier = 'free' }: SlateProps)
if (sportsToFetch.length === 0) {
setGames([]);
- setUnsupportedSports(unsupported);
setLoading(false);
return;
}
- const results = await Promise.allSettled(
- sportsToFetch.flatMap(({ sport, urls }) =>
- urls.map((url) =>
- fetch(url, { cache: 'no-store' })
- .then(async (r) => {
- const body = (await r.json().catch(() => ({}))) as OddsResponse;
- if (!r.ok) throw new Error(body?.error || `HTTP ${r.status}`);
- return { sport, body };
- })
- .catch((err) => {
- // Re-throw so allSettled catches it, but attach the
- // sport so the per-sport error-tracking below can
- // surface "Soccer odds unavailable" without blanking
- // the rest of the slate.
- const e = err instanceof Error ? err : new Error(String(err));
- (e as Error & { _vyndrSport?: SlateSport })._vyndrSport = sport;
- throw e;
- })
- ),
- ),
+ const getJson = async (url: string): Promise => {
+ try {
+ const r = await fetch(url, { cache: 'no-store' });
+ if (!r.ok) return null;
+ return (await r.json()) as T;
+ } catch {
+ return null;
+ }
+ };
+
+ // Per sport: odds + schedule + gamelines, all settled independently.
+ const perSport = await Promise.all(
+ sportsToFetch.map(async (sport) => {
+ const oddsUrls = FETCH_URLS[sport] as string[];
+ const [oddsResults, schedule, lines] = await Promise.all([
+ Promise.all(oddsUrls.map((u) => getJson(u))),
+ SCHEDULE_SPORTS.has(sport) ? getJson(`/api/schedule/${sport}`) : Promise.resolve(null),
+ SCHEDULE_SPORTS.has(sport) ? getJson(`/api/gamelines/${sport}`) : Promise.resolve(null),
+ ]);
+
+ const oddsOk = oddsResults.some((o) => o !== null);
+ const oddsProps = oddsResults.flatMap((o) => o?.props || []);
+ const oddsGames = groupByGame(oddsProps, sport);
+ const scheduleGames = schedule?.games || [];
+ const merged = mergeSlate(sport, scheduleGames, oddsGames, lines?.games);
+ return { sport, merged, oddsOk, hadSchedule: scheduleGames.length > 0 };
+ }),
);
const allGames: SlateGame[] = [];
- const failedSports: SlateSport[] = [];
- const sportsAttempted = new Set(sportsToFetch.map((s) => s.sport));
- const sportsThatSucceeded = new Set();
- for (const r of results) {
- if (r.status === 'fulfilled') {
- sportsThatSucceeded.add(r.value.sport);
- const grouped = groupByGame(r.value.body.props || [], r.value.sport);
- allGames.push(...grouped);
- } else {
- const failed = (r.reason as Error & { _vyndrSport?: SlateSport })._vyndrSport;
- if (failed && !failedSports.includes(failed)) failedSports.push(failed);
- }
+ let anyOddsOk = false;
+ let anyScheduleShown = false;
+ for (const s of perSport) {
+ allGames.push(...s.merged);
+ if (s.oddsOk) anyOddsOk = true;
+ if (s.hadSchedule) anyScheduleShown = true;
}
setGames(allGames);
- setUnsupportedSports([...unsupported, ...failedSports.filter((s) => !sportsThatSucceeded.has(s))]);
- // Session 17 — only surface a top-level error when EVERY sport
- // attempted in this tab failed. Partial successes (NBA ok,
- // soccer 503) silently drop the failed sport's row and surface
- // it via the existing "endpoint not configured" footer note.
- if (sportsAttempted.size > 0 && sportsThatSucceeded.size === 0) {
- const firstError = results.find((r) => r.status === 'rejected') as PromiseRejectedResult | undefined;
- setFetchError(firstError ? (firstError.reason as Error).message : 'Odds fetch failed');
+ // Odds down but schedule carried the slate → soft notice, not a wall.
+ if (!anyOddsOk && anyScheduleShown) setOddsNotice(true);
+ // Genuine total failure (no odds, no schedule, anywhere) → error.
+ if (!anyOddsOk && !anyScheduleShown && allGames.length === 0) {
+ setFetchError('No games available right now. Check back soon.');
}
setLoading(false);
}, []);
useEffect(() => { fetchSlate(tab); }, [tab, fetchSlate]);
+ // Session 24 — switching sport resets the stat filter. The categories
+ // differ per sport (Points vs Hits), so a stale "points" filter would
+ // silently blank the MLB panels. Always land back on 'all'.
+ useEffect(() => { setActiveStat('all'); }, [tab]);
+
// Grading call site. Single source of truth so we never have two
// PropRows in-flight from the same prop (the loadingKey enforces it).
const onGrade = useCallback(async (prop: PropRowProp) => {
@@ -426,13 +527,17 @@ export default function Slate({ initialTab = 'all', tier = 'free' }: SlateProps)
);
})}
- {/* Session 23 — stat filter pills, below the sport tabs and above
- all content. Narrows the streaks + hot list panels. */}
-
+ {/* Session 23/24 — stat filter pills, below the sport tabs and
+ above all content. Sport-specific categories. Hidden on the
+ ALL tab: filtering by "points" makes no sense when the slate
+ mixes NBA + MLB + soccer. Pills appear only on a single sport. */}
+ {tab !== 'all' && (
+
+ )}
{/* Body */}
@@ -467,6 +572,24 @@ export default function Slate({ initialTab = 'all', tier = 'free' }: SlateProps)
)}
+ {/* Session 24 — soft notice when props are loading but the schedule
+ (and lines) carry the slate. NOT a wall-of-error: the games are
+ right below it. */}
+ {oddsNotice && !loading && !fetchError && (
+
+ Player props are loading — today's schedule, game lines, and stats are shown below.
+
+ )}
+
{fetchError && !loading && (
- {unsupportedSports.length > 0 && !loading && (
-
- {unsupportedSports.map((s) => s.toUpperCase()).join(', ')} odds endpoint not configured yet.
-
- )}
+ {/* Session 24 — removed the developer-facing "odds endpoint not
+ configured yet" footer note. A sport with no data simply doesn't
+ render a row; users never see internal wiring state. */}
);
}