205 lines
6.9 KiB
JavaScript
205 lines
6.9 KiB
JavaScript
process.env.ENGINE2_ENABLED = 'true';
|
|
|
|
const mockAnalyze = jest.fn();
|
|
jest.mock('../../src/services/adapters/openRouterAdapter', () => ({
|
|
configured: () => true,
|
|
analyze: (...args) => mockAnalyze(...args),
|
|
getUsage: () => ({ requestsToday: 0, requestsRemaining: 1000 }),
|
|
}));
|
|
|
|
const mockUpdates = [];
|
|
jest.mock('../../src/utils/supabase', () => ({
|
|
getSupabaseServiceClient: () => ({
|
|
from() {
|
|
const proxy = {
|
|
update(patch) {
|
|
mockUpdates.push(patch);
|
|
return {
|
|
eq() { return Promise.resolve({ error: null }); },
|
|
};
|
|
},
|
|
};
|
|
return proxy;
|
|
},
|
|
}),
|
|
}));
|
|
|
|
const engine2 = require('../../src/services/intelligence/engine2');
|
|
|
|
beforeEach(() => {
|
|
engine2.clearQueue();
|
|
mockAnalyze.mockReset();
|
|
mockUpdates.length = 0;
|
|
});
|
|
|
|
function sampleContext(overrides = {}) {
|
|
return {
|
|
player_name: 'Jalen Brunson',
|
|
team: 'NYK',
|
|
sport: 'nba',
|
|
direction: 'over',
|
|
line: 26.5,
|
|
stat_type: 'points',
|
|
home_team: 'NYK',
|
|
away_team: 'BOS',
|
|
game_date: '2026-06-08',
|
|
engine1_grade: 'A-',
|
|
engine1_factors: ['l5_avg', 'opp_rank_stat', 'home_away'],
|
|
features: { l5_avg: 30.4, l20_avg: 26.1, home_away: 1.0 },
|
|
trap: { composite: 0.18, recommendation: 'proceed', signals: {} },
|
|
consistency: { consistency: 'reliable', cv: 0.18, score: 0.7 },
|
|
recentGames: [
|
|
{ date: '2026-06-05', value: 32, opponent: 'BOS', home: true },
|
|
{ date: '2026-06-03', value: 28, opponent: 'MIA', home: false },
|
|
],
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
describe('engine2 — eligibility', () => {
|
|
test('queues A/B tier grades', () => {
|
|
engine2.queueAnalysis('g1', sampleContext({ engine1_grade: 'A-' }));
|
|
engine2.queueAnalysis('g2', sampleContext({ engine1_grade: 'B+' }));
|
|
expect(engine2.getQueueSize()).toBe(2);
|
|
});
|
|
|
|
test('SKIPS C/D/F grades', () => {
|
|
engine2.queueAnalysis('c1', sampleContext({ engine1_grade: 'C' }));
|
|
engine2.queueAnalysis('d1', sampleContext({ engine1_grade: 'D' }));
|
|
engine2.queueAnalysis('f1', sampleContext({ engine1_grade: 'F' }));
|
|
expect(engine2.getQueueSize()).toBe(0);
|
|
});
|
|
|
|
test('respects ENGINE2_ENABLED=false', () => {
|
|
process.env.ENGINE2_ENABLED = 'false';
|
|
jest.resetModules();
|
|
const fresh = require('../../src/services/intelligence/engine2');
|
|
fresh.queueAnalysis('g3', sampleContext({ engine1_grade: 'A' }));
|
|
expect(fresh.getQueueSize()).toBe(0);
|
|
process.env.ENGINE2_ENABLED = 'true';
|
|
});
|
|
});
|
|
|
|
describe('engine2 — prompt construction', () => {
|
|
test('buildPrompt includes player, features, traps, and recent games', () => {
|
|
const prompt = engine2.__internals.buildPrompt(sampleContext());
|
|
expect(prompt).toContain('Jalen Brunson');
|
|
expect(prompt).toContain('PROP: over 26.5 points');
|
|
expect(prompt).toContain('l5_avg');
|
|
expect(prompt).toContain('Trap composite: 0.18');
|
|
expect(prompt).toContain('cv=0.18');
|
|
});
|
|
|
|
test('prompt never contains the literal string "VYNDR"', () => {
|
|
const prompt = engine2.__internals.buildPrompt(sampleContext());
|
|
expect(prompt).not.toMatch(/VYNDR/);
|
|
});
|
|
});
|
|
|
|
describe('engine2 — response parsing', () => {
|
|
test('parseResponse handles raw JSON', () => {
|
|
const r = engine2.__internals.parseResponse('{"grade":"A","confidence":0.7,"narrative":"..."}');
|
|
expect(r.grade).toBe('A');
|
|
});
|
|
|
|
test('parseResponse handles markdown fenced JSON', () => {
|
|
const r = engine2.__internals.parseResponse('Here you go:\n```json\n{"grade":"B"}\n```');
|
|
expect(r.grade).toBe('B');
|
|
});
|
|
|
|
test('parseResponse extracts JSON from chatty preamble', () => {
|
|
const r = engine2.__internals.parseResponse('Sure! Here is the analysis: {"grade":"B+","confidence":0.6}');
|
|
expect(r.grade).toBe('B+');
|
|
});
|
|
|
|
test('parseResponse returns null when no JSON found', () => {
|
|
expect(engine2.__internals.parseResponse('no json here')).toBeNull();
|
|
expect(engine2.__internals.parseResponse(null)).toBeNull();
|
|
});
|
|
});
|
|
|
|
describe('engine2 — validation', () => {
|
|
test('rejects unknown grade values', () => {
|
|
expect(engine2.__internals.validateAnalysis({
|
|
grade: 'AAA', confidence: 0.7, narrative: 'x',
|
|
})).toBeNull();
|
|
});
|
|
|
|
test('rejects out-of-range confidence', () => {
|
|
expect(engine2.__internals.validateAnalysis({
|
|
grade: 'A', confidence: 1.5, narrative: 'x',
|
|
})).toBeNull();
|
|
});
|
|
|
|
test('rejects empty narrative', () => {
|
|
expect(engine2.__internals.validateAnalysis({
|
|
grade: 'A', confidence: 0.7, narrative: '',
|
|
})).toBeNull();
|
|
});
|
|
|
|
test('truncates narrative + concern + key_factor to caps', () => {
|
|
const out = engine2.__internals.validateAnalysis({
|
|
grade: 'A',
|
|
confidence: 0.7,
|
|
narrative: 'x'.repeat(800),
|
|
trap_concern: 'y'.repeat(500),
|
|
key_factor: 'z'.repeat(300),
|
|
agrees_with_engine1: true,
|
|
});
|
|
expect(out.narrative.length).toBe(500);
|
|
expect(out.trap_concern.length).toBe(300);
|
|
expect(out.key_factor.length).toBe(200);
|
|
});
|
|
|
|
test('passes explicit null grade through', () => {
|
|
const out = engine2.__internals.validateAnalysis({ grade: null, reason: 'not enough data' });
|
|
expect(out).toEqual({ grade: null, reason: 'not enough data' });
|
|
});
|
|
});
|
|
|
|
describe('engine2 — processQueue', () => {
|
|
test('processes queued analysis, persists to grade_history', async () => {
|
|
mockAnalyze.mockResolvedValue({
|
|
response: JSON.stringify({
|
|
grade: 'A',
|
|
confidence: 0.78,
|
|
agrees_with_engine1: true,
|
|
narrative: 'Brunson trending up + weak D matchup + rested.',
|
|
trap_concern: null,
|
|
key_factor: 'opp_rank_stat',
|
|
}),
|
|
modelUsed: 'deepseek/deepseek-chat',
|
|
latencyMs: 4200,
|
|
});
|
|
engine2.queueAnalysis('grade-1', sampleContext());
|
|
const summary = await engine2.processQueue();
|
|
expect(summary).toMatchObject({ processed: 1, succeeded: 1, failed: 0 });
|
|
expect(mockUpdates).toHaveLength(1);
|
|
expect(mockUpdates[0].engine2_grade).toBe('A');
|
|
expect(mockUpdates[0].engine2_confidence).toBe(0.78);
|
|
expect(mockUpdates[0].engine2_model).toBe('deepseek/deepseek-chat');
|
|
});
|
|
|
|
test('records failures without crashing', async () => {
|
|
mockAnalyze.mockResolvedValue(null); // adapter unavailable
|
|
engine2.queueAnalysis('grade-2', sampleContext());
|
|
const summary = await engine2.processQueue();
|
|
expect(summary).toMatchObject({ processed: 1, succeeded: 0, failed: 1 });
|
|
});
|
|
|
|
test('respects BATCH_SIZE — leaves remaining items for next call', async () => {
|
|
mockAnalyze.mockResolvedValue({
|
|
response: JSON.stringify({ grade: 'A', confidence: 0.7, agrees_with_engine1: true, narrative: 'x' }),
|
|
modelUsed: 'deepseek/deepseek-chat',
|
|
latencyMs: 100,
|
|
});
|
|
// Queue more than BATCH_SIZE items.
|
|
for (let i = 0; i < engine2.__internals.BATCH_SIZE + 3; i += 1) {
|
|
engine2.queueAnalysis(`g-${i}`, sampleContext());
|
|
}
|
|
const summary = await engine2.processQueue();
|
|
expect(summary.processed).toBe(engine2.__internals.BATCH_SIZE);
|
|
expect(summary.remaining).toBe(3);
|
|
});
|
|
});
|