Session 16: Live hero prop, sport-specific markets fix, soccer weather, Sentry CSP (1429 tests)
This commit is contained in:
@@ -0,0 +1,116 @@
|
||||
// Session 16 — soccer weather wiring. The feature extractor fetches
|
||||
// Open-Meteo for outdoor WC venues. Dome venues skip the fetch
|
||||
// (operators close the roof); unknown venues skip silently.
|
||||
|
||||
const mockCache = new Map();
|
||||
jest.mock('../../src/utils/redis', () => ({
|
||||
cacheGet: async (k) => (mockCache.has(k) ? mockCache.get(k) : null),
|
||||
cacheSet: async (k, v) => { mockCache.set(k, v); return true; },
|
||||
cacheDel: async (k) => { mockCache.delete(k); return true; },
|
||||
isDegraded: () => false,
|
||||
}));
|
||||
|
||||
const mockWeather = jest.fn();
|
||||
jest.mock('../../src/services/weatherService', () => ({
|
||||
getWeather: (...a) => mockWeather(...a),
|
||||
}));
|
||||
|
||||
const { extractSoccerFeatures } = require('../../src/services/intelligence/soccerFeatureExtractor');
|
||||
const { normalizeName } = require('../../src/utils/normalize');
|
||||
|
||||
beforeEach(() => {
|
||||
mockCache.clear();
|
||||
mockWeather.mockReset();
|
||||
});
|
||||
|
||||
function primePlayerAndMatch(player, team, opts = {}) {
|
||||
mockCache.set(`soccer:player:${normalizeName(player)}`, {
|
||||
team, goals_per_90: 0.5,
|
||||
});
|
||||
mockCache.set(`soccer:nextmatch:${team}`, {
|
||||
opponent: opts.opponent || 'X',
|
||||
venue: opts.venue || 'MetLife Stadium',
|
||||
isHome: opts.isHome ?? true,
|
||||
referee: opts.referee || null,
|
||||
});
|
||||
}
|
||||
|
||||
describe('soccer weather wiring (Session 16)', () => {
|
||||
test('outdoor WC venue → weather features populated', async () => {
|
||||
primePlayerAndMatch('Harry Kane', 'England', { venue: 'MetLife Stadium' });
|
||||
mockWeather.mockResolvedValueOnce({
|
||||
temp_f: 81.2, wind_mph: 9.4, wind_dir: 220, precip_mm: 0,
|
||||
});
|
||||
|
||||
const r = await extractSoccerFeatures({
|
||||
player: 'Harry Kane', stat_type: 'goals', line: 0.5, direction: 'over',
|
||||
});
|
||||
|
||||
expect(mockWeather).toHaveBeenCalledTimes(1);
|
||||
expect(r.features.weather_temp_f).toBeCloseTo(81.2);
|
||||
expect(r.features.weather_wind_mph).toBeCloseTo(9.4);
|
||||
expect(r.features.weather_wind_dir).toBe(220);
|
||||
expect(r.features.weather_precip_mm).toBe(0);
|
||||
});
|
||||
|
||||
test('dome WC venue (BC Place) → weather fetch skipped, fields null', async () => {
|
||||
primePlayerAndMatch('Sub Player', 'Canada', { venue: 'BC Place' });
|
||||
const r = await extractSoccerFeatures({
|
||||
player: 'Sub Player', stat_type: 'goals', line: 0.5, direction: 'over',
|
||||
});
|
||||
expect(mockWeather).not.toHaveBeenCalled();
|
||||
expect(r.features.weather_temp_f).toBeNull();
|
||||
expect(r.features.weather_wind_mph).toBeNull();
|
||||
});
|
||||
|
||||
test('Estadio Azteca (open-air, high-altitude) → weather fetched + altitude_impact still high', async () => {
|
||||
primePlayerAndMatch('Forward', 'Mexico', { venue: 'Estadio Azteca' });
|
||||
mockWeather.mockResolvedValueOnce({ temp_f: 68, wind_mph: 5, wind_dir: 90, precip_mm: 0 });
|
||||
const r = await extractSoccerFeatures({
|
||||
player: 'Forward', stat_type: 'goals', line: 0.5, direction: 'over',
|
||||
});
|
||||
expect(r.features.weather_temp_f).toBe(68);
|
||||
expect(r.features.altitude_impact).toBe('high');
|
||||
});
|
||||
|
||||
test('venue not in the WC index → weather fetch skipped', async () => {
|
||||
primePlayerAndMatch('X', 'Y', { venue: 'Random Stadium' });
|
||||
const r = await extractSoccerFeatures({
|
||||
player: 'X', stat_type: 'goals', line: 0.5, direction: 'over',
|
||||
});
|
||||
expect(mockWeather).not.toHaveBeenCalled();
|
||||
expect(r.features.weather_temp_f).toBeNull();
|
||||
});
|
||||
|
||||
test('weather service returns null → feature fields stay null (no throw)', async () => {
|
||||
primePlayerAndMatch('X', 'England', { venue: 'MetLife Stadium' });
|
||||
mockWeather.mockResolvedValueOnce(null);
|
||||
const r = await extractSoccerFeatures({
|
||||
player: 'X', stat_type: 'goals', line: 0.5, direction: 'over',
|
||||
});
|
||||
expect(mockWeather).toHaveBeenCalledTimes(1);
|
||||
expect(r.features.weather_temp_f).toBeNull();
|
||||
expect(r.features.weather_wind_mph).toBeNull();
|
||||
});
|
||||
|
||||
test('weather service throws → graceful degrade, grade still produced', async () => {
|
||||
primePlayerAndMatch('X', 'England', { venue: 'MetLife Stadium' });
|
||||
mockWeather.mockRejectedValueOnce(new Error('timeout'));
|
||||
const r = await extractSoccerFeatures({
|
||||
player: 'X', stat_type: 'goals', line: 0.5, direction: 'over',
|
||||
});
|
||||
expect(r.features.weather_temp_f).toBeNull();
|
||||
// Other features (goals_per_90 etc.) still populated.
|
||||
expect(r.features.goals_per_90).toBe(0.5);
|
||||
});
|
||||
|
||||
test('no venue resolved → weather skipped entirely (no fetch attempt)', async () => {
|
||||
mockCache.set(`soccer:player:${normalizeName('Solo')}`, { team: 'England', goals_per_90: 0.5 });
|
||||
// No nextmatch entry → venueName is null.
|
||||
const r = await extractSoccerFeatures({
|
||||
player: 'Solo', stat_type: 'goals', line: 0.5, direction: 'over',
|
||||
});
|
||||
expect(mockWeather).not.toHaveBeenCalled();
|
||||
expect(r.features.weather_temp_f).toBeNull();
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user