From 927c4a5c6557fe63523aa0acd1bc8ea321e0aac7 Mon Sep 17 00:00:00 2001 From: Kev Date: Sat, 13 Jun 2026 21:30:57 -0400 Subject: [PATCH] =?UTF-8?q?Session=2029:=20Content=20generation=20template?= =?UTF-8?q?s=20=E2=80=94=20slate=20threads,=20POTD,=20recaps,=20matchup=20?= =?UTF-8?q?previews=20(1660=20tests)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- BUILD-STATE.md | 56 +++ data/training/resolutions-2026-06.jsonl | 14 + src/app.js | 4 + src/routes/content.js | 102 +++++ src/services/contentFormatter.js | 105 +++++ src/services/contentTemplateService.js | 421 +++++++++++++++++++++ tests/integration/contentRoutes.test.js | 110 ++++++ tests/unit/contentFormatter.test.js | 59 +++ tests/unit/contentTemplateService.test.js | 166 ++++++++ web/public/sw.js | 2 +- web/src/app/api/content/[...path]/route.ts | 25 ++ 11 files changed, 1063 insertions(+), 1 deletion(-) create mode 100644 src/routes/content.js create mode 100644 src/services/contentFormatter.js create mode 100644 src/services/contentTemplateService.js create mode 100644 tests/integration/contentRoutes.test.js create mode 100644 tests/unit/contentFormatter.test.js create mode 100644 tests/unit/contentTemplateService.test.js create mode 100644 web/src/app/api/content/[...path]/route.ts diff --git a/BUILD-STATE.md b/BUILD-STATE.md index 5212d7b..feb2d15 100755 --- a/BUILD-STATE.md +++ b/BUILD-STATE.md @@ -4,6 +4,62 @@ 2026-06-12 ## Current Phase +SHIP BUILD v29.0 — Content generation templates: structured social/newsletter content from live data (Session 29) + +## Session 29 (2026-06-13) — SHIPPED + +The data engine that produces raw material for daily social content. Each +template consumes live VYNDR data and returns STRUCTURED OBJECTS (not text, +not images) that degrade gracefully by data level. A formatter renders +plain text; the image/design layer comes later. + +Backend 1623 → **1660 tests** (+37), 133 suites, zero regressions. Web +build clean. + +### PHASE 1-3 — Template engine + slate thread + POTD +- `contentTemplateService.js`: + - `collectSlateData(sport, deps?)` — gathers schedule + game lines + + grades + streaks + movers + best lines via Promise.allSettled, + INJECTABLE collectors (default wires the real services). Sets + `dataLevel`: full / lines / schedule / empty. + - `generateSlateThread` — hook + content posts + CTA. Full → top-5 + graded picks; lines → game-line highlights (best ML, consensus + total/spread, book disagreement) + movers; schedule → game list. + - `generatePOTD` — best grade (full) or game-of-the-day (lines) or + `{ available: false }`. + - Field-alias normalizers so grades from any shape (player/player_name, + side/direction, edge/edge_pct) work. + +### PHASE 4-5 — Recap + matchup preview +- `generateResultsRecap(sport, resolvedGrades)` — record, win rate, top + hits, biggest miss, by-tier (A/B/C), Brier score + avg CLV. Pure. +- `generateMatchupPreview(game, gameLines, streaks)` — teams, lines + summary (consensus spread/total, home-favorite), streaks matched to the + two teams, one-line narrative. Degrades to `lines: null`. + +### PHASE 6 — Content API +- `GET /api/content/{slate,potd,recap,preview}/:sport` (preview takes + `/:gameId`). `?format=text` adds post-ready strings. Mounted in app.js; + Next proxy `api/content/[...path]/route.ts`. + +### PHASE 7 — Formatter +- `contentFormatter.js` — slate thread → array of plain-text posts (one + per role), POTD + recap text blocks. Defensive: never emits "undefined". + +### Files created +- `src/services/contentTemplateService.js`, `src/services/contentFormatter.js` +- `src/routes/content.js` +- `web/src/app/api/content/[...path]/route.ts` +- `tests/unit/contentTemplateService.test.js` (22), + `tests/unit/contentFormatter.test.js` (7), + `tests/integration/contentRoutes.test.js` (8) + +### Files modified +- `src/app.js` (mount /api/content) + +--- + +## Previous Phase SHIP BUILD v28.0 — Parlay builder, line-movement tracking, book comparison (Session 28) ## Session 28 (2026-06-13) — SHIPPED diff --git a/data/training/resolutions-2026-06.jsonl b/data/training/resolutions-2026-06.jsonl index 07a5be9..d4b144d 100644 --- a/data/training/resolutions-2026-06.jsonl +++ b/data/training/resolutions-2026-06.jsonl @@ -780,3 +780,17 @@ {"ts":"2026-06-13T15:56:53.881Z","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-13T15:56:53.881Z","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-13T15:56:53.940Z","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-13T20:27:36.907Z","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-13T20:27:36.973Z","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-13T20:27:37.223Z","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-13T20:27:38.042Z","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-13T20:27:38.042Z","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-13T20:27:38.042Z","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-13T20:27:38.259Z","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-13T20:44:17.269Z","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-13T20:44:18.291Z","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-13T20:44:18.386Z","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-13T20:44:19.184Z","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-13T20:44:19.184Z","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-13T20:44:19.184Z","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-13T20:44:19.233Z","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/app.js b/src/app.js index e9613ff..e3238a7 100644 --- a/src/app.js +++ b/src/app.js @@ -159,6 +159,10 @@ const lineMovementRoutes = require('./routes/lineMovement'); app.use('/api/lines', lineMovementRoutes); const bookComparisonRoutes = require('./routes/bookComparison'); app.use('/api/books', bookComparisonRoutes); +// Session 29 — content templates: structured social/newsletter content +// generated from live data, degrading gracefully by data level. +const contentRoutes = require('./routes/content'); +app.use('/api/content', contentRoutes); // Session 18 — internal ops endpoints (admin dashboard triggers, // shared-key auth via `VYNDR_INTERNAL_KEY`). Never reachable from // the public surface; the Next.js admin route proxies through with diff --git a/src/routes/content.js b/src/routes/content.js new file mode 100644 index 0000000..b9cb0fd --- /dev/null +++ b/src/routes/content.js @@ -0,0 +1,102 @@ +'use strict'; + +/** + * /api/content (Session 29) + * + * Structured content objects generated from live VYNDR data, degrading + * gracefully by data level. Read-only, zero-credit (reads cached data). + * `?format=text` additionally returns post-ready plain text. + * + * GET /api/content/slate/:sport → daily slate thread + * GET /api/content/potd/:sport → prop (or game) of the day + * GET /api/content/recap/:sport → results recap (needs resolved grades) + * GET /api/content/preview/:sport/:gameId → matchup preview for one game + */ + +const express = require('express'); +const template = require('../services/contentTemplateService'); +const formatter = require('../services/contentFormatter'); +const { cacheGet } = require('../utils/redis'); + +const router = express.Router(); + +const MISSION_HEADER = { 'X-VYNDR-Mission': 'Every post is a free ad' }; +const SUPPORTED = new Set(['nba', 'wnba', 'mlb', 'soccer', 'nfl', 'nhl']); + +function guard(req, res) { + const sport = String(req.params.sport || '').toLowerCase(); + if (!SUPPORTED.has(sport)) { + res.status(404).set(MISSION_HEADER).json({ error: `No content for sport: ${sport}` }); + return null; + } + return sport; +} + +router.get('/slate/:sport', async (req, res) => { + const sport = guard(req, res); + if (!sport) return undefined; + try { + const data = await template.collectSlateData(sport); + const thread = template.generateSlateThread(sport, data); + const body = { ...thread }; + if (req.query.format === 'text') body.text = formatter.formatSlateThread(thread); + return res.set(MISSION_HEADER).json(body); + } catch (err) { + console.error(`[content/slate/${sport}]`, err.message); + return res.status(500).set(MISSION_HEADER).json({ error: 'Content generation failed' }); + } +}); + +router.get('/potd/:sport', async (req, res) => { + const sport = guard(req, res); + if (!sport) return undefined; + try { + const data = await template.collectSlateData(sport); + const potd = template.generatePOTD(sport, data); + const body = { ...potd }; + if (req.query.format === 'text') body.text = formatter.formatPOTD(potd); + return res.set(MISSION_HEADER).json(body); + } catch (err) { + console.error(`[content/potd/${sport}]`, err.message); + return res.status(500).set(MISSION_HEADER).json({ error: 'Content generation failed' }); + } +}); + +router.get('/recap/:sport', async (req, res) => { + const sport = guard(req, res); + if (!sport) return undefined; + try { + // Resolved grades land in a daily cache when the grading flow settles. + const utc = new Date().toISOString().split('T')[0]; + const cache = (await cacheGet(`results:${sport}:${utc}`)) ?? (await cacheGet(`results:${sport}`)); + const resolved = Array.isArray(cache) ? cache : Array.isArray(cache?.grades) ? cache.grades : []; + const recap = template.generateResultsRecap(sport, resolved); + const body = { ...recap }; + if (req.query.format === 'text') body.text = formatter.formatRecap(recap); + return res.set(MISSION_HEADER).json(body); + } catch (err) { + console.error(`[content/recap/${sport}]`, err.message); + return res.status(500).set(MISSION_HEADER).json({ error: 'Content generation failed' }); + } +}); + +router.get('/preview/:sport/:gameId', async (req, res) => { + const sport = guard(req, res); + if (!sport) return undefined; + const { gameId } = req.params; + try { + const data = await template.collectSlateData(sport); + const game = (data.schedule || []).find((g) => String(g.id) === String(gameId)); + if (!game) { + return res.status(404).set(MISSION_HEADER).json({ error: 'Game not on today\'s schedule.' }); + } + const lines = template.__internals.findGameLinesFor(game, data.gameLines); + const preview = template.generateMatchupPreview(game, lines, data.streaks); + return res.set(MISSION_HEADER).json(preview); + } catch (err) { + console.error(`[content/preview/${sport}]`, err.message); + return res.status(500).set(MISSION_HEADER).json({ error: 'Content generation failed' }); + } +}); + +module.exports = router; diff --git a/src/services/contentFormatter.js b/src/services/contentFormatter.js new file mode 100644 index 0000000..d957c0b --- /dev/null +++ b/src/services/contentFormatter.js @@ -0,0 +1,105 @@ +'use strict'; + +/** + * Content formatter (Session 29). + * + * Transforms structured content objects (from contentTemplateService) into + * platform-ready PLAIN TEXT — suitable for X threads / Telegram. Image + * formatting is a separate design-layer concern. Pure + defensive: a + * missing field renders as a sensible blank, never "undefined". + */ + +function fmtOdds(o) { + const n = Number(o); + if (!Number.isFinite(n)) return String(o ?? ''); + return n > 0 ? `+${n}` : `${n}`; +} + +function fmtTime(iso) { + if (!iso) return ''; + try { + return new Date(iso).toLocaleTimeString('en-US', { hour: 'numeric', minute: '2-digit', timeZoneName: 'short' }); + } catch { + return String(iso); + } +} + +function formatPickPost(p) { + const lines = [`${p.player} · ${p.stat} ${p.side || ''} ${p.line ?? ''}`.replace(/\s+/g, ' ').trim()]; + lines.push(`Grade: ${p.grade}${p.confidence ? ` · ${p.confidence}% confidence` : ''}`); + if (p.edge) lines.push(`Edge: ${p.edge > 0 ? '+' : ''}${p.edge}%`); + if (p.analysis) lines.push(p.analysis); + return lines.join('\n'); +} + +function formatGameHighlight(p) { + const parts = [`${p.game}${p.time ? ` · ${fmtTime(p.time)}` : ''}`]; + if (p.bestAwayML) parts.push(`Best away ML: ${fmtOdds(p.bestAwayML.odds)} (${p.bestAwayML.book})`); + if (p.bestHomeML) parts.push(`Best home ML: ${fmtOdds(p.bestHomeML.odds)} (${p.bestHomeML.book})`); + if (p.total != null) parts.push(`O/U ${p.total}`); + if (p.bookCount) parts.push(`${p.bookCount} books`); + return parts.join('\n'); +} + +function formatMovers(p) { + const head = '📈 Biggest line moves:'; + const rows = (p.movers || []).map((m) => { + const d = m.delta > 0 ? `+${m.delta}` : `${m.delta}`; + return `${m.player} ${String(m.stat || '').replace(/_/g, ' ')}: ${m.opening ?? '?'} → ${m.current ?? '?'} (${d})${m.sharpSignal ? ' ⚡' : ''}`; + }); + return [head, ...rows].join('\n'); +} + +function formatSchedule(p) { + return (p.games || []) + .map((g) => `${g.away || '?'} @ ${g.home || '?'}${g.time ? ` · ${fmtTime(g.time)}` : ''}`) + .join('\n'); +} + +function formatPost(post) { + switch (post.role) { + case 'hook': return post.text || ''; + case 'cta': return post.text || ''; + case 'pick': return formatPickPost(post); + case 'game_highlight': return formatGameHighlight(post); + case 'movers': return formatMovers(post); + case 'schedule': return formatSchedule(post); + default: return ''; + } +} + +/** Returns an array of post-ready strings (one per thread post). */ +function formatSlateThread(thread) { + if (!thread || !Array.isArray(thread.posts)) return []; + return thread.posts.map(formatPost); +} + +/** Single-block POTD text. */ +function formatPOTD(potd) { + if (!potd || potd.available === false) return 'No standout pick today — check the full slate at vyndr.app.'; + if (potd.dataLevel === 'lines') { + return `🎯 GAME OF THE DAY\n${potd.game}${potd.time ? ` · ${fmtTime(potd.time)}` : ''}\n${potd.total != null ? `O/U ${potd.total}` : ''}${potd.bestHomeML ? `\nBest home ML ${fmtOdds(potd.bestHomeML.odds)} (${potd.bestHomeML.book})` : ''}`.trim(); + } + return `🎯 PROP OF THE DAY\n${formatPickPost(potd)}`; +} + +/** Recap text block. */ +function formatRecap(recap) { + if (!recap || recap.available === false) return 'No graded results in yet today.'; + const { record, winRate } = recap; + const lines = [`📊 VYNDR RECAP — ${recap.date}`]; + lines.push(`Record: ${record.wins}-${record.losses}${record.pushes ? `-${record.pushes}` : ''}${winRate != null ? ` (${Math.round(winRate * 100)}%)` : ''}`); + if (recap.topHits && recap.topHits.length) { + lines.push('Top hits:'); + for (const h of recap.topHits) lines.push(`✅ ${h.player} ${h.stat} ${h.side || ''} ${h.line ?? ''}`.replace(/\s+/g, ' ').trim()); + } + if (recap.metrics && recap.metrics.brierScore != null) lines.push(`Brier: ${recap.metrics.brierScore}`); + return lines.join('\n'); +} + +module.exports = { + formatSlateThread, + formatPOTD, + formatRecap, + __internals: { formatPost, formatPickPost, formatGameHighlight, formatMovers, formatSchedule }, +}; diff --git a/src/services/contentTemplateService.js b/src/services/contentTemplateService.js new file mode 100644 index 0000000..3d8abc2 --- /dev/null +++ b/src/services/contentTemplateService.js @@ -0,0 +1,421 @@ +'use strict'; + +/** + * Content template engine (Session 29). + * + * Generates STRUCTURED DATA objects (not text, not images) from live VYNDR + * data. A separate formatter (contentFormatter) renders text; a future + * design layer renders images. + * + * Templates degrade gracefully by data level: + * full → graded props exist → rich content (grade, edge, projection) + * lines → Tank01 game lines exist → line-focused content + * schedule → only ESPN schedule → matchup/time content + * empty → nothing today → minimal, never a crash + * + * The pure generators take a `data` bundle and never touch the network. + * `collectSlateData` is the thin orchestrator that gathers the bundle from + * the live services; its collectors are injectable so it stays testable. + */ + +const GRADE_RANK = { 'A+': 0, A: 1, 'A-': 2, 'B+': 3, B: 4, 'B-': 5, 'C+': 6, C: 7, 'C-': 8, 'D+': 9, D: 10, 'D-': 11, F: 12 }; +const SITE = 'vyndr.app'; + +// ---- field-alias normalizers (grades come from several shapes) -------- +function gradeField(g, ...keys) { + for (const k of keys) if (g && g[k] != null && g[k] !== '') return g[k]; + return null; +} +function normalizeGrade(g) { + return { + player: gradeField(g, 'player_name', 'player'), + stat: gradeField(g, 'stat_type', 'stat'), + line: gradeField(g, 'line'), + side: gradeField(g, 'side', 'direction'), + grade: gradeField(g, 'grade') || 'C', + confidence: Number(gradeField(g, 'confidence')) || 0, + edge: Number(gradeField(g, 'edge', 'edge_pct')) || 0, + projection: gradeField(g, 'projection'), + matchup: gradeField(g, 'matchup'), + killConditions: g?.killConditions || g?.kill_conditions_triggered || [], + analysis: gradeField(g, 'analysis', 'one_line_reason') || (g?.reasoning && g.reasoning.summary) || null, + _raw: g, + }; +} + +// ---- data level ------------------------------------------------------- +function determineDataLevel(data) { + const grades = data?.grades || []; + const gameLines = data?.gameLines || {}; + const schedule = data?.schedule || []; + if (grades.length > 0) return 'full'; + if (gameLines && Object.keys(gameLines).length > 0) return 'lines'; + if (schedule.length > 0) return 'schedule'; + return 'empty'; +} + +function prettyDate(now = Date.now()) { + return new Date(now).toLocaleDateString('en-US', { weekday: 'long', month: 'short', day: 'numeric' }); +} + +function sportLabel(sport) { + return String(sport || '').toUpperCase(); +} + +// ---- game-line highlight extraction ----------------------------------- +function numOdds(v) { + const n = Number(v); + return Number.isFinite(n) ? n : null; +} +function americanToDecimal(odds) { + const n = Number(odds); + if (!Number.isFinite(n) || n === 0) return null; + return n > 0 ? 1 + n / 100 : 1 + 100 / Math.abs(n); +} + +// books: array of [bookName, { homeML, awayML, total, homeSpread, ... }] +function bestOdds(books, field) { + let best = null; + for (const [book, line] of books) { + const dec = americanToDecimal(line?.[field]); + if (dec == null) continue; + if (!best || dec > best.decimal) best = { book, odds: line[field], decimal: dec }; + } + return best; +} +function consensus(books, field) { + const vals = books.map(([, l]) => Number(l?.[field])).filter((n) => Number.isFinite(n)); + if (vals.length === 0) return null; + return Math.round((vals.reduce((a, b) => a + b, 0) / vals.length) * 10) / 10; +} +// Disagreement = spread between best & worst home-ML decimal payout. +function bookDisagreement(books) { + const decs = books.map(([, l]) => americanToDecimal(l?.homeML)).filter((n) => n != null); + if (decs.length < 2) return 0; + return Math.round((Math.max(...decs) - Math.min(...decs)) * 100) / 100; +} + +function findGameLinesFor(game, gameLines) { + if (!gameLines) return null; + const h = (game.homeTeam?.abbreviation || '').toUpperCase(); + const a = (game.awayTeam?.abbreviation || '').toUpperCase(); + for (const entry of Object.values(gameLines)) { + const eh = String(entry.homeTeam || '').toUpperCase(); + const ea = String(entry.awayTeam || '').toUpperCase(); + if ((eh === h && ea === a) || (eh === a && ea === h)) return entry; + } + return null; +} + +function extractGameLineHighlights(schedule, gameLines) { + const highlights = []; + for (const game of schedule || []) { + const lines = findGameLinesFor(game, gameLines); + const books = lines && lines.books ? Object.entries(lines.books) : []; + if (books.length === 0) continue; + highlights.push({ + type: 'game_line_highlight', + game: `${game.awayTeam?.abbreviation || '?'} @ ${game.homeTeam?.abbreviation || '?'}`, + time: game.gameTime || null, + venue: game.venue || null, + bookCount: books.length, + bestHomeML: bestOdds(books, 'homeML'), + bestAwayML: bestOdds(books, 'awayML'), + total: consensus(books, 'total'), + spread: consensus(books, 'homeSpread'), + disagreement: bookDisagreement(books), + }); + } + return highlights.sort((a, b) => (b.disagreement || 0) - (a.disagreement || 0)); +} + +// ---- hook + CTA ------------------------------------------------------- +function generateHookText(sport, data) { + const label = sportLabel(sport); + const date = prettyDate(); + const games = (data.schedule || []).length; + const bookCount = maxBookCount(data.gameLines); + if (data.dataLevel === 'full') { + return `🔥 VYNDR SLATE — ${date}. ${games} ${label} games graded${bookCount ? ` · lines from ${bookCount} books` : ''}. Top reads below. 🧵`; + } + if (data.dataLevel === 'lines') { + return `🔥 VYNDR SLATE — ${date}. ${games} ${label} games · lines from ${bookCount} sportsbooks. Here's what stands out. 🧵`; + } + if (data.dataLevel === 'schedule') { + return `🔥 VYNDR SLATE — ${date}. ${games} ${label} games on the board today. 🧵`; + } + return `VYNDR — ${date}. No ${label} games today. Back tomorrow.`; +} +function maxBookCount(gameLines) { + if (!gameLines) return 0; + let max = 0; + for (const entry of Object.values(gameLines)) { + const n = entry && entry.books ? Object.keys(entry.books).length : 0; + if (n > max) max = n; + } + return max; +} +function generateCTAText(dataLevel) { + if (dataLevel === 'empty') return `Full slate + intelligence at ${SITE}.`; + return `Full slate live at ${SITE}. Free tier: reads on us. Founder pricing from $14.99/mo.`; +} +function generateAnalysisLine(grade) { + if (grade.analysis) return grade.analysis; + const edge = grade.edge ? `${grade.edge > 0 ? '+' : ''}${grade.edge}% edge` : null; + const proj = grade.projection != null ? `projection ${grade.projection}` : null; + return [proj, edge].filter(Boolean).join(' · ') || 'Model edge on this line.'; +} + +// ---- slate thread ----------------------------------------------------- +function generateSlateThread(sport, data) { + const dataLevel = data.dataLevel || determineDataLevel(data); + const enriched = { ...data, dataLevel }; + const thread = { type: 'slate_thread', sport, date: prettyDate(), dataLevel, posts: [] }; + + thread.posts.push({ role: 'hook', text: generateHookText(sport, enriched) }); + + if (dataLevel === 'full') { + const top = (data.grades || []).map(normalizeGrade) + .sort((a, b) => b.confidence - a.confidence) + .slice(0, 5); + for (const g of top) { + thread.posts.push({ + role: 'pick', + player: g.player, stat: g.stat, line: g.line, side: g.side, + grade: g.grade, confidence: g.confidence, edge: g.edge, projection: g.projection, + analysis: generateAnalysisLine(g), cardData: g._raw, + }); + } + } else if (dataLevel === 'lines') { + const highlights = extractGameLineHighlights(data.schedule, data.gameLines); + for (const h of highlights.slice(0, 4)) thread.posts.push({ role: 'game_highlight', ...h }); + if ((data.movers || []).length > 0) { + thread.posts.push({ role: 'movers', movers: data.movers.slice(0, 3) }); + } + } else if (dataLevel === 'schedule') { + const games = (data.schedule || []).slice(0, 8).map((g) => ({ + away: g.awayTeam?.name || g.awayTeam?.abbreviation || null, + home: g.homeTeam?.name || g.homeTeam?.abbreviation || null, + time: g.gameTime || null, + venue: g.venue || null, + })); + thread.posts.push({ role: 'schedule', games }); + } + + thread.posts.push({ role: 'cta', text: generateCTAText(dataLevel) }); + return thread; +} + +// ---- prop / game of the day ------------------------------------------- +function generatePOTD(sport, data) { + const dataLevel = data.dataLevel || determineDataLevel(data); + if (dataLevel === 'full' && (data.grades || []).length > 0) { + const best = (data.grades || []).map(normalizeGrade).reduce((a, b) => (b.confidence > a.confidence ? b : a)); + return { + type: 'potd', dataLevel: 'full', + player: best.player, stat: best.stat, line: best.line, side: best.side, + grade: best.grade, confidence: best.confidence, edge: best.edge, + projection: best.projection, matchup: best.matchup, + killConditions: best.killConditions, + analysis: generateAnalysisLine(best), cardData: best._raw, + }; + } + if (dataLevel === 'lines') { + const top = extractGameLineHighlights(data.schedule, data.gameLines)[0]; + if (top) return { type: 'potd', dataLevel: 'lines', subtype: 'game_of_the_day', ...top }; + } + return { type: 'potd', dataLevel: 'empty', available: false }; +} + +// ---- results recap (pure, takes resolved grades) ---------------------- +function gradeTier(grade) { + return String(grade || '').charAt(0).toUpperCase(); +} +function tierRecord(grades, tier) { + const inTier = grades.filter((g) => gradeTier(gradeField(g, 'grade')) === tier); + const wins = inTier.filter((g) => g.result === 'win').length; + const losses = inTier.filter((g) => g.result === 'loss').length; + return { wins, losses, total: inTier.length }; +} +function brierScore(grades) { + // Brier = mean( (p - outcome)^2 ); p from confidence%, outcome 1 win / 0 loss. + const scored = grades.filter((g) => g.result === 'win' || g.result === 'loss'); + if (scored.length === 0) return null; + const sum = scored.reduce((acc, g) => { + const p = (Number(gradeField(g, 'confidence')) || 50) / 100; + const outcome = g.result === 'win' ? 1 : 0; + return acc + (p - outcome) ** 2; + }, 0); + return Math.round((sum / scored.length) * 1000) / 1000; +} +function avgCLV(grades) { + const vals = grades.map((g) => Number(gradeField(g, 'clv'))).filter((n) => Number.isFinite(n)); + if (vals.length === 0) return null; + return Math.round((vals.reduce((a, b) => a + b, 0) / vals.length) * 100) / 100; +} + +function generateResultsRecap(sport, resolvedGrades) { + if (!Array.isArray(resolvedGrades) || resolvedGrades.length === 0) { + return { type: 'recap', available: false }; + } + const wins = resolvedGrades.filter((g) => g.result === 'win'); + const losses = resolvedGrades.filter((g) => g.result === 'loss'); + const pushes = resolvedGrades.filter((g) => g.result === 'push'); + const decided = wins.length + losses.length; + + const byConfidence = (arr) => [...arr].sort((a, b) => (Number(gradeField(b, 'confidence')) || 0) - (Number(gradeField(a, 'confidence')) || 0)); + + return { + type: 'recap', + available: true, + sport, + date: prettyDate(), + record: { wins: wins.length, losses: losses.length, pushes: pushes.length }, + winRate: decided > 0 ? Math.round((wins.length / decided) * 1000) / 1000 : null, + topHits: byConfidence(wins).slice(0, 3).map((g) => ({ + player: gradeField(g, 'player_name', 'player'), + stat: gradeField(g, 'stat_type', 'stat'), + line: gradeField(g, 'line'), + side: gradeField(g, 'side', 'direction'), + grade: gradeField(g, 'grade'), + actual: gradeField(g, 'actual'), + })), + biggestMiss: losses.length > 0 ? (() => { + const m = byConfidence(losses)[0]; + return { + player: gradeField(m, 'player_name', 'player'), + stat: gradeField(m, 'stat_type', 'stat'), + line: gradeField(m, 'line'), + side: gradeField(m, 'side', 'direction'), + grade: gradeField(m, 'grade'), + actual: gradeField(m, 'actual'), + }; + })() : null, + byTier: { A: tierRecord(resolvedGrades, 'A'), B: tierRecord(resolvedGrades, 'B'), C: tierRecord(resolvedGrades, 'C') }, + metrics: { brierScore: brierScore(resolvedGrades), clv: avgCLV(resolvedGrades) }, + }; +} + +// ---- matchup preview -------------------------------------------------- +function generateMatchupNarrative(game, lines, streaks) { + const away = game.awayTeam?.name || game.awayTeam?.abbreviation || 'Away'; + const home = game.homeTeam?.name || game.homeTeam?.abbreviation || 'Home'; + const matched = (streaks || []).length; + if (lines && lines.total != null) { + return `${away} at ${home} — consensus total ${lines.total}${matched ? `, ${matched} active player streak${matched === 1 ? '' : 's'} in play` : ''}.`; + } + return `${away} at ${home}${matched ? ` — ${matched} player${matched === 1 ? '' : 's'} riding a streak` : ''}.`; +} + +function generateMatchupPreview(game, gameLines, streaks) { + const books = gameLines && gameLines.books ? Object.entries(gameLines.books) : []; + const lines = books.length > 0 ? { + spread: consensus(books, 'homeSpread'), + total: consensus(books, 'total'), + homeFavorite: (() => { + const hs = consensus(books, 'homeSpread'); + return hs != null ? hs < 0 : null; + })(), + bookCount: books.length, + } : null; + + const h = (game.homeTeam?.abbreviation || '').toUpperCase(); + const a = (game.awayTeam?.abbreviation || '').toUpperCase(); + const playerStreaks = (streaks || []).filter((s) => { + const t = (s.team || '').toUpperCase(); + return t && (t === h || t === a); + }); + + return { + type: 'matchup_preview', + away: game.awayTeam || null, + home: game.homeTeam || null, + time: game.gameTime || null, + venue: game.venue || null, + lines, + playerStreaks, + narrative: generateMatchupNarrative(game, lines, playerStreaks), + }; +} + +// ---- orchestrator (injectable collectors for testability) ------------- +function defaultCollectors() { + const scheduleService = require('./scheduleService'); + const lineSnapshotService = require('./lineSnapshotService'); + const bookComparisonService = require('./bookComparisonService'); + const streaksService = require('./streaksService'); + const { loadRosterLogs } = require('./rosterLogs'); + const { cacheGet } = require('../utils/redis'); + const nbaAdapter = require('./adapters/tank01NbaAdapter'); + const mlbAdapter = require('./adapters/tank01MlbAdapter'); + const { __internals: glInternals } = require('../routes/gameLines'); + + return { + getSchedule: (sport, date) => scheduleService.getSchedule(sport, date), + getGameLines: async (sport, date) => { + const fetcher = sport === 'nba' ? nbaAdapter.getNBABettingOdds : sport === 'mlb' ? mlbAdapter.getMLBBettingOdds : null; + if (!fetcher) return {}; + const body = await fetcher(date).catch(() => null); + return body ? glInternals.normalizeGameLines(body) : {}; + }, + getGrades: async (sport) => { + // odds-api grades cache (when present); degrade to [] otherwise. + const utc = new Date().toISOString().split('T')[0]; + const cache = (await cacheGet(`grades:${sport}:${utc}`)) ?? (await cacheGet(`grades:${sport}`)); + return Array.isArray(cache) ? cache : Array.isArray(cache?.grades) ? cache.grades : []; + }, + getStreaks: async (sport) => { + const roster = await loadRosterLogs(sport); + return streaksService.computeStreaks(roster, sport, {}); + }, + getMovers: (sport) => lineSnapshotService.getBiggestMovers(sport, { limit: 10 }), + getBestLines: async (sport) => { + const utc = new Date().toISOString().split('T')[0]; + const cache = (await cacheGet(`odds:${sport}:${utc}`)) ?? (await cacheGet(`odds:${sport}`)); + const props = Array.isArray(cache?.props) ? cache.props : []; + return bookComparisonService.bestLines(props, { limit: 10 }); + }, + }; +} + +async function collectSlateData(sport, deps) { + const c = deps || defaultCollectors(); + const date = (() => { + try { return require('./scheduleService').todayET(); } catch { return undefined; } + })(); + + const [schedule, gameLines, grades, streaks, movers, bestLines] = await Promise.allSettled([ + c.getSchedule(sport, date), + c.getGameLines(sport, date), + c.getGrades(sport), + c.getStreaks(sport), + c.getMovers(sport), + c.getBestLines(sport), + ]); + + const val = (r, fallback) => (r.status === 'fulfilled' && r.value != null ? r.value : fallback); + const data = { + sport, + schedule: val(schedule, []), + gameLines: val(gameLines, {}), + grades: val(grades, []), + streaks: val(streaks, []), + movers: val(movers, []), + bestLines: val(bestLines, []), + }; + data.dataLevel = determineDataLevel(data); + return data; +} + +module.exports = { + collectSlateData, + determineDataLevel, + generateSlateThread, + generatePOTD, + generateResultsRecap, + generateMatchupPreview, + __internals: { + normalizeGrade, extractGameLineHighlights, generateHookText, generateCTAText, + brierScore, avgCLV, tierRecord, bestOdds, consensus, bookDisagreement, findGameLinesFor, + }, +}; diff --git a/tests/integration/contentRoutes.test.js b/tests/integration/contentRoutes.test.js new file mode 100644 index 0000000..959efbc --- /dev/null +++ b/tests/integration/contentRoutes.test.js @@ -0,0 +1,110 @@ +// Integration: /api/content routes (Session 29). +// The template service is mocked at the collector boundary via redis + +// real generators; recap reads a mocked results cache. + +const express = require('express'); +const request = require('supertest'); + +const mockStore = {}; +jest.mock('../../src/utils/redis', () => ({ + cacheGet: jest.fn(async (k) => (k in mockStore ? mockStore[k] : null)), + getRedisClient: () => ({ scan: async () => ['0', []], lrange: async () => [] }), + isDegraded: () => false, +})); + +// Mock collectSlateData so the route doesn't hit live adapters; keep the +// real generators so we exercise the actual content shapes. +jest.mock('../../src/services/contentTemplateService', () => { + const actual = jest.requireActual('../../src/services/contentTemplateService'); + return { ...actual, collectSlateData: jest.fn() }; +}); +const template = require('../../src/services/contentTemplateService'); + +function mountApp() { + delete require.cache[require.resolve('../../src/routes/content')]; + const app = express(); + app.use(express.json()); + app.use('/api/content', require('../../src/routes/content')); + return app; +} + +const SLATE = { + sport: 'mlb', + schedule: [{ id: 'g1', homeTeam: { name: 'Reds', abbreviation: 'CIN' }, awayTeam: { name: 'D-backs', abbreviation: 'ARI' }, gameTime: '2026-06-13T23:10:00Z', venue: 'GABP' }], + gameLines: { '20260613_ARI@CIN': { homeTeam: 'CIN', awayTeam: 'ARI', books: { bet365: { homeML: '-120', awayML: '+100', total: '9.5', homeSpread: '-1.5' } } } }, + grades: [], + streaks: [{ player: 'Acuna', team: 'ARI', description: '3-game hit streak', currentStreak: 3 }], + movers: [], + bestLines: [], + dataLevel: 'lines', +}; + +beforeEach(() => { + for (const k of Object.keys(mockStore)) delete mockStore[k]; + jest.clearAllMocks(); + template.collectSlateData.mockResolvedValue(SLATE); +}); + +describe('GET /api/content/slate/:sport', () => { + test('returns a slate thread structure', async () => { + const res = await request(mountApp()).get('/api/content/slate/mlb'); + expect(res.status).toBe(200); + expect(res.body.type).toBe('slate_thread'); + expect(res.body.dataLevel).toBe('lines'); + expect(res.body.posts[0].role).toBe('hook'); + }); + + test('?format=text adds formatted post strings', async () => { + const res = await request(mountApp()).get('/api/content/slate/mlb?format=text'); + expect(Array.isArray(res.body.text)).toBe(true); + expect(res.body.text[0]).toMatch(/VYNDR SLATE/); + }); + + test('unsupported sport → 404', async () => { + const res = await request(mountApp()).get('/api/content/slate/cricket'); + expect(res.status).toBe(404); + }); +}); + +describe('GET /api/content/potd/:sport', () => { + test('lines data → game of the day', async () => { + const res = await request(mountApp()).get('/api/content/potd/mlb'); + expect(res.status).toBe(200); + expect(res.body.subtype).toBe('game_of_the_day'); + }); +}); + +describe('GET /api/content/recap/:sport', () => { + test('no resolved grades → available:false', async () => { + const res = await request(mountApp()).get('/api/content/recap/mlb'); + expect(res.status).toBe(200); + expect(res.body.available).toBe(false); + }); + + test('with cached results → record computed', async () => { + const utc = new Date().toISOString().split('T')[0]; + mockStore[`results:mlb:${utc}`] = [ + { player: 'A', grade: 'A', confidence: 80, result: 'win' }, + { player: 'B', grade: 'B', confidence: 60, result: 'loss' }, + ]; + const res = await request(mountApp()).get('/api/content/recap/mlb'); + expect(res.body.available).toBe(true); + expect(res.body.record).toEqual({ wins: 1, losses: 1, pushes: 0 }); + }); +}); + +describe('GET /api/content/preview/:sport/:gameId', () => { + test('returns matchup data for a scheduled game', async () => { + const res = await request(mountApp()).get('/api/content/preview/mlb/g1'); + expect(res.status).toBe(200); + expect(res.body.type).toBe('matchup_preview'); + expect(res.body.home.abbreviation).toBe('CIN'); + expect(res.body.lines.bookCount).toBe(1); + expect(res.body.playerStreaks.map((s) => s.player)).toEqual(['Acuna']); + }); + + test('unknown game → 404', async () => { + const res = await request(mountApp()).get('/api/content/preview/mlb/nope'); + expect(res.status).toBe(404); + }); +}); diff --git a/tests/unit/contentFormatter.test.js b/tests/unit/contentFormatter.test.js new file mode 100644 index 0000000..45befe1 --- /dev/null +++ b/tests/unit/contentFormatter.test.js @@ -0,0 +1,59 @@ +// Unit: content formatter (Session 29). + +const fmt = require('../../src/services/contentFormatter'); + +describe('contentFormatter — slate thread', () => { + test('formats each post role into readable text', () => { + const thread = { + posts: [ + { role: 'hook', text: '🔥 VYNDR SLATE' }, + { role: 'pick', player: 'Wemby', stat: 'points', side: 'over', line: 28.5, grade: 'A', confidence: 78, edge: 6.2, analysis: 'Edge on the line.' }, + { role: 'game_highlight', game: 'ARI @ CIN', time: null, bestHomeML: { odds: '-120', book: 'betmgm' }, total: 9.5, bookCount: 2 }, + { role: 'movers', movers: [{ player: 'X', stat: 'hits', opening: 1.5, current: 2.5, delta: 1, sharpSignal: false }] }, + { role: 'schedule', games: [{ away: 'ATL', home: 'NYM', time: null }] }, + { role: 'cta', text: 'Full slate at vyndr.app' }, + ], + }; + const out = fmt.formatSlateThread(thread); + expect(out[0]).toBe('🔥 VYNDR SLATE'); + expect(out[1]).toContain('Wemby · points over 28.5'); + expect(out[1]).toContain('Grade: A · 78% confidence'); + expect(out[1]).toContain('Edge: +6.2%'); + expect(out[2]).toContain('ARI @ CIN'); + expect(out[2]).toContain('betmgm'); + expect(out[3]).toContain('Biggest line moves'); + expect(out[4]).toContain('ATL @ NYM'); + expect(out[5]).toBe('Full slate at vyndr.app'); + }); + + test('never emits "undefined"', () => { + const out = fmt.formatSlateThread({ posts: [{ role: 'pick', player: 'X', stat: 'pts', grade: 'B' }] }); + expect(out[0]).not.toMatch(/undefined/); + }); + + test('non-thread input → empty array', () => { + expect(fmt.formatSlateThread(null)).toEqual([]); + }); +}); + +describe('contentFormatter — POTD + recap', () => { + test('POTD full', () => { + const t = fmt.formatPOTD({ dataLevel: 'full', player: 'Wemby', stat: 'points', side: 'over', line: 28.5, grade: 'A', confidence: 78 }); + expect(t).toContain('PROP OF THE DAY'); + expect(t).toContain('Wemby'); + }); + test('POTD game of the day', () => { + const t = fmt.formatPOTD({ dataLevel: 'lines', subtype: 'game_of_the_day', game: 'ARI @ CIN', total: 9.5 }); + expect(t).toContain('GAME OF THE DAY'); + expect(t).toContain('O/U 9.5'); + }); + test('POTD unavailable', () => { + expect(fmt.formatPOTD({ available: false })).toMatch(/vyndr\.app/); + }); + test('recap formats record + win rate', () => { + const t = fmt.formatRecap({ available: true, date: 'Friday', record: { wins: 3, losses: 2, pushes: 1 }, winRate: 0.6, topHits: [{ player: 'A', stat: 'points', side: 'over', line: 20 }], metrics: { brierScore: 0.18 } }); + expect(t).toContain('3-2-1 (60%)'); + expect(t).toContain('✅ A points over 20'); + expect(t).toContain('Brier: 0.18'); + }); +}); diff --git a/tests/unit/contentTemplateService.test.js b/tests/unit/contentTemplateService.test.js new file mode 100644 index 0000000..4b5a004 --- /dev/null +++ b/tests/unit/contentTemplateService.test.js @@ -0,0 +1,166 @@ +// Unit: content template engine (Session 29). Pure generators. + +const svc = require('../../src/services/contentTemplateService'); + +// ---- fixtures ---- +const schedule = [ + { id: '1', homeTeam: { name: 'Cincinnati Reds', abbreviation: 'CIN' }, awayTeam: { name: 'Arizona Diamondbacks', abbreviation: 'ARI' }, gameTime: '2026-06-13T23:10:00Z', venue: 'GABP' }, + { id: '2', homeTeam: { name: 'New York Mets', abbreviation: 'NYM' }, awayTeam: { name: 'Atlanta Braves', abbreviation: 'ATL' }, gameTime: '2026-06-13T23:40:00Z', venue: 'Citi Field' }, +]; +const gameLines = { + '20260613_ARI@CIN': { + homeTeam: 'CIN', awayTeam: 'ARI', + books: { + bet365: { homeML: '-130', awayML: '+110', total: '9.5', homeSpread: '-1.5' }, + betmgm: { homeML: '-120', awayML: '+100', total: '9', homeSpread: '-1.5' }, + }, + }, +}; +const grades = [ + { player: 'Wembanyama', stat: 'points', line: 28.5, side: 'over', grade: 'A', confidence: 78, edge: 6.2, projection: 31.1 }, + { player_name: 'Brunson', stat_type: 'assists', line: 7.5, direction: 'under', grade: 'B+', confidence: 64, edge_pct: 3.1, projection: 6.8 }, +]; + +describe('determineDataLevel', () => { + test('full when grades exist', () => { + expect(svc.determineDataLevel({ grades, gameLines, schedule })).toBe('full'); + }); + test('lines when only game lines', () => { + expect(svc.determineDataLevel({ grades: [], gameLines, schedule })).toBe('lines'); + }); + test('schedule when only schedule', () => { + expect(svc.determineDataLevel({ grades: [], gameLines: {}, schedule })).toBe('schedule'); + }); + test('empty when nothing', () => { + expect(svc.determineDataLevel({ grades: [], gameLines: {}, schedule: [] })).toBe('empty'); + }); +}); + +describe('generateSlateThread', () => { + test('full data → hook + up-to-5 picks + CTA', () => { + const t = svc.generateSlateThread('nba', { schedule, gameLines, grades, dataLevel: 'full' }); + expect(t.posts[0].role).toBe('hook'); + expect(t.posts[t.posts.length - 1].role).toBe('cta'); + const picks = t.posts.filter((p) => p.role === 'pick'); + expect(picks).toHaveLength(2); + // Sorted by confidence desc — Wemby (78) first. + expect(picks[0].player).toBe('Wembanyama'); + expect(picks[1].player).toBe('Brunson'); // normalized from player_name + expect(picks[1].side).toBe('under'); // normalized from direction + }); + + test('lines-only → game highlights (+ movers when present)', () => { + const t = svc.generateSlateThread('mlb', { schedule, gameLines, grades: [], movers: [{ player: 'X', delta: 2 }] }); + expect(t.dataLevel).toBe('lines'); + const highlights = t.posts.filter((p) => p.role === 'game_highlight'); + expect(highlights.length).toBeGreaterThan(0); + expect(highlights[0].bookCount).toBe(2); + expect(t.posts.some((p) => p.role === 'movers')).toBe(true); + }); + + test('schedule-only → game list', () => { + const t = svc.generateSlateThread('mlb', { schedule, gameLines: {}, grades: [] }); + expect(t.dataLevel).toBe('schedule'); + const sched = t.posts.find((p) => p.role === 'schedule'); + expect(sched.games).toHaveLength(2); + expect(sched.games[0].home).toBe('Cincinnati Reds'); + }); + + test('hook text adapts to data level', () => { + expect(svc.generateSlateThread('mlb', { schedule, grades }).posts[0].text).toMatch(/graded/i); + expect(svc.generateSlateThread('mlb', { schedule, gameLines, grades: [] }).posts[0].text).toMatch(/sportsbooks/i); + }); + + test('empty schedule → minimal thread, not a crash', () => { + const t = svc.generateSlateThread('mlb', { schedule: [], gameLines: {}, grades: [] }); + expect(t.dataLevel).toBe('empty'); + expect(t.posts[0].role).toBe('hook'); + expect(t.posts.some((p) => p.role === 'cta')).toBe(true); + }); +}); + +describe('generatePOTD', () => { + test('full → best grade selected', () => { + const p = svc.generatePOTD('nba', { grades, dataLevel: 'full' }); + expect(p.dataLevel).toBe('full'); + expect(p.player).toBe('Wembanyama'); // highest confidence + expect(p.grade).toBe('A'); + }); + test('lines → game of the day', () => { + const p = svc.generatePOTD('mlb', { grades: [], gameLines, schedule }); + expect(p.dataLevel).toBe('lines'); + expect(p.subtype).toBe('game_of_the_day'); + expect(p.game).toContain('@'); + }); + test('no data → available:false', () => { + const p = svc.generatePOTD('mlb', { grades: [], gameLines: {}, schedule: [] }); + expect(p.available).toBe(false); + }); +}); + +describe('generateResultsRecap', () => { + const resolved = [ + { player: 'A', stat: 'points', grade: 'A', confidence: 80, result: 'win', clv: 1.2 }, + { player: 'B', stat: 'hits', grade: 'A-', confidence: 70, result: 'win', clv: 0.8 }, + { player: 'C', stat: 'reb', grade: 'B', confidence: 60, result: 'win', clv: -0.2 }, + { player: 'D', stat: 'ast', grade: 'A', confidence: 75, result: 'loss', clv: -1.0 }, + { player: 'E', stat: 'ks', grade: 'B', confidence: 55, result: 'loss', clv: 0.1 }, + { player: 'F', stat: 'tb', grade: 'C', confidence: 50, result: 'push' }, + ]; + test('record + win rate', () => { + const r = svc.generateResultsRecap('mlb', resolved); + expect(r.record).toEqual({ wins: 3, losses: 2, pushes: 1 }); + expect(r.winRate).toBe(0.6); // 3 / (3+2) + }); + test('top hits sorted by confidence', () => { + const r = svc.generateResultsRecap('mlb', resolved); + expect(r.topHits[0].player).toBe('A'); // 80 conf win + expect(r.topHits).toHaveLength(3); + }); + test('biggest miss = highest-confidence loss', () => { + const r = svc.generateResultsRecap('mlb', resolved); + expect(r.biggestMiss.player).toBe('D'); // 75 conf loss + }); + test('by-tier breakdown', () => { + const r = svc.generateResultsRecap('mlb', resolved); + expect(r.byTier.A).toEqual({ wins: 2, losses: 1, total: 3 }); + expect(r.byTier.B).toEqual({ wins: 1, losses: 1, total: 2 }); + }); + test('brier + clv computed', () => { + const r = svc.generateResultsRecap('mlb', resolved); + expect(typeof r.metrics.brierScore).toBe('number'); + expect(typeof r.metrics.clv).toBe('number'); + }); + test('no resolved grades → available:false', () => { + expect(svc.generateResultsRecap('mlb', []).available).toBe(false); + }); +}); + +describe('generateMatchupPreview', () => { + const streaks = [ + { player: 'Acuna', team: 'ARI', description: '3-game hit streak', currentStreak: 3 }, + { player: 'Soto', team: 'NYM', description: '2-game HR streak', currentStreak: 2 }, + ]; + test('includes teams, time, venue', () => { + const p = svc.generateMatchupPreview(schedule[0], gameLines['20260613_ARI@CIN'], streaks); + expect(p.home.abbreviation).toBe('CIN'); + expect(p.away.abbreviation).toBe('ARI'); + expect(p.venue).toBe('GABP'); + }); + test('lines summary from game lines', () => { + const p = svc.generateMatchupPreview(schedule[0], gameLines['20260613_ARI@CIN'], streaks); + expect(p.lines.bookCount).toBe(2); + expect(p.lines.total).toBe(9.3); // consensus of 9.5 + 9, rounded to 1dp + expect(p.lines.homeFavorite).toBe(true); // homeSpread -1.5 + }); + test('player streaks matched to the two teams', () => { + const p = svc.generateMatchupPreview(schedule[0], gameLines['20260613_ARI@CIN'], streaks); + expect(p.playerStreaks.map((s) => s.player)).toEqual(['Acuna']); // only ARI matches this game + }); + test('no game lines → lines:null, not a crash', () => { + const p = svc.generateMatchupPreview(schedule[0], null, streaks); + expect(p.lines).toBeNull(); + expect(p.narrative).toContain('@'.length ? '' : ''); // narrative present + expect(typeof p.narrative).toBe('string'); + }); +}); diff --git a/web/public/sw.js b/web/public/sw.js index 9a52ddc..e6ce2a1 100644 --- a/web/public/sw.js +++ b/web/public/sw.js @@ -1,2 +1,2 @@ (()=>{"use strict";let e,t,a,s,r,i={googleAnalytics:"googleAnalytics",precache:"precache-v2",prefix:"serwist",runtime:"runtime",suffix:"u">typeof registration?registration.scope:""},n=e=>[i.prefix,e,i.suffix].filter(e=>e&&e.length>0).join("-"),c=e=>e||n(i.precache),o=e=>e||n(i.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 f(e,t,a,s){let r=d(t.url,a);if(t.url===r)return e.match(t,s);let i={...s,ignoreSearch:!0};for(let n of(await e.keys(t,i)))if(r===d(n.url,a))return e.match(n,s)}var p=class{promise;resolve;reject;constructor(){this.promise=new Promise((e,t)=>{this.resolve=e,this.reject=t})}};let m=async()=>{for(let e of u)await e()},w="-precache-",g=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),b=new WeakMap,v=new WeakMap,R=new WeakMap,E={get(e,t,a){if(e instanceof IDBTransaction){if("done"===t)return b.get(e);if("store"===t)return a.objectStoreNames[1]?void 0:a.objectStore(a.objectStoreNames[0])}return q(e[t])},set:(e,t,a)=>(e[t]=a,!0),has:(e,t)=>e instanceof IDBTransaction&&("done"===t||"store"===t)||t in e};function q(e){if(e instanceof IDBRequest){let t;return t=new Promise((t,a)=>{let s=()=>{e.removeEventListener("success",r),e.removeEventListener("error",i)},r=()=>{t(q(e.result)),s()},i=()=>{a(e.error),s()};e.addEventListener("success",r),e.addEventListener("error",i)}),R.set(t,e),t}if(v.has(e))return v.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(x(this),t),q(this.request)}:function(...t){return q(e.apply(x(this),t))};return(e instanceof IDBTransaction&&function(e){if(b.has(e))return;let t=new Promise((t,a)=>{let s=()=>{e.removeEventListener("complete",r),e.removeEventListener("error",i),e.removeEventListener("abort",i)},r=()=>{t(),s()},i=()=>{a(e.error||new DOMException("AbortError","AbortError")),s()};e.addEventListener("complete",r),e.addEventListener("error",i),e.addEventListener("abort",i)});b.set(e,t)}(e),_(e,s||(s=[IDBDatabase,IDBObjectStore,IDBIndex,IDBCursor,IDBTransaction])))?new Proxy(e,E):e}(e);return t!==e&&(v.set(e,t),R.set(t,e)),t}let x=e=>R.get(e);function D(e,t,{blocked:a,upgrade:s,blocking:r,terminated:i}={}){let n=indexedDB.open(e,t),c=q(n);return s&&n.addEventListener("upgradeneeded",e=>{s(q(n.result),e.oldVersion,e.newVersion,q(n.transaction),e)}),a&&n.addEventListener("blocked",e=>a(e.oldVersion,e.newVersion,e)),c.then(e=>{i&&e.addEventListener("close",()=>i()),r&&e.addEventListener("versionchange",e=>r(e.oldVersion,e.newVersion,e))}).catch(()=>{}),c}let S=["get","getKey","getAll","getAllKeys","count"],k=["put","add","delete","clear"],T=new Map;function P(e,t){if(!(e instanceof IDBDatabase&&!(t in e)&&"string"==typeof t))return;if(T.get(t))return T.get(t);let a=t.replace(/FromIndex$/,""),s=t!==a,r=k.includes(a);if(!(a in(s?IDBIndex:IDBObjectStore).prototype)||!(r||S.includes(a)))return;let i=async function(e,...t){let i=this.transaction(e,r?"readwrite":"readonly"),n=i.store;return s&&(n=n.index(t.shift())),(await Promise.all([n[a](...t),r&&i.done]))[0]};return T.set(t,i),i}E={...e=E,get:(t,a,s)=>P(t,a)||e.get(t,a,s),has:(t,a)=>!!P(t,a)||e.has(t,a)};let C=["continue","continuePrimaryKey","advance"],N={},I=new WeakMap,U=new WeakMap,L={get(e,t){if(!C.includes(t))return e[t];let a=N[t];return a||(a=N[t]=function(...e){I.set(this,U.get(this)[t](...e))}),a}};async function*A(...e){let t=this;if(t instanceof IDBCursor||(t=await t.openCursor(...e)),!t)return;let a=new Proxy(t,L);for(U.set(a,t),R.set(a,x(t));t;)yield a,t=await (I.get(a)||t.continue()),I.delete(a)}function O(e,t){return t===Symbol.asyncIterator&&_(e,[IDBIndex,IDBObjectStore,IDBCursor])||"iterate"===t&&_(e,[IDBIndex,IDBObjectStore])}E={...t=E,get:(e,a,s)=>O(e,a)?A:t.get(e,a,s),has:(e,a)=>O(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(),i={headers:new Headers(r.headers),status:r.status,statusText:r.statusText},n=t?t(i):i,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,n)},B="requests",K="queueName";var F=class{_db=null;async addEntry(e){let t=(await this.getDb()).transaction(B,"readwrite",{durability:"relaxed"});await t.store.add(e),await t.done}async getFirstEntryId(){return(await (await this.getDb()).transaction(B).store.openCursor())?.value.id}async getAllEntriesByQueueName(e){return await (await this.getDb()).getAllFromIndex(B,K,IDBKeyRange.only(e))||[]}async getEntryCountByQueueName(e){return(await this.getDb()).countFromIndex(B,K,IDBKeyRange.only(e))}async deleteEntry(e){await (await this.getDb()).delete(B,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(B).store.index(K).openCursor(e,t))?.value}async getDb(){return this._db||(this._db=await D("serwist-background-sync",3,{upgrade:this._upgradeDb})),this._db}_upgradeDb(e,t){t>0&&t<3&&e.objectStoreNames.contains(B)&&e.deleteObjectStore(B),e.createObjectStore(B,{autoIncrement:!0,keyPath:"id"}).createIndex(K,K,{unique:!1})}},W=class{_queueName;_queueDb;constructor(e){this._queueName=e,this._queueDb=new F}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 H=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 $="serwist-background-sync",V=new Set,Q=e=>{let t={request:new H(e.requestData).toRequest(),timestamp:e.timestamp};return e.metadata&&(t.metadata=e.metadata),t};var G=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(Q(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 H.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):Q(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(`${$}:${this._name}`)}catch(e){}}_addSyncListener(){"sync"in self.registration&&!this._forceSyncFallback?self.addEventListener("sync",e=>{if(e.tag===`${$}:${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 G(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 p,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 i=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:i,response:e});return e}catch(e){throw r&&await this.runCallbacks("fetchDidFail",{error:e,event:t,originalRequest:r.clone(),request:i.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,i=await this.getCacheKey(a,"read"),n={...r,cacheName:s};for(let e of(t=await caches.match(i,n),this.iterateCallbacks("cachedResponseWillBeUsed")))t=await e({cacheName:s,matchOptions:r,cachedResponse:t,request:i,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:i,matchOptions:n}=this._strategy,c=await self.caches.open(i),o=this.hasCallback("cacheDidUpdate"),u=o?await f(c,s.clone(),["__WB_REVISION__"],n):null;try{await c.put(s,o?r.clone():r)}catch(e){if(e instanceof Error)throw"QuotaExceededError"===e.name&&await m(),e}for(let e of this.iterateCallbacks("cacheDidUpdate"))await e({cacheName:i,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 i of e.iterateCallbacks("handlerDidError"))if(void 0!==(s=await i({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,i;try{r=await e}catch{}try{await t.runCallbacks("handlerDidRespond",{event:s,request:a,response:r}),await t.doneWaiting()}catch(e){e instanceof Error&&(i=e)}if(await t.runCallbacks("handlerDidComplete",{event:s,request:a,response:r,error:i}),t.destroy(),i)throw i}},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:i,promise:n}=this._getTimeoutPromise({request:e,logs:s,handler:t});a=i,r.push(n)}let i=this._getNetworkPromise({timeoutId:a,request:e,logs:s,handler:t});r.push(i);let n=await t.waitUntil((async()=>await t.waitUntil(Promise.race(r))||await i)());if(!n)throw new l("no-response",{url:e.url});return n}_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,i;try{i=await s.fetchAndCachePut(t)}catch(e){e instanceof Error&&(r=e)}return e&&clearTimeout(e),(r||!i)&&(i=await s.cacheMatch(t)),i}},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,i=e.integrity,n=!i||i===r;a=await t.fetch(new Request(e,{integrity:"no-cors"!==e.mode?i||r:void 0})),r&&n&&"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)}},ei=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))}},en=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 i=await a(r.item);t.push({result:i,index:r.index})}},i=Array.from({length:e},()=>new Promise(r));return(await Promise.all(i)).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)),q(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 i=a.value;i.cacheName===this._cacheName&&(e&&i.timestamp=t?(a.delete(),s.push(i.url)):r++),a=await a.continue()}return s}async getDb(){return this._db||(this._db=await D("serwist-expiration",1,{upgrade:this._upgradeDbAndDeleteOldDbs.bind(this)})),this._db}},ef=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 ef(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),i=this._getCacheExpiration(t),n="last-used"===this._config.maxAgeFrom,c=(async()=>{n&&await i.updateTimestamp(a.url),await i.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 em=/^\/(\w+\/)?collect/,ew=({serwist:e,cacheName:t,...a})=>{let s,r,c=t||n(i.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,i=new URL(s.url);try{let e="POST"===s.method?new URLSearchParams(await s.clone().text()):i.searchParams,t=r-(Number(e.get("qt"))||0),n=Date.now()-t;if(e.set("qt",String(n)),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(i.origin+i.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&&em.test(e.pathname),r=new et({plugins:[o]}),"GET"),new es(s,r,"POST")])e.registerRoute(t)};var eg=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}}},ey=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}},e_=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 i=new URL(e,location.href);i.hash="",yield i.href;let n=((e,t=[])=>{for(let a of[...e.searchParams.keys()])t.some(e=>e.test(a))&&e.searchParams.delete(a);return e})(i,a);if(yield n.href,t&&n.pathname.endsWith("/")){let e=new URL(n.href);e.pathname+=t,yield e.href}if(s){let e=new URL(n.href);e.pathname+=".html",yield e.href}if(r)for(let e of r({url:i}))yield e.href}(a.url,t)){let t=s.get(r);if(t)return{cacheKey:t,integrity:e.getIntegrityForPrecacheKey(t)}}},e.precacheStrategy)}},eb=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}},ev=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:n,clientsClaim:o=!1,runtimeCaching:l,offlineAnalyticsConfig:h,disableDevLogs:u=!1,fallbacks:d,requestRules:f}={}){const{precacheStrategyOptions:p,precacheRouteOptions:m,precacheMiscOptions:w}=((e,t={})=>{let{cacheName:a,plugins:s=[],fetchOptions:r,matchOptions:i,fallbackToNetwork:n,directoryIndex:o,ignoreURLParametersMatching:l,cleanURLs:h,urlManipulation:u,cleanupOutdatedCaches:d,concurrency:f=10,navigateFallback:p,navigateFallbackAllowlist:m,navigateFallbackDenylist:w}=t??{};return{precacheStrategyOptions:{cacheName:c(a),plugins:[...s,new eb({precacheController:e})],fetchOptions:r,matchOptions:i,fallbackToNetwork:n},precacheRouteOptions:{directoryIndex:o,ignoreURLParametersMatching:l,cleanURLs:h,urlManipulation:u},precacheMiscOptions:{cleanupOutdatedCaches:d,concurrency:f,navigateFallback:p,navigateFallbackAllowlist:m,navigateFallbackDenylist:w}}})(this,t);if(this._concurrentPrecaching=w.concurrency,this._precacheStrategy=new er(p),this._routes=new Map,this._defaultHandlerMap=new Map,this._requestRules=f,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!==n&&(e=>{var t=e;for(let e of Object.keys(i))(e=>{let a=t[e];"string"==typeof a&&(i[e]=a)})(e)})({prefix:n}),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(g(c(e)).then(e=>{}))})})(p.cacheName),this.registerRoute(new e_(this,m)),w.navigateFallback&&this.registerRoute(new ei(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 eg({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),i=new Request(t,{integrity:s,cache:r,credentials:"same-origin"});await Promise.all(this.precacheStrategy.handleAll({event:e,request:i,url:new URL(i.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 en(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:i,route:n}=this.findMatchingRoute({event:t,request:e,sameOrigin:r,url:s}),c=n?.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:i})}catch(e){a=Promise.reject(e)}let l=n?.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:i})}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 i,n=r.match({url:e,sameOrigin:t,request:a,event:s});if(n)return Array.isArray(i=n)&&0===i.length||n.constructor===Object&&0===Object.keys(n).length?i=void 0:"boolean"==typeof n&&(i=void 0),{route:r,params:i}}return{}}};let eR="/offline",eE=["pages","api-responses","next-static","static-media","fallback","offline-fallback"],eq=[{matcher:({url:e,sameOrigin:t})=>t&&e.pathname.startsWith("/api/"),handler:new ee({cacheName:"api-responses",networkTimeoutSeconds:5,plugins:[new ep({maxEntries:100,maxAgeSeconds:3600})]})},{matcher:({request:e})=>"navigate"===e.mode,handler:new ee({cacheName:"pages",networkTimeoutSeconds:5,plugins:[new ep({maxEntries:50,maxAgeSeconds:86400}),{handlerDidError:async()=>await caches.match(eR)||Response.error()}]})},{matcher:({url:e,sameOrigin:t})=>t&&e.pathname.startsWith("/_next/static/"),handler:new ey({cacheName:"next-static",plugins:[new ep({maxEntries:200,maxAgeSeconds:2592e3})]})},{matcher:({url:e})=>e.pathname.startsWith("/images/")||e.pathname.startsWith("/icons/")||/\.(?:png|jpe?g|gif|svg|webp|ico|woff2?)$/.test(e.pathname),handler:new ey({cacheName:"static-media",plugins:[new ep({maxEntries:100,maxAgeSeconds:2592e3})]})},{matcher:()=>!0,handler:new ee({cacheName:"fallback",networkTimeoutSeconds:5,plugins:[new ep({maxEntries:50,maxAgeSeconds:86400})]})}];new ev({precacheEntries:[{'revision':'f2636d328ec971216a28fd25d0ad5283','url':'/_next/static/GrOWVkWbsVYxDVGVsTB2n/_buildManifest.js'},{'revision':'b6652df95db52feb4daf4eca35380933','url':'/_next/static/GrOWVkWbsVYxDVGVsTB2n/_ssgManifest.js'},{'revision':null,'url':'/_next/static/chunks/1896-359ed2e68a8c4f1d.js'},{'revision':null,'url':'/_next/static/chunks/1958-56cbbb8dba15fd2f.js'},{'revision':null,'url':'/_next/static/chunks/2346-d63ba3de1d09a41e.js'},{'revision':null,'url':'/_next/static/chunks/4bd1b696-e356ca5ba0218e27.js'},{'revision':null,'url':'/_next/static/chunks/52774a7f.f2ab7a9b4b8ba576.js'},{'revision':null,'url':'/_next/static/chunks/5838-7e0aa455e9f24259.js'},{'revision':null,'url':'/_next/static/chunks/6810-65ea0987ee90a78b.js'},{'revision':null,'url':'/_next/static/chunks/7602.2868ff821d53ee09.js'},{'revision':null,'url':'/_next/static/chunks/7918-9cfe56e3ee27ed27.js'},{'revision':null,'url':'/_next/static/chunks/8500-f62a38ff68ab7f42.js'},{'revision':null,'url':'/_next/static/chunks/9da6db1e-9623f3245f088d02.js'},{'revision':null,'url':'/_next/static/chunks/app/_global-error/page-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/_not-found/page-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/account/page-f5e2a6d5e729cc87.js'},{'revision':null,'url':'/_next/static/chunks/app/admin/page-7653c5a94021e122.js'},{'revision':null,'url':'/_next/static/chunks/app/api/admin/stats/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/books/%5B...path%5D/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/checkout/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/gamelines/%5Bsport%5D/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/props/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/tonight/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/hero-prop/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/hotlist/%5Bsport%5D/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/intelligence/feed/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/accuracy/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/lines/%5B...path%5D/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/mlb/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/nba/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/soccer/%5Bleague%5D/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/wnba/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/add-leg/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/calculate/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/grade/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/players/search/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/live/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/most-parlayed/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/top-graded/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/scan/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/schedule/%5Bsport%5D/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/parlays-graded/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/public/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/streaks/%5Bsport%5D/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/profile/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/recent-scans/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/scans/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/waitlist/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/webhook/nexapay/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/api/welcome-email/route-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/auth/callback/page-0cb8d3790d034de0.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/%5Bslug%5D/page-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/page-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/dashboard/page-79c033d3218c9272.js'},{'revision':null,'url':'/_next/static/chunks/app/forgot-password/page-a251f52b4206e8d1.js'},{'revision':null,'url':'/_next/static/chunks/app/game/%5Bid%5D/page-56eb3dff903c944e.js'},{'revision':null,'url':'/_next/static/chunks/app/intelligence/page-5f6c106fd6c53851.js'},{'revision':null,'url':'/_next/static/chunks/app/layout-c72e0130bc50baac.js'},{'revision':null,'url':'/_next/static/chunks/app/ledger/page-3bfc2406ebeb6d2e.js'},{'revision':null,'url':'/_next/static/chunks/app/login/page-69c1ebd6ff5c12d8.js'},{'revision':null,'url':'/_next/static/chunks/app/marketplace/page-cbada49ae96a7527.js'},{'revision':null,'url':'/_next/static/chunks/app/not-found-1fd99a98dab3c771.js'},{'revision':null,'url':'/_next/static/chunks/app/offline/page-a7a57ff039afc1af.js'},{'revision':null,'url':'/_next/static/chunks/app/page-70c10ec17b5cf43c.js'},{'revision':null,'url':'/_next/static/chunks/app/pricing/page-3f26af475b3f8452.js'},{'revision':null,'url':'/_next/static/chunks/app/privacy/page-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/profile/page-0a7a9e981d344a6f.js'},{'revision':null,'url':'/_next/static/chunks/app/responsible-gambling/page-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/scan/page-8f1c2629fa6951f2.js'},{'revision':null,'url':'/_next/static/chunks/app/settings/security/page-59df87aff0c62ec8.js'},{'revision':null,'url':'/_next/static/chunks/app/signup/page-33e502bb04569f60.js'},{'revision':null,'url':'/_next/static/chunks/app/soccer/page-9ba3b46b0b1dfa59.js'},{'revision':null,'url':'/_next/static/chunks/app/terms/page-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/app/tracker/page-d6937e1faa494c49.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/cancel/page-1fd99a98dab3c771.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/desk/page-0ed948b44f9c64d7.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/success/page-b6baded9f44ae951.js'},{'revision':null,'url':'/_next/static/chunks/app/verify/page-cdeec62ab3a5d8ff.js'},{'revision':null,'url':'/_next/static/chunks/app/welcome/page-e3ab6d5822dee6dc.js'},{'revision':null,'url':'/_next/static/chunks/framework-95da69ac6843d788.js'},{'revision':null,'url':'/_next/static/chunks/main-a354262253293123.js'},{'revision':null,'url':'/_next/static/chunks/main-app-4231d5f500f33972.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/app-error-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/forbidden-30c428a14fcfa3f4.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/global-error-756d787dba63aa4d.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/unauthorized-30c428a14fcfa3f4.js'},{'revision':'846118c33b2c0e922d7b3a7676f81f6f','url':'/_next/static/chunks/polyfills-42372ed130431b0a.js'},{'revision':null,'url':'/_next/static/chunks/webpack-a41d593db6ea66d4.js'},{'revision':null,'url':'/_next/static/css/0e3cdd5e3f69836f.css'},{'revision':'d7bc0f63f9618b8b700d028d7d242de8','url':'/apple-touch-icon.png'},{'revision':'5616f81ec5f8a9f726a17fb7ac66a54a','url':'/favicon-16.png'},{'revision':'1fc285cfb0116a0523eb34c779a7d282','url':'/favicon-32.png'},{'revision':'583b034f36839a8af92841771eef38b6','url':'/favicon.ico'},{'revision':'b15795128b3691e5703d0ce14ce10827','url':'/favicon.png'},{'revision':'44e9d71551be13574ac2e10063d03aa9','url':'/favicon.svg'},{'revision':'252b8f8c4d0f64c7cd120f182a6c74ab','url':'/icon-192.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icon-512.png'},{'revision':'252b8f8c4d0f64c7cd120f182a6c74ab','url':'/icons/icon-192.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icons/icon-512.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icons/icon-maskable-512.png'},{'revision':'bf670b1fafa1b3127bd50abe86d723b9','url':'/images/player-silhouette.svg'},{'revision':'d31c450ad857fff6798872411d72f42b','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("install",e=>{e.waitUntil(caches.open("offline-fallback").then(e=>e.add(eR)).catch(()=>{}))}),self.addEventListener("activate",e=>{e.waitUntil(caches.keys().then(e=>Promise.all(e.filter(e=>!eE.includes(e)&&!e.startsWith("serwist")).map(e=>(console.log("[SW] deleting stale cache:",e),caches.delete(e))))))}),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:i="/",tag:n="vyndr-notification"}=t;e.waitUntil(self.registration.showNotification(a,{body:s,icon:r,badge:"/icons/icon-192.png",tag:n,data:{url:i}}))}),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),i=new Request(t,{integrity:s,cache:r,credentials:"same-origin"});await Promise.all(this.precacheStrategy.handleAll({event:e,request:i,url:new URL(i.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 en(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:i,route:n}=this.findMatchingRoute({event:t,request:e,sameOrigin:r,url:s}),c=n?.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:i})}catch(e){a=Promise.reject(e)}let l=n?.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:i})}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 i,n=r.match({url:e,sameOrigin:t,request:a,event:s});if(n)return Array.isArray(i=n)&&0===i.length||n.constructor===Object&&0===Object.keys(n).length?i=void 0:"boolean"==typeof n&&(i=void 0),{route:r,params:i}}return{}}};let eR="/offline",eE=["pages","api-responses","next-static","static-media","fallback","offline-fallback"],eq=[{matcher:({url:e,sameOrigin:t})=>t&&e.pathname.startsWith("/api/"),handler:new ee({cacheName:"api-responses",networkTimeoutSeconds:5,plugins:[new ep({maxEntries:100,maxAgeSeconds:3600})]})},{matcher:({request:e})=>"navigate"===e.mode,handler:new ee({cacheName:"pages",networkTimeoutSeconds:5,plugins:[new ep({maxEntries:50,maxAgeSeconds:86400}),{handlerDidError:async()=>await caches.match(eR)||Response.error()}]})},{matcher:({url:e,sameOrigin:t})=>t&&e.pathname.startsWith("/_next/static/"),handler:new ey({cacheName:"next-static",plugins:[new ep({maxEntries:200,maxAgeSeconds:2592e3})]})},{matcher:({url:e})=>e.pathname.startsWith("/images/")||e.pathname.startsWith("/icons/")||/\.(?:png|jpe?g|gif|svg|webp|ico|woff2?)$/.test(e.pathname),handler:new ey({cacheName:"static-media",plugins:[new ep({maxEntries:100,maxAgeSeconds:2592e3})]})},{matcher:()=>!0,handler:new ee({cacheName:"fallback",networkTimeoutSeconds:5,plugins:[new ep({maxEntries:50,maxAgeSeconds:86400})]})}];new ev({precacheEntries:[{'revision':'97139c7de68b95f4af6e4f938ed537f8','url':'/_next/static/8MeesYaRWPVld3dsOVnjq/_buildManifest.js'},{'revision':'b6652df95db52feb4daf4eca35380933','url':'/_next/static/8MeesYaRWPVld3dsOVnjq/_ssgManifest.js'},{'revision':null,'url':'/_next/static/chunks/1896-359ed2e68a8c4f1d.js'},{'revision':null,'url':'/_next/static/chunks/1958-56cbbb8dba15fd2f.js'},{'revision':null,'url':'/_next/static/chunks/2346-d63ba3de1d09a41e.js'},{'revision':null,'url':'/_next/static/chunks/4bd1b696-e356ca5ba0218e27.js'},{'revision':null,'url':'/_next/static/chunks/52774a7f.f2ab7a9b4b8ba576.js'},{'revision':null,'url':'/_next/static/chunks/5838-7e0aa455e9f24259.js'},{'revision':null,'url':'/_next/static/chunks/6810-65ea0987ee90a78b.js'},{'revision':null,'url':'/_next/static/chunks/7602.2868ff821d53ee09.js'},{'revision':null,'url':'/_next/static/chunks/7918-9cfe56e3ee27ed27.js'},{'revision':null,'url':'/_next/static/chunks/8500-f62a38ff68ab7f42.js'},{'revision':null,'url':'/_next/static/chunks/9da6db1e-9623f3245f088d02.js'},{'revision':null,'url':'/_next/static/chunks/app/_global-error/page-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/_not-found/page-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/account/page-f5e2a6d5e729cc87.js'},{'revision':null,'url':'/_next/static/chunks/app/admin/page-7653c5a94021e122.js'},{'revision':null,'url':'/_next/static/chunks/app/api/admin/stats/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/books/%5B...path%5D/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/checkout/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/content/%5B...path%5D/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/gamelines/%5Bsport%5D/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/props/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/%5Bid%5D/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/games/tonight/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/hero-prop/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/hotlist/%5Bsport%5D/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/intelligence/feed/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/accuracy/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/ledger/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/lines/%5B...path%5D/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/mlb/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/nba/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/soccer/%5Bleague%5D/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/odds/wnba/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/add-leg/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/calculate/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/parlay/grade/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/players/search/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/live/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/most-parlayed/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/props/top-graded/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/scan/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/schedule/%5Bsport%5D/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/parlays-graded/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/stats/public/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/streaks/%5Bsport%5D/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/profile/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/recent-scans/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/user/scans/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/waitlist/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/webhook/nexapay/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/api/welcome-email/route-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/auth/callback/page-0cb8d3790d034de0.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/%5Bslug%5D/page-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/blog/page-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/dashboard/page-79c033d3218c9272.js'},{'revision':null,'url':'/_next/static/chunks/app/forgot-password/page-a251f52b4206e8d1.js'},{'revision':null,'url':'/_next/static/chunks/app/game/%5Bid%5D/page-56eb3dff903c944e.js'},{'revision':null,'url':'/_next/static/chunks/app/intelligence/page-5f6c106fd6c53851.js'},{'revision':null,'url':'/_next/static/chunks/app/layout-c72e0130bc50baac.js'},{'revision':null,'url':'/_next/static/chunks/app/ledger/page-3bfc2406ebeb6d2e.js'},{'revision':null,'url':'/_next/static/chunks/app/login/page-69c1ebd6ff5c12d8.js'},{'revision':null,'url':'/_next/static/chunks/app/marketplace/page-cbada49ae96a7527.js'},{'revision':null,'url':'/_next/static/chunks/app/not-found-1fd99a98dab3c771.js'},{'revision':null,'url':'/_next/static/chunks/app/offline/page-a7a57ff039afc1af.js'},{'revision':null,'url':'/_next/static/chunks/app/page-70c10ec17b5cf43c.js'},{'revision':null,'url':'/_next/static/chunks/app/pricing/page-3f26af475b3f8452.js'},{'revision':null,'url':'/_next/static/chunks/app/privacy/page-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/profile/page-0a7a9e981d344a6f.js'},{'revision':null,'url':'/_next/static/chunks/app/responsible-gambling/page-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/scan/page-8f1c2629fa6951f2.js'},{'revision':null,'url':'/_next/static/chunks/app/settings/security/page-59df87aff0c62ec8.js'},{'revision':null,'url':'/_next/static/chunks/app/signup/page-33e502bb04569f60.js'},{'revision':null,'url':'/_next/static/chunks/app/soccer/page-9ba3b46b0b1dfa59.js'},{'revision':null,'url':'/_next/static/chunks/app/terms/page-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/app/tracker/page-d6937e1faa494c49.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/cancel/page-1fd99a98dab3c771.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/desk/page-0ed948b44f9c64d7.js'},{'revision':null,'url':'/_next/static/chunks/app/upgrade/success/page-b6baded9f44ae951.js'},{'revision':null,'url':'/_next/static/chunks/app/verify/page-cdeec62ab3a5d8ff.js'},{'revision':null,'url':'/_next/static/chunks/app/welcome/page-e3ab6d5822dee6dc.js'},{'revision':null,'url':'/_next/static/chunks/framework-95da69ac6843d788.js'},{'revision':null,'url':'/_next/static/chunks/main-a354262253293123.js'},{'revision':null,'url':'/_next/static/chunks/main-app-4231d5f500f33972.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/app-error-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/forbidden-4c572a92615aa628.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/global-error-756d787dba63aa4d.js'},{'revision':null,'url':'/_next/static/chunks/next/dist/client/components/builtin/unauthorized-4c572a92615aa628.js'},{'revision':'846118c33b2c0e922d7b3a7676f81f6f','url':'/_next/static/chunks/polyfills-42372ed130431b0a.js'},{'revision':null,'url':'/_next/static/chunks/webpack-a41d593db6ea66d4.js'},{'revision':null,'url':'/_next/static/css/0e3cdd5e3f69836f.css'},{'revision':'d7bc0f63f9618b8b700d028d7d242de8','url':'/apple-touch-icon.png'},{'revision':'5616f81ec5f8a9f726a17fb7ac66a54a','url':'/favicon-16.png'},{'revision':'1fc285cfb0116a0523eb34c779a7d282','url':'/favicon-32.png'},{'revision':'583b034f36839a8af92841771eef38b6','url':'/favicon.ico'},{'revision':'b15795128b3691e5703d0ce14ce10827','url':'/favicon.png'},{'revision':'44e9d71551be13574ac2e10063d03aa9','url':'/favicon.svg'},{'revision':'252b8f8c4d0f64c7cd120f182a6c74ab','url':'/icon-192.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icon-512.png'},{'revision':'252b8f8c4d0f64c7cd120f182a6c74ab','url':'/icons/icon-192.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icons/icon-512.png'},{'revision':'3e3b53bbd210bd772b36b70165657e7c','url':'/icons/icon-maskable-512.png'},{'revision':'bf670b1fafa1b3127bd50abe86d723b9','url':'/images/player-silhouette.svg'},{'revision':'d31c450ad857fff6798872411d72f42b','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("install",e=>{e.waitUntil(caches.open("offline-fallback").then(e=>e.add(eR)).catch(()=>{}))}),self.addEventListener("activate",e=>{e.waitUntil(caches.keys().then(e=>Promise.all(e.filter(e=>!eE.includes(e)&&!e.startsWith("serwist")).map(e=>(console.log("[SW] deleting stale cache:",e),caches.delete(e))))))}),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:i="/",tag:n="vyndr-notification"}=t;e.waitUntil(self.registration.showNotification(a,{body:s,icon:r,badge:"/icons/icon-192.png",tag:n,data:{url:i}}))}),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/content/[...path]/route.ts b/web/src/app/api/content/[...path]/route.ts new file mode 100644 index 0000000..8255d55 --- /dev/null +++ b/web/src/app/api/content/[...path]/route.ts @@ -0,0 +1,25 @@ +import { NextRequest, NextResponse } from 'next/server'; + +export const dynamic = 'force-dynamic'; + +const BACKEND_URL = process.env.BACKEND_URL || 'http://localhost:3000'; + +/** + * Content proxy (Session 29). Forwards /api/content/* to Express + * (slate thread / POTD / recap / matchup preview). Read-only, zero-credit. + */ +export async function GET(req: NextRequest, { params }: { params: Promise<{ path: string[] }> }) { + const { path } = await params; + const segments = (path || []).map(encodeURIComponent).join('/'); + const qs = req.nextUrl.search; + try { + const upstream = await fetch(`${BACKEND_URL}/api/content/${segments}${qs}`, { + method: 'GET', + headers: { Accept: 'application/json' }, + }); + const data = await upstream.json().catch(() => ({})); + return NextResponse.json(data, { status: upstream.status }); + } catch { + return NextResponse.json({ error: 'Content service unreachable.' }, { status: 502 }); + } +}