136 lines
5.6 KiB
JavaScript
136 lines
5.6 KiB
JavaScript
// Unit: rosterLogs loader (Session 23). Redis-only; no network.
|
|
|
|
const store = {};
|
|
const mockScan = jest.fn();
|
|
jest.mock('../../src/utils/redis', () => ({
|
|
cacheGet: jest.fn(async (k) => (k in store ? store[k] : null)),
|
|
getRedisClient: () => ({ scan: mockScan }),
|
|
isDegraded: () => false,
|
|
}));
|
|
|
|
const { loadRosterLogs, __internals } = require('../../src/services/rosterLogs');
|
|
|
|
beforeEach(() => {
|
|
for (const k of Object.keys(store)) delete store[k];
|
|
mockScan.mockReset();
|
|
});
|
|
|
|
describe('rosterLogs', () => {
|
|
test('fast path: returns a prefetched roster blob without scanning', async () => {
|
|
store['rosterlogs:nba'] = [{ name: 'Wemby', games: [{ points: 30 }] }];
|
|
const roster = await loadRosterLogs('nba');
|
|
expect(roster).toHaveLength(1);
|
|
expect(mockScan).not.toHaveBeenCalled();
|
|
});
|
|
|
|
test('scan path: assembles roster from per-player gamelogs keys', async () => {
|
|
store['gamelogs:nba:Wembanyama:20'] = [{ points: 30, team: 'SA', playerId: 'W1' }];
|
|
store['gamelogs:nba:Brunson:20'] = [{ points: 24, team: 'NYK' }];
|
|
mockScan
|
|
.mockResolvedValueOnce(['0', ['gamelogs:nba:Wembanyama:20', 'gamelogs:nba:Brunson:20']]);
|
|
const roster = await loadRosterLogs('nba');
|
|
expect(roster.map((p) => p.name).sort()).toEqual(['Brunson', 'Wembanyama']);
|
|
const wemby = roster.find((p) => p.name === 'Wembanyama');
|
|
expect(wemby.team).toBe('SA');
|
|
expect(wemby.playerId).toBe('W1');
|
|
});
|
|
|
|
test('dedupes by highest game-count variant', async () => {
|
|
store['gamelogs:nba:Star:10'] = [{ points: 1 }, { points: 2 }];
|
|
store['gamelogs:nba:Star:20'] = [{ points: 1 }, { points: 2 }, { points: 3 }];
|
|
mockScan.mockResolvedValueOnce(['0', ['gamelogs:nba:Star:10', 'gamelogs:nba:Star:20']]);
|
|
const roster = await loadRosterLogs('nba');
|
|
expect(roster).toHaveLength(1);
|
|
expect(roster[0].games).toHaveLength(3); // the :20 variant wins
|
|
});
|
|
|
|
test('empty cache → empty roster, never throws', async () => {
|
|
mockScan.mockResolvedValueOnce(['0', []]);
|
|
expect(await loadRosterLogs('nba')).toEqual([]);
|
|
});
|
|
|
|
test('scan failure → returns what it had, no throw', async () => {
|
|
mockScan.mockRejectedValueOnce(new Error('redis down'));
|
|
expect(await loadRosterLogs('nba')).toEqual([]);
|
|
});
|
|
|
|
test('playerFromKey parses the player name out of the key', () => {
|
|
expect(__internals.playerFromKey('gamelogs:nba:LeBron James:20', 'nba')).toBe('LeBron James');
|
|
});
|
|
});
|
|
|
|
// Session 25 — box-score aggregation bridge (prefetch key alignment).
|
|
describe('rosterLogs — Tank01 box-score aggregation', () => {
|
|
// Route scans by their MATCH pattern (3rd arg) so gamelogs + boxscore
|
|
// scans can return different key sets.
|
|
function routeScan(map) {
|
|
mockScan.mockImplementation(async (_cursor, _m, match) => ['0', map[match] || []]);
|
|
}
|
|
|
|
test('aggregates MLB box scores into per-player multi-game logs', async () => {
|
|
routeScan({
|
|
'gamelogs:mlb:*': [],
|
|
'tank01:mlb:boxscore:*': [
|
|
'tank01:mlb:boxscore:20260612_ARI@CIN',
|
|
'tank01:mlb:boxscore:20260611_ARI@LAD',
|
|
],
|
|
});
|
|
// Newer game (0612) and older game (0611) for the same batter.
|
|
store['tank01:mlb:boxscore:20260612_ARI@CIN'] = [
|
|
{ role: 'batter', name: 'Acuna', playerId: 'B1', team: 'ARI', _raw: { H: 2, HR: 1 }, _final: true },
|
|
];
|
|
store['tank01:mlb:boxscore:20260611_ARI@LAD'] = [
|
|
{ role: 'batter', name: 'Acuna', playerId: 'B1', team: 'ARI', _raw: { H: 1, HR: 0 }, _final: true },
|
|
];
|
|
const roster = await loadRosterLogs('mlb');
|
|
expect(roster).toHaveLength(1);
|
|
const acuna = roster[0];
|
|
expect(acuna.name).toBe('Acuna');
|
|
expect(acuna.team).toBe('ARI');
|
|
expect(acuna.games).toHaveLength(2);
|
|
// Most-recent first → the 2-hit game leads (flattened from _raw).
|
|
expect(acuna.games[0].H).toBe(2);
|
|
expect(acuna.games[1].H).toBe(1);
|
|
});
|
|
|
|
test('aggregated MLB logs feed the streaks engine (hit streak)', async () => {
|
|
routeScan({
|
|
'gamelogs:mlb:*': [],
|
|
'tank01:mlb:boxscore:*': ['tank01:mlb:boxscore:20260612_A@B', 'tank01:mlb:boxscore:20260611_A@B'],
|
|
});
|
|
store['tank01:mlb:boxscore:20260612_A@B'] = [{ role: 'batter', name: 'X', team: 'A', _raw: { H: 2 }, _final: true }];
|
|
store['tank01:mlb:boxscore:20260611_A@B'] = [{ role: 'batter', name: 'X', team: 'A', _raw: { H: 1 }, _final: true }];
|
|
const roster = await loadRosterLogs('mlb');
|
|
const { computeStreaks } = require('../../src/services/streaksService');
|
|
const streaks = computeStreaks(roster, 'mlb');
|
|
const hit = streaks.find((s) => s.type === 'hit_streak');
|
|
expect(hit).toBeDefined();
|
|
expect(hit.currentStreak).toBe(2);
|
|
});
|
|
|
|
test('NBA box-score rows are consumed without _raw flattening', async () => {
|
|
routeScan({
|
|
'gamelogs:nba:*': [],
|
|
'tank01:nba:boxscore:*': ['tank01:nba:boxscore:20260612_NYK@SA'],
|
|
});
|
|
store['tank01:nba:boxscore:20260612_NYK@SA'] = [
|
|
{ playerId: 'W1', name: 'Wemby', team: 'SA', pts: 30, reb: 12, ast: 3, threes: 2, blk: 4, stl: 1 },
|
|
];
|
|
const roster = await loadRosterLogs('nba');
|
|
expect(roster[0].games[0].pts).toBe(30);
|
|
});
|
|
|
|
test('gamelogs path still wins over box scores when present', async () => {
|
|
routeScan({ 'gamelogs:mlb:*': ['gamelogs:mlb:Star:20'] });
|
|
store['gamelogs:mlb:Star:20'] = [{ hits: 3 }, { hits: 2 }];
|
|
const roster = await loadRosterLogs('mlb');
|
|
expect(roster[0].name).toBe('Star');
|
|
expect(roster[0].games).toHaveLength(2);
|
|
});
|
|
|
|
test('boxScoreKeyDate extracts the date for recency ordering', () => {
|
|
expect(__internals.boxScoreKeyDate('tank01:mlb:boxscore:20260612_ARI@CIN')).toBe('20260612');
|
|
expect(__internals.boxScoreKeyDate('tank01:mlb:boxscore:weird')).toBe('0');
|
|
});
|
|
});
|