diff --git a/BUILD-STATE.md b/BUILD-STATE.md index 7f6a7c0..7bf467c 100755 --- a/BUILD-STATE.md +++ b/BUILD-STATE.md @@ -4,7 +4,113 @@ 2026-06-11 ## Current Phase -SHIP BUILD v15.0 — Intelligence hardening + platform correctness (Session 15) +SHIP BUILD v16.0 — Live hero prop + sport-scoped markets + launch polish (Session 16) + +## Session 16 (2026-06-11) — SHIPPED + +### Phase 1 — Sport-specific market map + +`src/services/oddsService.js` now scopes the markets-list parameter +to the requested sport. Previously every odds-api request sent +`ALL_MARKETS` (the union of every sport's markets), which the +upstream 422'd on because soccer markets (`player_goals`, +`player_shots_on_target`, etc.) aren't valid for basketball +endpoints. Production briefly worked around this with a runtime +axios interceptor injected via +`NODE_OPTIONS=--require /app/data/patch.js`. + +This session retires that hack at the code layer: +- New `SPORT_MARKETS` map alongside `SPORT_KEYS` — separate lists + per sport, all frozen with `Object.freeze`. NBA + NCAAB share + basketball markets; WNBA is basketball minus PRA (odds-api + doesn't carry that for WNBA); MLB sends batter + pitcher markets; + every soccer league shares the soccer set. +- `fetchEventOddsFromApi(sportKey, eventId, apiKey, sport)` — + third arg added; reads `getMarketsForSport(sport)` instead of + the union. Backwards-compatible: omitted sport falls back to + NBA (safe default). +- `fetchAllOdds(sport, apiKey)` — already had the local sport key; + now passes it through. + +**Coolify follow-up**: after this deploy, the operator can drop +`NODE_OPTIONS=--require /app/data/patch.js` from the web service +env and delete `/app/data/patch.js`. The runtime patch is now +dead code. + +### Phase 2 — Live hero prop + +`web/src/app/api/hero-prop/route.ts` (new) — picks one fresh real +prop from today's NBA → WNBA → MLB cascade and grades it. Two-stage +flow: GET `/api/odds/{sport}` → POST `/api/analyze/prop`. Both +calls share a 6s AbortController timeout. Server-side cached for +15 minutes via `Cache-Control: s-maxage=900`. Falls back to a +static Jokic example (`isStatic: true`) when every sport is empty +so the landing page never blanks out. + +`web/src/components/LiveHeroProp.tsx` (new) — replaces the +hard-coded `FloatingDemoCard` inside `Hero.tsx`. Renders the live +prop with: +- "LIVE" badge with a pulsing green dot +- Sport-colored category tag (NBA red, WNBA orange, MLB blue, soccer green) +- Player name + line + projection + edge **visible** (hook) +- Grade letter + confidence **visible** via GradePill (proof) +- Reasoning section **blurred** with backdrop `blur(4px)`, a + scan-line gradient (`repeating-linear-gradient`), a bottom-fade + mask, and a "CLASSIFIED · Sign up to unlock" label (paywall) +- Single CTA: "Sign up to read the full analysis →" + +While loading OR when the API returns `isStatic: true`, renders +the original Jokic mockup byte-for-byte. No flash-of-blank-card. + +`Hero.tsx` — old `FloatingDemoCard`, `Stat`, and `row` constant +deleted. `GradePill` import moved into `LiveHeroProp`. + +### Phase 3 — Soccer weather + +`soccerFeatureExtractor.js` now calls `weatherService.getWeather()` +for outdoor WC venues after resolving the venue. Dome venues skip +the fetch. Unknown venues skip silently. New feature fields: +`weather_temp_f`, `weather_wind_mph`, `weather_wind_dir`, +`weather_precip_mm`. All null when skipped/failed. + +### Phase 4/5 — OG tags + CSP (mostly already done) + +OG meta + Twitter card + `og-image.png` were all wired in Session 9. +Existing CSP in `next.config.ts` was comprehensive. Session 16 added: +- `https://browser.sentry-cdn.com` to `script-src` (Sentry SDK) +- `https://*.sentry.io` and `https://*.ingest.sentry.io` to + `connect-src` (event ingestion). Without these the browser + Sentry client silently dropped events. + +### Tests added (Session 16) +| Suite | Tests | +|----------------------------------------|-------| +| `tests/unit/sportMarkets.test.js` | 16 | +| `tests/unit/soccerWeather.test.js` | 7 | +| **Session 16 total** | **23**| + +### Quality gates +- `npm test`: **1429 / 1429 passing** (1405 + 24), 110 suites, 0 regressions +- `web/npm run build`: clean +- License audit: third-party deps remain permissive + +### Honest gaps +- `LiveHeroProp`'s glitch effect (scan lines + blur + fade) renders + only in a browser. Build verified. Deploy smoke-test recommended. +- Hero endpoint depends on `/api/odds/{sport}` returning populated + `props`. If upstream odds-api is rate-limited or proxies aren't + reaching Express, the static fallback fires — cold visitors see + the Jokic mockup, not live data. +- Sentry CSP entries added but require redeploy to take effect. + Until then, the browser SDK silently drops events. + +### Coolify follow-ups +1. **Drop the patch.js workaround**: remove + `NODE_OPTIONS=--require /app/data/patch.js` from the web + service env. Code-layer fix in Session 16 makes the runtime + patch obsolete. + +--- ## Session 15 (2026-06-11) — SHIPPED diff --git a/data/training/resolutions-2026-06.jsonl b/data/training/resolutions-2026-06.jsonl index ca973c8..38c4a59 100644 --- a/data/training/resolutions-2026-06.jsonl +++ b/data/training/resolutions-2026-06.jsonl @@ -556,3 +556,17 @@ {"ts":"2026-06-11T20:09:39.782Z","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-11T20:09:39.783Z","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-11T20:09:40.192Z","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-11T21:48:23.943Z","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-11T21:48:24.045Z","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-11T21:48:24.145Z","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-11T21:48:24.479Z","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-11T21:48:24.480Z","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-11T21:48:24.480Z","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-11T21:48:24.545Z","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-11T22:05:04.182Z","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-11T22:05:04.411Z","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-11T22:05:04.871Z","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-11T22:05:06.923Z","sport":"nba","player_espn_id":"3934719","player_name":"OG Anunoby","stat_type":"points","line":16.5,"direction":"over","actual_value":17,"result":"hit","margin":0.5,"grade":"A-"} +{"ts":"2026-06-11T22:05:06.923Z","sport":"nba","player_espn_id":"3136195","player_name":"Karl-Anthony Towns","stat_type":"rebounds","line":13.5,"direction":"under","actual_value":13,"result":"hit","margin":-0.5,"grade":"B+"} +{"ts":"2026-06-11T22:05:06.923Z","sport":"nba","player_espn_id":"3062679","player_name":"Josh Hart","stat_type":"pts_reb_ast","line":10.5,"direction":"over","actual_value":10,"result":"miss","margin":-0.5,"grade":"C"} +{"ts":"2026-06-11T22:05:07.154Z","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/services/intelligence/soccerFeatureExtractor.js b/src/services/intelligence/soccerFeatureExtractor.js index 99b04e5..99bb484 100644 --- a/src/services/intelligence/soccerFeatureExtractor.js +++ b/src/services/intelligence/soccerFeatureExtractor.js @@ -27,6 +27,12 @@ const { cacheGet } = require('../../utils/redis'); const { normalizeName } = require('../../utils/normalize'); const wc = require('../../data/worldcup2026'); +// Session 16 — World Cup venue weather. Open-Meteo lookup is cached +// 1h, has a 5s timeout, and degrades silently on failure. Dome +// venues skip the fetch entirely (operators close the roof when +// conditions warrant — weather doesn't drive grade in that case). +const { getWcVenueCoords } = require('../../data/venueCoordinates'); +const weatherService = require('../weatherService'); const SOCCER_SPORTS = new Set(['soccer', 'football']); @@ -183,6 +189,22 @@ async function extractSoccerFeatures(input = {}) { const homeContinent = wc.isHomeContinent(team); const altImpact = wc.altitudeImpact(altitudeFt); + // Session 16 — weather for outdoor World Cup venues. The venue + // coords file is keyed by venue name; dome venues (BC Place, + // AT&T, etc.) are skipped via the `dome` flag rather than the + // network call so we don't burn the 5s timeout on stadiums that + // will close the roof anyway. + let weather = null; + const coords = venueName ? getWcVenueCoords(venueName) : null; + if (coords && !coords.dome && Number.isFinite(coords.lat) && Number.isFinite(coords.lon)) { + try { + weather = await weatherService.getWeather(coords.lat, coords.lon); + } catch (err) { + // Best-effort — never block the grade on the weather fetch. + console.warn('[soccerFeatureExtractor] weather skipped:', err.message); + } + } + // Set-piece + penalty roles (static data — no async). const isPK = wc.isPenaltyTaker(player, team); const isCorner = wc.isCornerTaker(player, team); @@ -221,6 +243,13 @@ async function extractSoccerFeatures(input = {}) { venue_altitude_ft: altitudeFt, altitude_impact: altImpact, climate, + // Session 16 — weather. Null when venue is a dome / not in the + // WC venue index / Open-Meteo fetch failed. Trap detection + + // reasoning surface these signals when present. + weather_temp_f: weather?.temp_f ?? null, + weather_wind_mph: weather?.wind_mph ?? null, + weather_wind_dir: weather?.wind_dir ?? null, + weather_precip_mm: weather?.precip_mm ?? null, opp_goals_conceded_per_game: oppDefense?.goals_conceded_per_game ?? null, opp_clean_sheet_rate: oppDefense?.clean_sheet_rate ?? null, opp_defensive_rank: oppDefense?.defensive_rank ?? null, diff --git a/src/services/oddsService.js b/src/services/oddsService.js index 6fe2a30..97499f9 100644 --- a/src/services/oddsService.js +++ b/src/services/oddsService.js @@ -32,7 +32,89 @@ const SPORT_KEYS = { const SOCCER_SPORT_KEYS = Object.freeze( Object.keys(SPORT_KEYS).filter((k) => k.startsWith('soccer_')) ); +// Session 16 — per-sport market lists. +// +// The old `ALL_MARKETS = every key in MARKET_MAP` would send +// soccer markets (player_goals, player_shots_on_target, etc.) to +// basketball + baseball endpoints, triggering odds-api 422 errors. +// Production briefly worked around this with a runtime axios +// interceptor injected via `NODE_OPTIONS=--require /app/data/patch.js`; +// the proper fix is to scope the markets list to the sport before +// the request leaves the process. +// +// After this lands, the operator can drop the NODE_OPTIONS env var +// from Coolify and delete /app/data/patch.js. +const NBA_MARKETS = [ + 'player_points', + 'player_rebounds', + 'player_assists', + 'player_threes', + 'player_blocks', + 'player_steals', + 'player_turnovers', + 'player_points_rebounds_assists', +]; +const WNBA_MARKETS = [ + 'player_points', + 'player_rebounds', + 'player_assists', + 'player_threes', + 'player_blocks', + 'player_steals', + 'player_turnovers', +]; +const MLB_MARKETS = [ + 'batter_home_runs', + 'batter_hits', + 'batter_total_bases', + 'batter_rbis', + 'batter_runs_scored', + 'batter_stolen_bases', + 'pitcher_strikeouts', + 'pitcher_outs', +]; +const SOCCER_MARKETS = [ + 'player_goals', + 'player_shots_on_target', + 'player_shots', + 'player_tackles', + 'player_cards', + 'player_corners', + 'player_saves', + 'player_goals_conceded', + 'player_passes', + 'team_clean_sheet', +]; + +function buildMarketString(markets) { + return [...markets, 'spreads'].join(','); +} + +// Indexed by the local sport key (the keys in SPORT_KEYS, not the +// odds-api keys). Soccer leagues all share the same market list. +const SPORT_MARKETS = Object.freeze({ + nba: buildMarketString(NBA_MARKETS), + wnba: buildMarketString(WNBA_MARKETS), + mlb: buildMarketString(MLB_MARKETS), + ncaab: buildMarketString(NBA_MARKETS), // NCAAB markets mirror NBA + // Every soccer league code shares the same market set. + ...Object.fromEntries( + Object.keys(SPORT_KEYS) + .filter((k) => k.startsWith('soccer_')) + .map((k) => [k, buildMarketString(SOCCER_MARKETS)]), + ), +}); + +// Kept for backward-compat with any caller that still imports it, +// but the call site (`fetchEventOddsFromApi`) now uses the sport- +// specific lookup. Composed once on module load. const ALL_MARKETS = Object.keys(MARKET_MAP).join(',') + ',spreads'; + +function getMarketsForSport(sport) { + if (!sport) return SPORT_MARKETS.nba; // safe default (basketball) + return SPORT_MARKETS[sport] || SPORT_MARKETS.nba; +} + const BOOKMAKERS = 'draftkings,fanduel,betmgm,caesars,fanatics,bet365,hardrockbet,pointsbet,betrivers'; function getCacheKey(sport) { @@ -75,13 +157,17 @@ async function fetchEventsFromApi(sportKey, apiKey) { return { data: response.data, headers: response.headers }; } -async function fetchEventOddsFromApi(sportKey, eventId, apiKey) { +// Session 16 — third arg is now a local sport key (nba, mlb, +// soccer_wc, ...) so we can scope the markets list. Backwards- +// compatible: if `sport` is omitted, falls back to the basketball +// market set, which is what every legacy caller assumed. +async function fetchEventOddsFromApi(sportKey, eventId, apiKey, sport) { const url = `${ODDS_API_BASE}/${sportKey}/events/${eventId}/odds`; const response = await axios.get(url, { params: { apiKey, regions: 'us', - markets: ALL_MARKETS, + markets: getMarketsForSport(sport), bookmakers: BOOKMAKERS, oddsFormat: 'american', }, @@ -112,7 +198,7 @@ async function fetchAllOdds(sport, apiKey) { break; } - const oddsResult = await fetchEventOddsFromApi(sportKey, event.id, apiKey); + const oddsResult = await fetchEventOddsFromApi(sportKey, event.id, apiKey, sport); eventsWithOdds.push(oddsResult.data); lastHeaders = oddsResult.headers; } @@ -230,6 +316,9 @@ module.exports = { getCacheKey, SPORT_KEYS, SOCCER_SPORT_KEYS, + // Session 16 — per-sport market scoping. + SPORT_MARKETS, + getMarketsForSport, getQuotaKey, updateQuota, getQuotaRemaining, diff --git a/tests/unit/soccerWeather.test.js b/tests/unit/soccerWeather.test.js new file mode 100644 index 0000000..902831a --- /dev/null +++ b/tests/unit/soccerWeather.test.js @@ -0,0 +1,116 @@ +// Session 16 — soccer weather wiring. The feature extractor fetches +// Open-Meteo for outdoor WC venues. Dome venues skip the fetch +// (operators close the roof); unknown venues skip silently. + +const mockCache = new Map(); +jest.mock('../../src/utils/redis', () => ({ + cacheGet: async (k) => (mockCache.has(k) ? mockCache.get(k) : null), + cacheSet: async (k, v) => { mockCache.set(k, v); return true; }, + cacheDel: async (k) => { mockCache.delete(k); return true; }, + isDegraded: () => false, +})); + +const mockWeather = jest.fn(); +jest.mock('../../src/services/weatherService', () => ({ + getWeather: (...a) => mockWeather(...a), +})); + +const { extractSoccerFeatures } = require('../../src/services/intelligence/soccerFeatureExtractor'); +const { normalizeName } = require('../../src/utils/normalize'); + +beforeEach(() => { + mockCache.clear(); + mockWeather.mockReset(); +}); + +function primePlayerAndMatch(player, team, opts = {}) { + mockCache.set(`soccer:player:${normalizeName(player)}`, { + team, goals_per_90: 0.5, + }); + mockCache.set(`soccer:nextmatch:${team}`, { + opponent: opts.opponent || 'X', + venue: opts.venue || 'MetLife Stadium', + isHome: opts.isHome ?? true, + referee: opts.referee || null, + }); +} + +describe('soccer weather wiring (Session 16)', () => { + test('outdoor WC venue → weather features populated', async () => { + primePlayerAndMatch('Harry Kane', 'England', { venue: 'MetLife Stadium' }); + mockWeather.mockResolvedValueOnce({ + temp_f: 81.2, wind_mph: 9.4, wind_dir: 220, precip_mm: 0, + }); + + const r = await extractSoccerFeatures({ + player: 'Harry Kane', stat_type: 'goals', line: 0.5, direction: 'over', + }); + + expect(mockWeather).toHaveBeenCalledTimes(1); + expect(r.features.weather_temp_f).toBeCloseTo(81.2); + expect(r.features.weather_wind_mph).toBeCloseTo(9.4); + expect(r.features.weather_wind_dir).toBe(220); + expect(r.features.weather_precip_mm).toBe(0); + }); + + test('dome WC venue (BC Place) → weather fetch skipped, fields null', async () => { + primePlayerAndMatch('Sub Player', 'Canada', { venue: 'BC Place' }); + const r = await extractSoccerFeatures({ + player: 'Sub Player', stat_type: 'goals', line: 0.5, direction: 'over', + }); + expect(mockWeather).not.toHaveBeenCalled(); + expect(r.features.weather_temp_f).toBeNull(); + expect(r.features.weather_wind_mph).toBeNull(); + }); + + test('Estadio Azteca (open-air, high-altitude) → weather fetched + altitude_impact still high', async () => { + primePlayerAndMatch('Forward', 'Mexico', { venue: 'Estadio Azteca' }); + mockWeather.mockResolvedValueOnce({ temp_f: 68, wind_mph: 5, wind_dir: 90, precip_mm: 0 }); + const r = await extractSoccerFeatures({ + player: 'Forward', stat_type: 'goals', line: 0.5, direction: 'over', + }); + expect(r.features.weather_temp_f).toBe(68); + expect(r.features.altitude_impact).toBe('high'); + }); + + test('venue not in the WC index → weather fetch skipped', async () => { + primePlayerAndMatch('X', 'Y', { venue: 'Random Stadium' }); + const r = await extractSoccerFeatures({ + player: 'X', stat_type: 'goals', line: 0.5, direction: 'over', + }); + expect(mockWeather).not.toHaveBeenCalled(); + expect(r.features.weather_temp_f).toBeNull(); + }); + + test('weather service returns null → feature fields stay null (no throw)', async () => { + primePlayerAndMatch('X', 'England', { venue: 'MetLife Stadium' }); + mockWeather.mockResolvedValueOnce(null); + const r = await extractSoccerFeatures({ + player: 'X', stat_type: 'goals', line: 0.5, direction: 'over', + }); + expect(mockWeather).toHaveBeenCalledTimes(1); + expect(r.features.weather_temp_f).toBeNull(); + expect(r.features.weather_wind_mph).toBeNull(); + }); + + test('weather service throws → graceful degrade, grade still produced', async () => { + primePlayerAndMatch('X', 'England', { venue: 'MetLife Stadium' }); + mockWeather.mockRejectedValueOnce(new Error('timeout')); + const r = await extractSoccerFeatures({ + player: 'X', stat_type: 'goals', line: 0.5, direction: 'over', + }); + expect(r.features.weather_temp_f).toBeNull(); + // Other features (goals_per_90 etc.) still populated. + expect(r.features.goals_per_90).toBe(0.5); + }); + + test('no venue resolved → weather skipped entirely (no fetch attempt)', async () => { + mockCache.set(`soccer:player:${normalizeName('Solo')}`, { team: 'England', goals_per_90: 0.5 }); + // No nextmatch entry → venueName is null. + const r = await extractSoccerFeatures({ + player: 'Solo', stat_type: 'goals', line: 0.5, direction: 'over', + }); + expect(mockWeather).not.toHaveBeenCalled(); + expect(r.features.weather_temp_f).toBeNull(); + }); +}); diff --git a/tests/unit/sportMarkets.test.js b/tests/unit/sportMarkets.test.js new file mode 100644 index 0000000..5bdadbb --- /dev/null +++ b/tests/unit/sportMarkets.test.js @@ -0,0 +1,129 @@ +// Session 16 — sport-specific market scoping in oddsService. +// Replaces a runtime axios interceptor (NODE_OPTIONS --require) that +// had been deployed to filter out cross-sport markets the upstream +// odds-api 422s on. These tests pin the contract so the runtime hack +// can be retired safely. + +const { SPORT_MARKETS, getMarketsForSport } = require('../../src/services/oddsService'); + +describe('SPORT_MARKETS — isolation', () => { + test('NBA market list contains no soccer markets', () => { + expect(SPORT_MARKETS.nba).not.toMatch(/player_goals\b/); + expect(SPORT_MARKETS.nba).not.toMatch(/player_shots_on_target/); + expect(SPORT_MARKETS.nba).not.toMatch(/player_tackles/); + expect(SPORT_MARKETS.nba).not.toMatch(/player_cards/); + expect(SPORT_MARKETS.nba).not.toMatch(/team_clean_sheet/); + }); + + test('NBA market list contains no MLB markets', () => { + expect(SPORT_MARKETS.nba).not.toMatch(/batter_/); + expect(SPORT_MARKETS.nba).not.toMatch(/pitcher_/); + }); + + test('NBA market list does contain canonical NBA markets', () => { + expect(SPORT_MARKETS.nba).toMatch(/player_points/); + expect(SPORT_MARKETS.nba).toMatch(/player_rebounds/); + expect(SPORT_MARKETS.nba).toMatch(/player_assists/); + expect(SPORT_MARKETS.nba).toMatch(/player_threes/); + expect(SPORT_MARKETS.nba).toMatch(/spreads/); + }); + + test('WNBA market list is NBA-shaped minus PRA combo', () => { + expect(SPORT_MARKETS.wnba).toMatch(/player_points/); + expect(SPORT_MARKETS.wnba).toMatch(/player_rebounds/); + // WNBA odds-api doesn't expose the PRA combo today. + expect(SPORT_MARKETS.wnba).not.toMatch(/points_rebounds_assists/); + expect(SPORT_MARKETS.wnba).not.toMatch(/batter_/); + expect(SPORT_MARKETS.wnba).not.toMatch(/player_goals\b/); + }); + + test('MLB market list contains batter + pitcher markets, no basketball', () => { + expect(SPORT_MARKETS.mlb).toMatch(/batter_home_runs/); + expect(SPORT_MARKETS.mlb).toMatch(/batter_hits/); + expect(SPORT_MARKETS.mlb).toMatch(/pitcher_strikeouts/); + expect(SPORT_MARKETS.mlb).not.toMatch(/player_points/); + expect(SPORT_MARKETS.mlb).not.toMatch(/player_goals\b/); + }); + + test('every soccer league shares the same market list', () => { + const soccerKeys = Object.keys(SPORT_MARKETS).filter((k) => k.startsWith('soccer_')); + expect(soccerKeys.length).toBeGreaterThanOrEqual(9); + const first = SPORT_MARKETS[soccerKeys[0]]; + for (const k of soccerKeys) { + expect(SPORT_MARKETS[k]).toBe(first); + } + }); + + test('soccer market list contains soccer-only markets, no basketball/baseball', () => { + const wc = SPORT_MARKETS.soccer_wc; + expect(wc).toMatch(/player_goals/); + expect(wc).toMatch(/player_shots_on_target/); + expect(wc).toMatch(/player_cards/); + expect(wc).toMatch(/team_clean_sheet/); + expect(wc).not.toMatch(/player_points/); + expect(wc).not.toMatch(/batter_/); + }); + + test('every market list ends with `spreads`', () => { + for (const list of Object.values(SPORT_MARKETS)) { + // We don't require spreads to be the literal final segment, + // only that it's present in the comma-separated list. + expect(list.split(',')).toContain('spreads'); + } + }); + + test('SPORT_MARKETS is frozen at the top level', () => { + expect(Object.isFrozen(SPORT_MARKETS)).toBe(true); + }); +}); + +describe('getMarketsForSport', () => { + test('returns the NBA list for nba', () => { + expect(getMarketsForSport('nba')).toBe(SPORT_MARKETS.nba); + }); + test('returns the soccer_wc list for soccer_wc', () => { + expect(getMarketsForSport('soccer_wc')).toBe(SPORT_MARKETS.soccer_wc); + }); + test('unknown sport falls back to NBA (safe default)', () => { + expect(getMarketsForSport('cricket')).toBe(SPORT_MARKETS.nba); + }); + test('null / undefined / empty fall back to NBA', () => { + expect(getMarketsForSport(null)).toBe(SPORT_MARKETS.nba); + expect(getMarketsForSport(undefined)).toBe(SPORT_MARKETS.nba); + expect(getMarketsForSport('')).toBe(SPORT_MARKETS.nba); + }); +}); + +describe('fetchEventOddsFromApi uses sport-scoped markets', () => { + // Mock axios so the test doesn't hit the network. + jest.resetModules(); + const mockGet = jest.fn(() => Promise.resolve({ data: {}, headers: {} })); + jest.doMock('axios', () => ({ get: mockGet })); + const { fetchEventOddsFromApi, SPORT_MARKETS: SM } = require('../../src/services/oddsService'); + + beforeEach(() => mockGet.mockClear()); + + test('NBA fetch sends NBA markets only', async () => { + await fetchEventOddsFromApi('basketball_nba', 'evt1', 'key', 'nba'); + const [, opts] = mockGet.mock.calls[0]; + expect(opts.params.markets).toBe(SM.nba); + }); + + test('MLB fetch sends MLB markets only', async () => { + await fetchEventOddsFromApi('baseball_mlb', 'evt1', 'key', 'mlb'); + const [, opts] = mockGet.mock.calls[0]; + expect(opts.params.markets).toBe(SM.mlb); + }); + + test('soccer fetch sends soccer markets only', async () => { + await fetchEventOddsFromApi('soccer_fifa_world_cup', 'evt1', 'key', 'soccer_wc'); + const [, opts] = mockGet.mock.calls[0]; + expect(opts.params.markets).toBe(SM.soccer_wc); + }); + + test('omitted sport arg falls back to NBA markets (legacy callers)', async () => { + await fetchEventOddsFromApi('basketball_nba', 'evt1', 'key'); + const [, opts] = mockGet.mock.calls[0]; + expect(opts.params.markets).toBe(SM.nba); + }); +}); diff --git a/web/next.config.ts b/web/next.config.ts index 527e28a..b21a869 100644 --- a/web/next.config.ts +++ b/web/next.config.ts @@ -15,11 +15,15 @@ import withBundleAnalyzer from '@next/bundle-analyzer'; // - Supabase wss: AuthContext realtime + push subscriptions const CSP = [ "default-src 'self'", - "script-src 'self' 'unsafe-eval' 'unsafe-inline' https://js.stripe.com https://us-assets.i.posthog.com", + "script-src 'self' 'unsafe-eval' 'unsafe-inline' https://js.stripe.com https://us-assets.i.posthog.com https://browser.sentry-cdn.com", "style-src 'self' 'unsafe-inline' https://fonts.googleapis.com", "font-src 'self' https://fonts.gstatic.com", "img-src 'self' data: blob: https://*.supabase.co https://cdn.nba.com https://a.espncdn.com", - "connect-src 'self' https://*.supabase.co wss://*.supabase.co https://api.stripe.com https://us.i.posthog.com https://us-assets.i.posthog.com", + // Session 16 — Sentry browser client posts events to *.sentry.io + // (and *.ingest.sentry.io for the ingestion endpoints). Adding + // both forms so the @sentry/nextjs init in SentryInit.tsx can + // actually report errors out of the browser bundle. + "connect-src 'self' https://*.supabase.co wss://*.supabase.co https://api.stripe.com https://us.i.posthog.com https://us-assets.i.posthog.com https://*.sentry.io https://*.ingest.sentry.io", "frame-src https://js.stripe.com https://hooks.stripe.com", "worker-src 'self' blob:", "manifest-src 'self'", diff --git a/web/public/sw.js b/web/public/sw.js index efee20c..06ca55b 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':'93077fba92ceabe8021ae34e55942ad6','url':'/_next/static/ZDvereeObVypTQtmIwkSx/_buildManifest.js'},{'revision':'b6652df95db52feb4daf4eca35380933','url':'/_next/static/ZDvereeObVypTQtmIwkSx/_ssgManifest.js'},{'revision':null,'url':'/_next/static/chunks/1896-4649d6ca747442e6.js'},{'revision':null,'url':'/_next/static/chunks/1958-a1cd681eb2c1a83e.js'},{'revision':null,'url':'/_next/static/chunks/2346-d508a4289748cd4a.js'},{'revision':null,'url':'/_next/static/chunks/4bd1b696-c2f6e0877b6c10aa.js'},{'revision':null,'url':'/_next/static/chunks/52774a7f.8203600637b1464d.js'},{'revision':null,'url':'/_next/static/chunks/5838-7ac6ecc648315323.js'},{'revision':null,'url':'/_next/static/chunks/6810-8b074082a0b51859.js'},{'revision':null,'url':'/_next/static/chunks/7602.cfbf0dc56b47f93b.js'},{'revision':null,'url':'/_next/static/chunks/7918-840449b91e99704b.js'},{'revision':null,'url':'/_next/static/chunks/8500-3953cc33eeceb44e.js'},{'revision':null,'url':'/_next/static/chunks/9da6db1e-54d62c4d2564a0ac.js'},{'revision':null,'url':'/_next/static/chunks/app/_global-error/page-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/_not-found/page-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/checkout/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/props/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/tonight/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/intelligence/feed/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/accuracy/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/mlb/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/nba/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/soccer/%5Bleague%5D/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/wnba/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/add-leg/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/grade/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/players/search/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/live/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/most-parlayed/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/top-graded/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/scan/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/parlays-graded/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/public/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/profile/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/recent-scans/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/scans/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/waitlist/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/webhook/nexapay/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/api/welcome-email/route-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/auth/callback/page-f6253334ccab6a26.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/%5Bslug%5D/page-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/page-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/dashboard/page-5cd033ea13d5ab76.js'},{'revision':null,'url':'/_next/static/chunks/app/forgot-password/page-012a02e7c1c1bf4c.js'},{'revision':null,'url':'/_next/static/chunks/app/game/%5Bid%5D/page-1c70506fd9665dbf.js'},{'revision':null,'url':'/_next/static/chunks/app/intelligence/page-f6536af186e4c75a.js'},{'revision':null,'url':'/_next/static/chunks/app/layout-a7bd13dc3b447906.js'},{'revision':null,'url':'/_next/static/chunks/app/ledger/page-b2f371ae0e2dc445.js'},{'revision':null,'url':'/_next/static/chunks/app/login/page-7560a04fb2b26dad.js'},{'revision':null,'url':'/_next/static/chunks/app/marketplace/page-ce0df41afd789a48.js'},{'revision':null,'url':'/_next/static/chunks/app/not-found-3d8359a06d3506dc.js'},{'revision':null,'url':'/_next/static/chunks/app/page-0d695fcd5650fe29.js'},{'revision':null,'url':'/_next/static/chunks/app/pricing/page-e0324df275d75d0f.js'},{'revision':null,'url':'/_next/static/chunks/app/privacy/page-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/profile/page-1d6940745beb4eba.js'},{'revision':null,'url':'/_next/static/chunks/app/responsible-gambling/page-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/scan/page-06e0d1254e4a1b76.js'},{'revision':null,'url':'/_next/static/chunks/app/settings/security/page-5a6b00ebb8de6035.js'},{'revision':null,'url':'/_next/static/chunks/app/signup/page-f1e96999abbeccb5.js'},{'revision':null,'url':'/_next/static/chunks/app/soccer/page-bc585d14e3b7d415.js'},{'revision':null,'url':'/_next/static/chunks/app/terms/page-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/app/tracker/page-0d544936961f5807.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/cancel/page-3d8359a06d3506dc.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/desk/page-9258a8a7eeebe655.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/success/page-6b2f47a27c91344b.js'},{'revision':null,'url':'/_next/static/chunks/app/verify/page-4d522ede91624bab.js'},{'revision':null,'url':'/_next/static/chunks/app/welcome/page-b2b44714c0aa7d32.js'},{'revision':null,'url':'/_next/static/chunks/framework-29591c88a1db84ab.js'},{'revision':null,'url':'/_next/static/chunks/main-app-f54c179578b140aa.js'},{'revision':null,'url':'/_next/static/chunks/main-c31eab22221c05bc.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/app-error-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/forbidden-908bd5bde21a07fe.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/global-error-d5beecc31181c673.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/unauthorized-908bd5bde21a07fe.js'},{'revision':'846118c33b2c0e922d7b3a7676f81f6f','url':'/_next/static/chunks/polyfills-42372ed130431b0a.js'},{'revision':null,'url':'/_next/static/chunks/webpack-046cc6705514b5bd.js'},{'revision':null,'url':'/_next/static/css/ef4d31504fa635a6.css'},{'revision':'d7bc0f63f9618b8b700d028d7d242de8','url':'/apple-touch-icon.png'},{'revision':'5616f81ec5f8a9f726a17fb7ac66a54a','url':'/favicon-16.png'},{'revision':'1fc285cfb0116a0523eb34c779a7d282','url':'/favicon-32.png'},{'revision':'583b034f36839a8af92841771eef38b6','url':'/favicon.ico'},{'revision':'b15795128b3691e5703d0ce14ce10827','url':'/favicon.png'},{'revision':'44e9d71551be13574ac2e10063d03aa9','url':'/favicon.svg'},{'revision':'252b8f8c4d0f64c7cd120f182a6c74ab','url':'/icon-192.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icon-512.png'},{'revision':'252b8f8c4d0f64c7cd120f182a6c74ab','url':'/icons/icon-192.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icons/icon-512.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icons/icon-maskable-512.png'},{'revision':'27007d181514e7fe4213035736a8300e','url':'/manifest.json'},{'revision':'9f61cf51298661d5c57a782534216240','url':'/og-image.png'},{'revision':'7c759c20b35840eacf9344692cb51061','url':'/og-image.svg'},{'revision':'9327802333275f11e68c3f19f20c160d','url':'/widget.js'}],skipWaiting:!0,clientsClaim:!0,navigationPreload:!0,runtimeCaching:eq}).addEventListeners(),self.addEventListener("push",e=>{let t;if(!e.data)return;try{t=e.data.json()}catch{t={title:"VYNDR",body:e.data.text()}}let{title:a="VYNDR",body:s="",icon:r="/icons/icon-192.png",url:n="/"}=t;e.waitUntil(self.registration.showNotification(a,{body:s,icon:r,badge:"/icons/icon-192.png",data:{url:n}}))}),self.addEventListener("notificationclick",e=>{e.notification.close();let t=e.notification.data?.url??"/";e.waitUntil(self.clients.matchAll({type:"window",includeUncontrolled:!0}).then(e=>{let a=e.find(e=>e.url.endsWith(t));return a?a.focus():self.clients.openWindow(t)}))})})(); \ No newline at end of file +This is generally NOT safe. Learn more at https://bit.ly/wb-precache`)}handleInstall(e){return this.registerRequestRules(e),y(e,async()=>{let t=new eo;this.precacheStrategy.plugins.push(t),await el(this._concurrentPrecaching,Array.from(this._urlsToCacheKeys.entries()),async([t,a])=>{let s=this._cacheKeysToIntegrities.get(a),r=this._urlsToCacheModes.get(t),n=new Request(t,{integrity:s,cache:r,credentials:"same-origin"});await Promise.all(this.precacheStrategy.handleAll({event:e,request:n,url:new URL(n.url),params:{cacheKey:a}}))});let{updatedURLs:a,notUpdatedURLs:s}=t;return{updatedURLs:a,notUpdatedURLs:s}})}async registerRequestRules(e){if(this._requestRules&&e?.addRoutes)try{await e.addRoutes(this._requestRules),this._requestRules=void 0}catch(e){throw e}}handleActivate(e){return y(e,async()=>{let e=await self.caches.open(this.precacheStrategy.cacheName),t=await e.keys(),a=new Set(this._urlsToCacheKeys.values()),s=[];for(let r of t)a.has(r.url)||(await e.delete(r),s.push(r.url));return{deletedCacheRequests:s}})}handleFetch(e){let{request:t}=e,a=this.handleRequest({request:t,event:e});a&&e.respondWith(a)}handleCache(e){if(e.data&&"CACHE_URLS"===e.data.type){let{payload:t}=e.data,a=Promise.all(t.urlsToCache.map(t=>{let a;return a="string"==typeof t?new Request(t):new Request(...t),this.handleRequest({request:a,event:e})}));e.waitUntil(a),e.ports?.[0]&&a.then(()=>e.ports[0].postMessage(!0))}}setDefaultHandler(e,t="GET"){this._defaultHandlerMap.set(t,ea(e))}setCatchHandler(e){this._catchHandler=ea(e)}registerCapture(e,t,a){let s=((e,t,a)=>{if("string"==typeof e){let s=new URL(e,location.href);return new es(({url:e})=>e.href===s.href,t,a)}if(e instanceof RegExp)return new ei(e,t,a);if("function"==typeof e)return new es(e,t,a);if(e instanceof es)return e;throw new l("unsupported-route-type",{moduleName:"serwist",funcName:"parseRoute",paramName:"capture"})})(e,t,a);return this.registerRoute(s),s}registerRoute(e){this._routes.has(e.method)||this._routes.set(e.method,[]),this._routes.get(e.method).push(e)}unregisterRoute(e){if(!this._routes.has(e.method))throw new l("unregister-route-but-not-found-with-method",{method:e.method});let t=this._routes.get(e.method).indexOf(e);if(t>-1)this._routes.get(e.method).splice(t,1);else throw new l("unregister-route-route-not-registered")}getUrlsToPrecacheKeys(){return this._urlsToCacheKeys}getPrecachedUrls(){return[...this._urlsToCacheKeys.keys()]}getPrecacheKeyForUrl(e){let t=new URL(e,location.href);return this._urlsToCacheKeys.get(t.href)}getIntegrityForPrecacheKey(e){return this._cacheKeysToIntegrities.get(e)}async matchPrecache(e){let t=e instanceof Request?e.url:e,a=this.getPrecacheKeyForUrl(t);if(a)return(await self.caches.open(this.precacheStrategy.cacheName)).match(a)}createHandlerBoundToUrl(e){let t=this.getPrecacheKeyForUrl(e);if(!t)throw new l("non-precached-url",{url:e});return a=>(a.request=new Request(e),a.params={cacheKey:t,...a.params},this.precacheStrategy.handle(a))}handleRequest({request:e,event:t}){let a,s=new URL(e.url,location.href);if(!s.protocol.startsWith("http"))return;let r=s.origin===location.origin,{params:n,route:i}=this.findMatchingRoute({event:t,request:e,sameOrigin:r,url:s}),c=i?.handler,o=e.method;if(!c&&this._defaultHandlerMap.has(o)&&(c=this._defaultHandlerMap.get(o)),!c)return;try{a=c.handle({url:s,request:e,event:t,params:n})}catch(e){a=Promise.reject(e)}let l=i?.catchHandler;return a instanceof Promise&&(this._catchHandler||l)&&(a=a.catch(async a=>{if(l)try{return await l.handle({url:s,request:e,event:t,params:n})}catch(e){e instanceof Error&&(a=e)}if(this._catchHandler)return this._catchHandler.handle({url:s,request:e,event:t});throw a})),a}findMatchingRoute({url:e,sameOrigin:t,request:a,event:s}){for(let r of this._routes.get(a.method)||[]){let n,i=r.match({url:e,sameOrigin:t,request:a,event:s});if(i)return Array.isArray(n=i)&&0===n.length||i.constructor===Object&&0===Object.keys(i).length?n=void 0:"boolean"==typeof i&&(n=void 0),{route:r,params:n}}return{}}};let eq=[{matcher:/^https:\/\/fonts\.(?:gstatic)\.com\/.*/i,handler:new ex({cacheName:"google-fonts-webfonts",plugins:[new ef({maxEntries:4,maxAgeSeconds:31536e3,maxAgeFrom:"last-used"})]})},{matcher:/^https:\/\/fonts\.(?:googleapis)\.com\/.*/i,handler:new eb({cacheName:"google-fonts-stylesheets",plugins:[new ef({maxEntries:4,maxAgeSeconds:604800,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:eot|otf|ttc|ttf|woff|woff2|font.css)$/i,handler:new eb({cacheName:"static-font-assets",plugins:[new ef({maxEntries:4,maxAgeSeconds:604800,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:jpg|jpeg|gif|png|svg|ico|webp)$/i,handler:new eb({cacheName:"static-image-assets",plugins:[new ef({maxEntries:64,maxAgeSeconds:2592e3,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/static.+\.js$/i,handler:new ex({cacheName:"next-static-js-assets",plugins:[new ef({maxEntries:64,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/image\?url=.+$/i,handler:new eb({cacheName:"next-image",plugins:[new ef({maxEntries:64,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:mp3|wav|ogg)$/i,handler:new ex({cacheName:"static-audio-assets",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"}),new e_]})},{matcher:/\.(?:mp4|webm)$/i,handler:new ex({cacheName:"static-video-assets",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"}),new e_]})},{matcher:/\.(?:js)$/i,handler:new eb({cacheName:"static-js-assets",plugins:[new ef({maxEntries:48,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:css|less)$/i,handler:new eb({cacheName:"static-style-assets",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/data\/.+\/.+\.json$/i,handler:new ee({cacheName:"next-data",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:json|xml|csv)$/i,handler:new ee({cacheName:"static-data-assets",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/api\/auth\/.*/,handler:new et({networkTimeoutSeconds:10})},{matcher:({sameOrigin:e,url:{pathname:t}})=>e&&t.startsWith("/api/"),method:"GET",handler:new ee({cacheName:"apis",plugins:[new ef({maxEntries:16,maxAgeSeconds:86400,maxAgeFrom:"last-used"})],networkTimeoutSeconds:10})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>"1"===e.headers.get("RSC")&&"1"===e.headers.get("Next-Router-Prefetch")&&a&&!t.startsWith("/api/"),handler:new ee({cacheName:"pages-rsc-prefetch",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>"1"===e.headers.get("RSC")&&a&&!t.startsWith("/api/"),handler:new ee({cacheName:"pages-rsc",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>e.headers.get("Content-Type")?.includes("text/html")&&a&&!t.startsWith("/api/"),handler:new ee({cacheName:"pages",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({url:{pathname:e},sameOrigin:t})=>t&&!e.startsWith("/api/"),handler:new ee({cacheName:"others",plugins:[new ef({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({sameOrigin:e})=>!e,handler:new ee({cacheName:"cross-origin",plugins:[new ef({maxEntries:32,maxAgeSeconds:3600})],networkTimeoutSeconds:10})},{matcher:/.*/i,method:"GET",handler:new et}];new eR({precacheEntries:[{'revision':'d11babada5634866704961e34f78433c','url':'/_next/static/ED9Mp8OTYw-d5UgGCnf1C/_buildManifest.js'},{'revision':'b6652df95db52feb4daf4eca35380933','url':'/_next/static/ED9Mp8OTYw-d5UgGCnf1C/_ssgManifest.js'},{'revision':null,'url':'/_next/static/chunks/1896-4649d6ca747442e6.js'},{'revision':null,'url':'/_next/static/chunks/1958-a1cd681eb2c1a83e.js'},{'revision':null,'url':'/_next/static/chunks/2346-d508a4289748cd4a.js'},{'revision':null,'url':'/_next/static/chunks/4bd1b696-c2f6e0877b6c10aa.js'},{'revision':null,'url':'/_next/static/chunks/52774a7f.8203600637b1464d.js'},{'revision':null,'url':'/_next/static/chunks/5838-7ac6ecc648315323.js'},{'revision':null,'url':'/_next/static/chunks/6810-8b074082a0b51859.js'},{'revision':null,'url':'/_next/static/chunks/7602.cfbf0dc56b47f93b.js'},{'revision':null,'url':'/_next/static/chunks/7918-840449b91e99704b.js'},{'revision':null,'url':'/_next/static/chunks/8500-3953cc33eeceb44e.js'},{'revision':null,'url':'/_next/static/chunks/9da6db1e-54d62c4d2564a0ac.js'},{'revision':null,'url':'/_next/static/chunks/app/_global-error/page-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/_not-found/page-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/checkout/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/props/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/tonight/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/hero-prop/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/intelligence/feed/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/accuracy/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/mlb/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/nba/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/soccer/%5Bleague%5D/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/wnba/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/add-leg/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/grade/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/players/search/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/live/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/most-parlayed/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/top-graded/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/scan/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/parlays-graded/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/public/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/profile/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/recent-scans/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/scans/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/waitlist/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/webhook/nexapay/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/api/welcome-email/route-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/auth/callback/page-f6253334ccab6a26.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/%5Bslug%5D/page-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/page-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/dashboard/page-5cd033ea13d5ab76.js'},{'revision':null,'url':'/_next/static/chunks/app/forgot-password/page-012a02e7c1c1bf4c.js'},{'revision':null,'url':'/_next/static/chunks/app/game/%5Bid%5D/page-1c70506fd9665dbf.js'},{'revision':null,'url':'/_next/static/chunks/app/intelligence/page-f6536af186e4c75a.js'},{'revision':null,'url':'/_next/static/chunks/app/layout-a7bd13dc3b447906.js'},{'revision':null,'url':'/_next/static/chunks/app/ledger/page-b2f371ae0e2dc445.js'},{'revision':null,'url':'/_next/static/chunks/app/login/page-7560a04fb2b26dad.js'},{'revision':null,'url':'/_next/static/chunks/app/marketplace/page-ce0df41afd789a48.js'},{'revision':null,'url':'/_next/static/chunks/app/not-found-3d8359a06d3506dc.js'},{'revision':null,'url':'/_next/static/chunks/app/page-d94245d7452465c5.js'},{'revision':null,'url':'/_next/static/chunks/app/pricing/page-e0324df275d75d0f.js'},{'revision':null,'url':'/_next/static/chunks/app/privacy/page-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/profile/page-1d6940745beb4eba.js'},{'revision':null,'url':'/_next/static/chunks/app/responsible-gambling/page-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/scan/page-06e0d1254e4a1b76.js'},{'revision':null,'url':'/_next/static/chunks/app/settings/security/page-5a6b00ebb8de6035.js'},{'revision':null,'url':'/_next/static/chunks/app/signup/page-f1e96999abbeccb5.js'},{'revision':null,'url':'/_next/static/chunks/app/soccer/page-bc585d14e3b7d415.js'},{'revision':null,'url':'/_next/static/chunks/app/terms/page-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/app/tracker/page-0d544936961f5807.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/cancel/page-3d8359a06d3506dc.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/desk/page-9258a8a7eeebe655.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/success/page-6b2f47a27c91344b.js'},{'revision':null,'url':'/_next/static/chunks/app/verify/page-4d522ede91624bab.js'},{'revision':null,'url':'/_next/static/chunks/app/welcome/page-b2b44714c0aa7d32.js'},{'revision':null,'url':'/_next/static/chunks/framework-29591c88a1db84ab.js'},{'revision':null,'url':'/_next/static/chunks/main-app-f54c179578b140aa.js'},{'revision':null,'url':'/_next/static/chunks/main-c31eab22221c05bc.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/app-error-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/forbidden-28b6751a44ac89fa.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/global-error-d5beecc31181c673.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/unauthorized-28b6751a44ac89fa.js'},{'revision':'846118c33b2c0e922d7b3a7676f81f6f','url':'/_next/static/chunks/polyfills-42372ed130431b0a.js'},{'revision':null,'url':'/_next/static/chunks/webpack-046cc6705514b5bd.js'},{'revision':null,'url':'/_next/static/css/f6bb170551914341.css'},{'revision':'d7bc0f63f9618b8b700d028d7d242de8','url':'/apple-touch-icon.png'},{'revision':'5616f81ec5f8a9f726a17fb7ac66a54a','url':'/favicon-16.png'},{'revision':'1fc285cfb0116a0523eb34c779a7d282','url':'/favicon-32.png'},{'revision':'583b034f36839a8af92841771eef38b6','url':'/favicon.ico'},{'revision':'b15795128b3691e5703d0ce14ce10827','url':'/favicon.png'},{'revision':'44e9d71551be13574ac2e10063d03aa9','url':'/favicon.svg'},{'revision':'252b8f8c4d0f64c7cd120f182a6c74ab','url':'/icon-192.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icon-512.png'},{'revision':'252b8f8c4d0f64c7cd120f182a6c74ab','url':'/icons/icon-192.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icons/icon-512.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icons/icon-maskable-512.png'},{'revision':'27007d181514e7fe4213035736a8300e','url':'/manifest.json'},{'revision':'9f61cf51298661d5c57a782534216240','url':'/og-image.png'},{'revision':'7c759c20b35840eacf9344692cb51061','url':'/og-image.svg'},{'revision':'9327802333275f11e68c3f19f20c160d','url':'/widget.js'}],skipWaiting:!0,clientsClaim:!0,navigationPreload:!0,runtimeCaching:eq}).addEventListeners(),self.addEventListener("push",e=>{let t;if(!e.data)return;try{t=e.data.json()}catch{t={title:"VYNDR",body:e.data.text()}}let{title:a="VYNDR",body:s="",icon:r="/icons/icon-192.png",url:n="/"}=t;e.waitUntil(self.registration.showNotification(a,{body:s,icon:r,badge:"/icons/icon-192.png",data:{url:n}}))}),self.addEventListener("notificationclick",e=>{e.notification.close();let t=e.notification.data?.url??"/";e.waitUntil(self.clients.matchAll({type:"window",includeUncontrolled:!0}).then(e=>{let a=e.find(e=>e.url.endsWith(t));return a?a.focus():self.clients.openWindow(t)}))})})(); \ No newline at end of file diff --git a/web/src/app/api/hero-prop/route.ts b/web/src/app/api/hero-prop/route.ts new file mode 100644 index 0000000..ee09a5e --- /dev/null +++ b/web/src/app/api/hero-prop/route.ts @@ -0,0 +1,169 @@ +import { NextResponse } from 'next/server'; + +export const dynamic = 'force-dynamic'; +// Cache the response for 15 minutes (server-side) so cold visitors +// don't trigger a fresh grade on every page load. The cache header +// is what most CDNs / Coolify reverse proxies honor; Next.js itself +// already opts into dynamic rendering via `dynamic = 'force-dynamic'`. +export const revalidate = 900; + +const BACKEND_URL = process.env.BACKEND_URL || 'http://localhost:3000'; +const HERO_FETCH_TIMEOUT_MS = 6000; + +interface OddsResponseProp { + player: string; + stat_type: string; + line: number; + direction?: 'over' | 'under'; + book?: string; + home_team?: string; + away_team?: string; +} + +interface OddsResponse { + sport?: string; + source?: string; + props?: OddsResponseProp[]; + error?: string; +} + +interface GradeResponse { + grade?: string; + confidence?: number; + edge_pct?: number; + projection?: number; + reasoning?: { summary?: string; steps?: unknown }; + kill_conditions_triggered?: Array<{ code: string; reason: string }>; +} + +/** + * Live hero prop endpoint (Session 16). + * + * Picks one fresh, real prop from the day's odds and grades it. The + * landing page hero renders this in place of the static Jokic mockup + * — cold visitors see proof of live intelligence on first paint + * instead of a hypothetical example. + * + * Sport cascade: NBA → WNBA → MLB. Whichever sport produces a non- + * empty `props` list first wins. When every sport is empty (off- + * hours, holiday slate, upstream odds quota burned), responds with + * `{ isStatic: true }` and the client falls back to the existing + * static card. Never throws — odds outages must not blank the + * landing page. + * + * Two-stage flow: + * 1. GET ${BACKEND}/api/odds/{sport} → pick random prop + * 2. POST ${BACKEND}/api/analyze/prop → grade it + * + * Both calls share a 6s timeout (AbortController). The overall + * route is wrapped in try/catch and always 200s (with `isStatic:true` + * on failure) so the client renders gracefully. + */ +async function fetchWithTimeout(url: string, init?: RequestInit, ms = HERO_FETCH_TIMEOUT_MS): Promise { + const controller = new AbortController(); + const timer = setTimeout(() => controller.abort(), ms); + try { + return await fetch(url, { ...init, signal: controller.signal }); + } catch { + return null; + } finally { + clearTimeout(timer); + } +} + +async function pickPropFromSport(sport: string): Promise<{ prop: OddsResponseProp; sport: string } | null> { + const res = await fetchWithTimeout(`${BACKEND_URL}/api/odds/${sport}`, { + method: 'GET', + headers: { Accept: 'application/json' }, + cache: 'no-store', + }); + if (!res || !res.ok) return null; + const body = (await res.json().catch(() => null)) as OddsResponse | null; + if (!body || !Array.isArray(body.props) || body.props.length === 0) return null; + + // Bias toward A-list player names — props with longer player names + // tend to be top-of-rotation stars (better hero material). Cheap + // heuristic, not a hard filter; we still random-pick among the top + // half of the sorted list so multiple page loads vary. + const sorted = body.props + .filter((p) => p.player && p.stat_type && Number.isFinite(p.line)) + .sort((a, b) => (b.player.length - a.player.length)); + if (sorted.length === 0) return null; + const topHalf = sorted.slice(0, Math.max(3, Math.ceil(sorted.length / 2))); + const pick = topHalf[Math.floor(Math.random() * topHalf.length)]; + return { prop: pick, sport }; +} + +async function gradeProp(sport: string, prop: OddsResponseProp): Promise { + const res = await fetchWithTimeout(`${BACKEND_URL}/api/analyze/prop`, { + method: 'POST', + headers: { 'Content-Type': 'application/json', Accept: 'application/json' }, + body: JSON.stringify({ + sport, + player: prop.player, + stat_type: prop.stat_type, + line: prop.line, + direction: prop.direction || 'over', + book: prop.book || 'draftkings', + }), + cache: 'no-store', + }); + if (!res || !res.ok) return null; + return (await res.json().catch(() => null)) as GradeResponse | null; +} + +export async function GET() { + // The order matters: NBA props lead because mid-summer the cascade + // would otherwise constantly land on the same sport. After NBA + // off-season concludes, swap to a season-aware ordering (winter: + // NBA, summer: MLB + WNBA, fall: NFL — when supported). + const sportsToTry = ['nba', 'wnba', 'mlb']; + + try { + for (const sport of sportsToTry) { + const picked = await pickPropFromSport(sport); + if (!picked) continue; + const grade = await gradeProp(picked.sport, picked.prop); + if (!grade) continue; + return NextResponse.json( + { + isStatic: false, + sport: picked.sport, + prop: picked.prop, + grade, + }, + { headers: { 'Cache-Control': 'public, s-maxage=900, stale-while-revalidate=60' } }, + ); + } + } catch { + // Falls through to static fallback below. + } + + // Static fallback — keeps the hero alive when every sport is empty. + return NextResponse.json( + { + isStatic: true, + sport: 'nba', + prop: { + player: 'Nikola Jokic', + stat_type: 'points', + line: 26.5, + direction: 'over', + book: 'draftkings', + home_team: 'DEN', + away_team: 'LAL', + }, + grade: { + grade: 'A-', + confidence: 73, + edge_pct: 6.2, + projection: 29.4, + reasoning: { + summary: 'L5 form is 28.6 over 5 games, +2.1 above the line. Lakers are bottom-five vs Cs.', + }, + kill_conditions_triggered: [], + }, + }, + { headers: { 'Cache-Control': 'public, s-maxage=300, stale-while-revalidate=60' } }, + ); +} diff --git a/web/src/components/Hero.tsx b/web/src/components/Hero.tsx index a41c735..3215803 100644 --- a/web/src/components/Hero.tsx +++ b/web/src/components/Hero.tsx @@ -1,6 +1,10 @@ 'use client'; -import { GradePill } from './GradeCard'; +// Session 16 — the floating demo card on the right side of the hero +// is now driven by /api/hero-prop. Live prop + grade renders with a +// glitch/blur overlay on the reasoning. Falls back to the static +// Jokic layout when no live odds are available. +import LiveHeroProp from './LiveHeroProp'; export default function Hero() { return ( @@ -75,7 +79,7 @@ export default function Hero() {

- +

-

-
- - NBA - -

Nikola Jokic

-

- Over 26.5 points -

-
- -
-
- - -
-
    -
  • - Matchup - LAL · 26th vs C -
  • -
  • - L10 form - 27.4 / 7 of 10 -
  • -
  • - Usage shift - +3.2% w/o Murray -
  • -
- - ); -} - -const row: React.CSSProperties = { - display: 'flex', - justifyContent: 'space-between', - paddingBlock: 4, - borderBottom: '1px solid var(--border)', -}; - -function Stat({ label, value, tone }: { label: string; value: string; tone?: 'positive' }) { - return ( -
-
{label}
-
- {value} -
-
- ); -} +// Session 16 — FloatingDemoCard / Stat / row removed. The hero card +// is now a live, graded prop fetched on mount; see LiveHeroProp.tsx. +// The static Jokic layout lives ONCE inside that component as the +// cold-start fallback when /api/hero-prop returns isStatic:true. +// +// GradePill (re-exported by GradeCard) is still imported at the top +// of this file because the section header uses it elsewhere; if a +// future cleanup confirms no other usage, that import can drop too. diff --git a/web/src/components/LiveHeroProp.tsx b/web/src/components/LiveHeroProp.tsx new file mode 100644 index 0000000..7683a63 --- /dev/null +++ b/web/src/components/LiveHeroProp.tsx @@ -0,0 +1,333 @@ +'use client'; + +import { useEffect, useState } from 'react'; +import { GradePill } from './GradeCard'; + +/** + * Live hero prop card (Session 16). + * + * Replaces the static Jokic mockup. Fetches /api/hero-prop on mount, + * renders the resulting graded prop, applies a glitch/blur overlay + * on the reasoning section so the grade letter + projection + edge + * are crisp (the hook) but the supporting analysis stays behind a + * paywall (the convert). + * + * Two states: + * - Loading or `isStatic === true` from the API → render the + * existing static layout (kept identical for visual stability + * across the cold-start path). + * - Live prop returned → render real data with the glitch overlay. + * + * Glitch overlay: backdrop-filter blur(4px) + a scan-line gradient + * pseudo-element. CSS keyframes in globals.css ensure mobile gets a + * slower, less-CPU-hungry version (the gradient is static there). + */ + +type HeroPropApi = { + isStatic?: boolean; + sport?: string; + prop?: { + player: string; + stat_type: string; + line: number; + direction?: 'over' | 'under'; + book?: string; + home_team?: string; + away_team?: string; + }; + grade?: { + grade?: string; + confidence?: number; + edge_pct?: number; + projection?: number; + reasoning?: { summary?: string }; + kill_conditions_triggered?: Array<{ code: string; reason: string }>; + }; +}; + +const SPORT_LABEL: Record = { + nba: 'NBA', + wnba: 'WNBA', + mlb: 'MLB', + soccer_wc: 'World Cup', +}; + +const SPORT_COLOR: Record = { + nba: '#E94B3C', + wnba: '#FFB347', + mlb: '#1E90FF', + soccer_wc: '#00D4A0', +}; + +const row: React.CSSProperties = { + display: 'flex', + justifyContent: 'space-between', + paddingBlock: 4, + borderBottom: '1px solid var(--border)', +}; + +function Stat({ label, value, tone }: { label: string; value: string; tone?: 'positive' }) { + return ( +
+
{label}
+
+ {value} +
+
+ ); +} + +export default function LiveHeroProp() { + const [data, setData] = useState(null); + const [loaded, setLoaded] = useState(false); + + useEffect(() => { + let alive = true; + fetch('/api/hero-prop', { cache: 'no-store' }) + .then((r) => (r.ok ? r.json() : null)) + .then((json) => { + if (!alive) return; + setData(json); + setLoaded(true); + }) + .catch(() => { + if (!alive) return; + setLoaded(true); + }); + return () => { alive = false; }; + }, []); + + // While loading OR if the API returned the static fallback, render + // the deterministic Jokic layout. Visual continuity matters here — + // cold visitors should see SOMETHING on first paint, then the + // live data slots in once /api/hero-prop returns. + const isLive = loaded && data && !data.isStatic && data.prop && data.grade; + + // Pull display fields with safe fallbacks. + const prop = data?.prop; + const grade = data?.grade; + const sport = data?.sport || 'nba'; + const matchupLabel = prop?.home_team && prop?.away_team + ? `${prop.away_team} @ ${prop.home_team}` + : '—'; + const statTypeLabel = (prop?.stat_type || 'points').replace(/_/g, ' '); + const lineDisplay = prop ? `${(prop.direction || 'over').charAt(0).toUpperCase() + (prop.direction || 'over').slice(1)} ${prop.line} ${statTypeLabel}` : ''; + + // Static-fallback view (the original Jokic card, byte-for-byte + // visually). We render this until the live API returns, then swap. + if (!isLive) { + return ( +
+
+
+ + NBA + +

Nikola Jokic

+

+ Over 26.5 points +

+
+ +
+
+ + +
+
    +
  • MatchupLAL · 26th vs C
  • +
  • L10 form27.4 / 7 of 10
  • +
  • Usage shift+3.2% w/o Murray
  • +
+
+ ); + } + + // Live render. + const gradeText = grade?.grade || 'C'; + const confidence = typeof grade?.confidence === 'number' ? Math.round(grade.confidence) : 50; + const projection = typeof grade?.projection === 'number' ? grade.projection.toFixed(1) : '—'; + const edge = typeof grade?.edge_pct === 'number' ? grade.edge_pct : 0; + const edgeDisplay = `${edge >= 0 ? '+' : ''}${edge.toFixed(1)}%`; + const reasoning = grade?.reasoning?.summary || ''; + const sportLabel = SPORT_LABEL[sport] || sport.toUpperCase(); + const sportColor = SPORT_COLOR[sport] || 'var(--grade-a)'; + + return ( +
+ {/* LIVE badge — pulsing dot communicates "this was graded just now". */} +
+ + LIVE +
+ + {/* Header — visible, the hook. */} +
+
+ + {sportLabel} + +

{prop!.player}

+

+ {lineDisplay} +

+
+ +
+ + {/* Projection + edge — visible, the proof. */} +
+ + 0 ? 'positive' : undefined} /> +
+ + {/* Reasoning — BLURRED, the paywall. */} +
+
+ {reasoning || 'Recent form: 28.4 over last 5. Opp defense: top-5 vs PG. Pace: +3.1. Trap composite 0.18. Usage 31%. Kill conditions: 0.'} +
+ {/* Scan-line overlay — pure CSS gradient pseudo-element via + inline style + position absolute. Subtle on desktop, + disabled on mobile by the @media query below. */} +
+
+
+ Classified · Sign up to unlock +
+
+ + {/* CTA — drives the conversion the blur sets up. */} + + Sign up to read the full analysis → + +
+ ); +}