reviewflow 3.29.0 → 3.31.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +24 -0
- package/dist/dashboard/index.html +54 -26
- package/dist/dashboard/modules/emberAvatar.d.ts +134 -0
- package/dist/dashboard/modules/emberAvatar.d.ts.map +1 -0
- package/dist/dashboard/modules/emberAvatar.js +192 -0
- package/dist/dashboard/modules/emberAvatar.js.map +1 -0
- package/dist/dashboard/modules/emberAvatarRenderer.d.ts +42 -0
- package/dist/dashboard/modules/emberAvatarRenderer.d.ts.map +1 -0
- package/dist/dashboard/modules/emberAvatarRenderer.js +119 -0
- package/dist/dashboard/modules/emberAvatarRenderer.js.map +1 -0
- package/dist/dashboard/styles.css +183 -8
- package/dist/frameworks/claude/claudeInvoker.d.ts +7 -0
- package/dist/frameworks/claude/claudeInvoker.d.ts.map +1 -1
- package/dist/frameworks/claude/claudeInvoker.js +1 -1
- package/dist/frameworks/claude/claudeInvoker.js.map +1 -1
- package/dist/main/routes.d.ts.map +1 -1
- package/dist/main/routes.js +6 -16
- package/dist/main/routes.js.map +1 -1
- package/dist/modules/claude-invocation/entities/claudeSession/claudeSession.gateway.d.ts +1 -1
- package/dist/modules/claude-invocation/entities/claudeSession/claudeSession.gateway.d.ts.map +1 -1
- package/dist/modules/claude-invocation/entities/claudeSession/claudeSession.guard.d.ts +1 -1
- package/dist/modules/claude-invocation/entities/claudeSession/claudeSession.schema.d.ts +2 -0
- package/dist/modules/claude-invocation/entities/claudeSession/claudeSession.schema.d.ts.map +1 -1
- package/dist/modules/claude-invocation/entities/claudeSession/claudeSession.schema.js +1 -1
- package/dist/modules/claude-invocation/entities/claudeSession/claudeSession.schema.js.map +1 -1
- package/dist/modules/ember-chat/entities/emberAnswer/emberAnswerTransport.gateway.d.ts +24 -0
- package/dist/modules/ember-chat/entities/emberAnswer/emberAnswerTransport.gateway.d.ts.map +1 -0
- package/dist/modules/ember-chat/entities/emberAnswer/emberAnswerTransport.gateway.js +2 -0
- package/dist/modules/ember-chat/entities/emberAnswer/emberAnswerTransport.gateway.js.map +1 -0
- package/dist/modules/ember-chat/interface-adapters/controllers/http/emberChat.routes.d.ts +2 -3
- package/dist/modules/ember-chat/interface-adapters/controllers/http/emberChat.routes.d.ts.map +1 -1
- package/dist/modules/ember-chat/interface-adapters/controllers/http/emberChat.routes.js +6 -4
- package/dist/modules/ember-chat/interface-adapters/controllers/http/emberChat.routes.js.map +1 -1
- package/dist/modules/ember-chat/interface-adapters/gateways/emberAnswerTransport.claude.gateway.d.ts +46 -0
- package/dist/modules/ember-chat/interface-adapters/gateways/emberAnswerTransport.claude.gateway.d.ts.map +1 -0
- package/dist/modules/ember-chat/interface-adapters/gateways/emberAnswerTransport.claude.gateway.js +176 -0
- package/dist/modules/ember-chat/interface-adapters/gateways/emberAnswerTransport.claude.gateway.js.map +1 -0
- package/dist/modules/ember-chat/interface-adapters/gateways/emberStreamJson.parser.d.ts +2 -1
- package/dist/modules/ember-chat/interface-adapters/gateways/emberStreamJson.parser.d.ts.map +1 -1
- package/dist/modules/ember-chat/interface-adapters/gateways/emberStreamJson.parser.js +8 -2
- package/dist/modules/ember-chat/interface-adapters/gateways/emberStreamJson.parser.js.map +1 -1
- package/dist/modules/ember-chat/interface-adapters/presenters/emberStatus.presenter.d.ts +1 -1
- package/dist/modules/ember-chat/interface-adapters/presenters/emberStatus.presenter.d.ts.map +1 -1
- package/dist/modules/ember-chat/services/emberSystemPrompt.d.ts.map +1 -1
- package/dist/modules/ember-chat/services/emberSystemPrompt.js +47 -4
- package/dist/modules/ember-chat/services/emberSystemPrompt.js.map +1 -1
- package/dist/modules/ember-chat/usecases/askEmber/askEmber.usecase.d.ts +5 -4
- package/dist/modules/ember-chat/usecases/askEmber/askEmber.usecase.d.ts.map +1 -1
- package/dist/modules/ember-chat/usecases/askEmber/askEmber.usecase.js +61 -9
- package/dist/modules/ember-chat/usecases/askEmber/askEmber.usecase.js.map +1 -1
- package/dist/modules/ember-chat/usecases/askEmber/emberStream.d.ts +8 -0
- package/dist/modules/ember-chat/usecases/askEmber/emberStream.d.ts.map +1 -0
- package/dist/modules/ember-chat/usecases/askEmber/emberStream.js +2 -0
- package/dist/modules/ember-chat/usecases/askEmber/emberStream.js.map +1 -0
- package/dist/tests/acceptance/190-ember-live-answers-subscription.acceptance.test.d.ts +2 -0
- package/dist/tests/acceptance/190-ember-live-answers-subscription.acceptance.test.d.ts.map +1 -0
- package/dist/tests/acceptance/190-ember-live-answers-subscription.acceptance.test.js +158 -0
- package/dist/tests/acceptance/190-ember-live-answers-subscription.acceptance.test.js.map +1 -0
- package/dist/tests/stubs/emberAnswerTransport.stub.d.ts +21 -0
- package/dist/tests/stubs/emberAnswerTransport.stub.d.ts.map +1 -0
- package/dist/tests/stubs/emberAnswerTransport.stub.js +54 -0
- package/dist/tests/stubs/emberAnswerTransport.stub.js.map +1 -0
- package/dist/tests/units/dashboard/modules/emberAvatar.test.d.ts +2 -0
- package/dist/tests/units/dashboard/modules/emberAvatar.test.d.ts.map +1 -0
- package/dist/tests/units/dashboard/modules/emberAvatar.test.js +74 -0
- package/dist/tests/units/dashboard/modules/emberAvatar.test.js.map +1 -0
- package/dist/tests/units/dashboard/modules/emberAvatarRenderer.test.d.ts +2 -0
- package/dist/tests/units/dashboard/modules/emberAvatarRenderer.test.d.ts.map +1 -0
- package/dist/tests/units/dashboard/modules/emberAvatarRenderer.test.js +44 -0
- package/dist/tests/units/dashboard/modules/emberAvatarRenderer.test.js.map +1 -0
- package/dist/tests/units/modules/ember-chat/controllers/emberChat.routes.test.js +21 -14
- package/dist/tests/units/modules/ember-chat/controllers/emberChat.routes.test.js.map +1 -1
- package/dist/tests/units/modules/ember-chat/gateways/emberStreamJson.parser.test.js +13 -0
- package/dist/tests/units/modules/ember-chat/gateways/emberStreamJson.parser.test.js.map +1 -1
- package/dist/tests/units/modules/ember-chat/services/emberSystemPrompt.test.js +19 -0
- package/dist/tests/units/modules/ember-chat/services/emberSystemPrompt.test.js.map +1 -1
- package/dist/tests/units/modules/ember-chat/usecases/askEmber.usecase.test.js +53 -20
- package/dist/tests/units/modules/ember-chat/usecases/askEmber.usecase.test.js.map +1 -1
- package/package.json +1 -1
- package/dist/modules/ember-chat/entities/emberSession/emberSession.schema.d.ts +0 -7
- package/dist/modules/ember-chat/entities/emberSession/emberSession.schema.d.ts.map +0 -1
- package/dist/modules/ember-chat/entities/emberSession/emberSession.schema.js +0 -3
- package/dist/modules/ember-chat/entities/emberSession/emberSession.schema.js.map +0 -1
- package/dist/modules/ember-chat/entities/emberSession/emberSessionState.d.ts +0 -13
- package/dist/modules/ember-chat/entities/emberSession/emberSessionState.d.ts.map +0 -1
- package/dist/modules/ember-chat/entities/emberSession/emberSessionState.js +0 -35
- package/dist/modules/ember-chat/entities/emberSession/emberSessionState.js.map +0 -1
- package/dist/modules/ember-chat/entities/emberSession/emberSessionTransport.gateway.d.ts +0 -26
- package/dist/modules/ember-chat/entities/emberSession/emberSessionTransport.gateway.d.ts.map +0 -1
- package/dist/modules/ember-chat/entities/emberSession/emberSessionTransport.gateway.js +0 -2
- package/dist/modules/ember-chat/entities/emberSession/emberSessionTransport.gateway.js.map +0 -1
- package/dist/modules/ember-chat/interface-adapters/gateways/emberSessionTransport.claude.gateway.d.ts +0 -13
- package/dist/modules/ember-chat/interface-adapters/gateways/emberSessionTransport.claude.gateway.d.ts.map +0 -1
- package/dist/modules/ember-chat/interface-adapters/gateways/emberSessionTransport.claude.gateway.js +0 -130
- package/dist/modules/ember-chat/interface-adapters/gateways/emberSessionTransport.claude.gateway.js.map +0 -1
- package/dist/modules/ember-chat/usecases/emberSession/emberSessionRegistry.d.ts +0 -36
- package/dist/modules/ember-chat/usecases/emberSession/emberSessionRegistry.d.ts.map +0 -1
- package/dist/modules/ember-chat/usecases/emberSession/emberSessionRegistry.js +0 -59
- package/dist/modules/ember-chat/usecases/emberSession/emberSessionRegistry.js.map +0 -1
- package/dist/tests/acceptance/189-ember-readonly-review-chat.acceptance.test.d.ts +0 -2
- package/dist/tests/acceptance/189-ember-readonly-review-chat.acceptance.test.d.ts.map +0 -1
- package/dist/tests/acceptance/189-ember-readonly-review-chat.acceptance.test.js +0 -124
- package/dist/tests/acceptance/189-ember-readonly-review-chat.acceptance.test.js.map +0 -1
- package/dist/tests/stubs/emberSessionTransport.stub.d.ts +0 -18
- package/dist/tests/stubs/emberSessionTransport.stub.d.ts.map +0 -1
- package/dist/tests/stubs/emberSessionTransport.stub.js +0 -75
- package/dist/tests/stubs/emberSessionTransport.stub.js.map +0 -1
- package/dist/tests/units/modules/ember-chat/entities/emberSessionState.test.d.ts +0 -2
- package/dist/tests/units/modules/ember-chat/entities/emberSessionState.test.d.ts.map +0 -1
- package/dist/tests/units/modules/ember-chat/entities/emberSessionState.test.js +0 -42
- package/dist/tests/units/modules/ember-chat/entities/emberSessionState.test.js.map +0 -1
- package/dist/tests/units/modules/ember-chat/usecases/emberSessionRegistry.test.d.ts +0 -2
- package/dist/tests/units/modules/ember-chat/usecases/emberSessionRegistry.test.d.ts.map +0 -1
- package/dist/tests/units/modules/ember-chat/usecases/emberSessionRegistry.test.js +0 -88
- package/dist/tests/units/modules/ember-chat/usecases/emberSessionRegistry.test.js.map +0 -1
|
@@ -1,23 +1,75 @@
|
|
|
1
1
|
import { buildEmberSystemPrompt } from '../../../../modules/ember-chat/services/emberSystemPrompt.js';
|
|
2
|
+
class AnswerRelay {
|
|
3
|
+
target = null;
|
|
4
|
+
bufferedChunks = [];
|
|
5
|
+
terminal = null;
|
|
6
|
+
attach(subscriber) {
|
|
7
|
+
this.target = subscriber;
|
|
8
|
+
subscriber.onStatus('working');
|
|
9
|
+
for (const chunk of this.bufferedChunks) {
|
|
10
|
+
subscriber.onChunk(chunk);
|
|
11
|
+
}
|
|
12
|
+
this.bufferedChunks.length = 0;
|
|
13
|
+
if (this.terminal !== null) {
|
|
14
|
+
this.flushTerminal(this.terminal);
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
onChunk(text) {
|
|
18
|
+
if (this.target === null) {
|
|
19
|
+
this.bufferedChunks.push(text);
|
|
20
|
+
return;
|
|
21
|
+
}
|
|
22
|
+
this.target.onChunk(text);
|
|
23
|
+
}
|
|
24
|
+
onDone() {
|
|
25
|
+
const terminal = { kind: 'done' };
|
|
26
|
+
if (this.target === null) {
|
|
27
|
+
this.terminal = terminal;
|
|
28
|
+
return;
|
|
29
|
+
}
|
|
30
|
+
this.flushTerminal(terminal);
|
|
31
|
+
}
|
|
32
|
+
onError(message) {
|
|
33
|
+
const terminal = { kind: 'error', message };
|
|
34
|
+
if (this.target === null) {
|
|
35
|
+
this.terminal = terminal;
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
this.flushTerminal(terminal);
|
|
39
|
+
}
|
|
40
|
+
flushTerminal(terminal) {
|
|
41
|
+
if (this.target === null) {
|
|
42
|
+
return;
|
|
43
|
+
}
|
|
44
|
+
if (terminal.kind === 'done') {
|
|
45
|
+
this.target.onStatus('idle');
|
|
46
|
+
this.target.onDone();
|
|
47
|
+
return;
|
|
48
|
+
}
|
|
49
|
+
this.target.onStatus('error');
|
|
50
|
+
this.target.onError(terminal.message);
|
|
51
|
+
}
|
|
52
|
+
}
|
|
2
53
|
export async function askEmber(message, dependencies) {
|
|
3
54
|
if (dependencies.environment.hasAnthropicApiKey()) {
|
|
4
55
|
return { status: 'billing-regression-prevented' };
|
|
5
56
|
}
|
|
6
|
-
const { readData, projectPath } = dependencies;
|
|
57
|
+
const { readData, projectPath, transport } = dependencies;
|
|
7
58
|
const systemPrompt = buildEmberSystemPrompt({
|
|
8
59
|
reviewScores: await readData.reviewScores(projectPath),
|
|
9
60
|
insights: await readData.insights(projectPath),
|
|
10
61
|
jobHistory: await readData.jobHistory(projectPath),
|
|
11
62
|
worktrees: await readData.worktrees(),
|
|
12
63
|
});
|
|
13
|
-
const
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
});
|
|
18
|
-
if (result.status === 'unavailable') {
|
|
19
|
-
return { status: 'unavailable', reason: result.reason };
|
|
64
|
+
const relay = new AnswerRelay();
|
|
65
|
+
const started = transport.start({ question: message.question, systemPrompt, projectPath }, relay);
|
|
66
|
+
if (started.status === 'failed') {
|
|
67
|
+
return { status: 'unavailable', reason: started.reason };
|
|
20
68
|
}
|
|
21
|
-
return {
|
|
69
|
+
return {
|
|
70
|
+
status: 'streaming',
|
|
71
|
+
subscribe: (subscriber) => relay.attach(subscriber),
|
|
72
|
+
cancel: () => started.run.cancel(),
|
|
73
|
+
};
|
|
22
74
|
}
|
|
23
75
|
//# sourceMappingURL=askEmber.usecase.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"askEmber.usecase.js","sourceRoot":"","sources":["../../../../../src/modules/ember-chat/usecases/askEmber/askEmber.usecase.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"askEmber.usecase.js","sourceRoot":"","sources":["../../../../../src/modules/ember-chat/usecases/askEmber/askEmber.usecase.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
export type EmberStatus = 'working' | 'idle' | 'error';
|
|
2
|
+
export interface EmberStreamSubscriber {
|
|
3
|
+
onStatus: (status: EmberStatus) => void;
|
|
4
|
+
onChunk: (text: string) => void;
|
|
5
|
+
onDone: () => void;
|
|
6
|
+
onError: (message: string) => void;
|
|
7
|
+
}
|
|
8
|
+
//# sourceMappingURL=emberStream.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"emberStream.d.ts","sourceRoot":"","sources":["../../../../../src/modules/ember-chat/usecases/askEmber/emberStream.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,WAAW,GAAG,SAAS,GAAG,MAAM,GAAG,OAAO,CAAC;AAEvD,MAAM,WAAW,qBAAqB;IACpC,QAAQ,EAAE,CAAC,MAAM,EAAE,WAAW,KAAK,IAAI,CAAC;IACxC,OAAO,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,CAAC;IAChC,MAAM,EAAE,MAAM,IAAI,CAAC;IACnB,OAAO,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,IAAI,CAAC;CACpC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"emberStream.js","sourceRoot":"","sources":["../../../../../src/modules/ember-chat/usecases/askEmber/emberStream.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"190-ember-live-answers-subscription.acceptance.test.d.ts","sourceRoot":"","sources":["../../../src/tests/acceptance/190-ember-live-answers-subscription.acceptance.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,158 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { ProjectStatsFactory, ReviewStatsFactory } from '../../tests/factories/projectStats.factory.js';
|
|
3
|
+
import { StubEnvironmentGateway } from '../../tests/stubs/environment.stub.js';
|
|
4
|
+
import { StubEmberReadDataGateway } from '../../tests/stubs/emberReadData.stub.js';
|
|
5
|
+
import { StubEmberAnswerTransportGateway } from '../../tests/stubs/emberAnswerTransport.stub.js';
|
|
6
|
+
import { askEmber } from '../../modules/ember-chat/usecases/askEmber/askEmber.usecase.js';
|
|
7
|
+
import { buildEmberSystemPrompt } from '../../modules/ember-chat/services/emberSystemPrompt.js';
|
|
8
|
+
const PROJECT_PATH = '/projects/alpha';
|
|
9
|
+
function reviewData() {
|
|
10
|
+
const project = ProjectStatsFactory.withReviews([
|
|
11
|
+
ReviewStatsFactory.create({ mrNumber: 42, score: 3, blocking: 4, warnings: 1 }),
|
|
12
|
+
ReviewStatsFactory.create({ mrNumber: 43, score: 4, blocking: 2, warnings: 0 }),
|
|
13
|
+
]);
|
|
14
|
+
const readData = new StubEmberReadDataGateway();
|
|
15
|
+
readData.setReviewScores(PROJECT_PATH, project);
|
|
16
|
+
return readData;
|
|
17
|
+
}
|
|
18
|
+
function largeReviewData() {
|
|
19
|
+
const reviews = Array.from({ length: 500 }, (_, index) => ReviewStatsFactory.create({ mrNumber: index, score: index % 10, blocking: index % 3 }));
|
|
20
|
+
const readData = new StubEmberReadDataGateway();
|
|
21
|
+
readData.setReviewScores(PROJECT_PATH, ProjectStatsFactory.withReviews(reviews));
|
|
22
|
+
return readData;
|
|
23
|
+
}
|
|
24
|
+
function environment(hasApiKey) {
|
|
25
|
+
const stub = new StubEnvironmentGateway();
|
|
26
|
+
stub.setHasAnthropicApiKey(hasApiKey);
|
|
27
|
+
return stub;
|
|
28
|
+
}
|
|
29
|
+
function collectStream(subscribe) {
|
|
30
|
+
return new Promise((resolve) => {
|
|
31
|
+
let answer = '';
|
|
32
|
+
const statuses = [];
|
|
33
|
+
subscribe({
|
|
34
|
+
onStatus: (state) => {
|
|
35
|
+
statuses.push(state);
|
|
36
|
+
},
|
|
37
|
+
onChunk: (text) => {
|
|
38
|
+
answer += text;
|
|
39
|
+
},
|
|
40
|
+
onDone: () => resolve({ answer, statuses }),
|
|
41
|
+
onError: () => resolve({ answer, statuses }),
|
|
42
|
+
});
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
describe('Answer Ember questions live via the Claude subscription (acceptance, SPEC-190)', () => {
|
|
46
|
+
describe('each answer is grounded on the current project review data and streams progressively', () => {
|
|
47
|
+
it('nominal: streams a grounded answer ending with an idle status', async () => {
|
|
48
|
+
const transport = new StubEmberAnswerTransportGateway();
|
|
49
|
+
transport.answerFromSystemPrompt();
|
|
50
|
+
const result = await askEmber({ question: 'Quelles reviews sont en cours ?' }, {
|
|
51
|
+
transport,
|
|
52
|
+
environment: environment(false),
|
|
53
|
+
readData: reviewData(),
|
|
54
|
+
projectPath: PROJECT_PATH,
|
|
55
|
+
});
|
|
56
|
+
expect(result.status).toBe('streaming');
|
|
57
|
+
if (result.status !== 'streaming') {
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
const { answer, statuses } = await collectStream(result.subscribe);
|
|
61
|
+
expect(answer).toContain('42');
|
|
62
|
+
expect(statuses[0]).toBe('working');
|
|
63
|
+
expect(statuses.at(-1)).toBe('idle');
|
|
64
|
+
expect(transport.startCount).toBe(1);
|
|
65
|
+
});
|
|
66
|
+
});
|
|
67
|
+
describe('grounding must succeed regardless of project size', () => {
|
|
68
|
+
it('large grounding: bounds the prompt and never fails on a huge history', async () => {
|
|
69
|
+
const transport = new StubEmberAnswerTransportGateway();
|
|
70
|
+
transport.answerFromSystemPrompt();
|
|
71
|
+
const result = await askEmber({ question: 'Résume mon historique' }, {
|
|
72
|
+
transport,
|
|
73
|
+
environment: environment(false),
|
|
74
|
+
readData: largeReviewData(),
|
|
75
|
+
projectPath: PROJECT_PATH,
|
|
76
|
+
});
|
|
77
|
+
expect(result.status).toBe('streaming');
|
|
78
|
+
if (result.status !== 'streaming') {
|
|
79
|
+
return;
|
|
80
|
+
}
|
|
81
|
+
const { answer, statuses } = await collectStream(result.subscribe);
|
|
82
|
+
expect(statuses.at(-1)).toBe('idle');
|
|
83
|
+
expect(answer.length).toBeLessThan(60_000);
|
|
84
|
+
});
|
|
85
|
+
});
|
|
86
|
+
describe('the subscription is the only allowed billing path', () => {
|
|
87
|
+
it('api key present: refuses to answer when an Anthropic API key is set', async () => {
|
|
88
|
+
const transport = new StubEmberAnswerTransportGateway();
|
|
89
|
+
transport.answerFromSystemPrompt();
|
|
90
|
+
const result = await askEmber({ question: 'Statut ?' }, {
|
|
91
|
+
transport,
|
|
92
|
+
environment: environment(true),
|
|
93
|
+
readData: reviewData(),
|
|
94
|
+
projectPath: PROJECT_PATH,
|
|
95
|
+
});
|
|
96
|
+
expect(result.status).toBe('billing-regression-prevented');
|
|
97
|
+
expect(transport.startCount).toBe(0);
|
|
98
|
+
});
|
|
99
|
+
});
|
|
100
|
+
describe('when the subscription is unavailable, Ember reports unavailability', () => {
|
|
101
|
+
it('not logged in: returns unavailable when the dispatch cannot start', async () => {
|
|
102
|
+
const transport = new StubEmberAnswerTransportGateway();
|
|
103
|
+
transport.failStart();
|
|
104
|
+
const result = await askEmber({ question: 'Statut ?' }, {
|
|
105
|
+
transport,
|
|
106
|
+
environment: environment(false),
|
|
107
|
+
readData: reviewData(),
|
|
108
|
+
projectPath: PROJECT_PATH,
|
|
109
|
+
});
|
|
110
|
+
expect(result.status).toBe('unavailable');
|
|
111
|
+
});
|
|
112
|
+
});
|
|
113
|
+
describe('when answering fails part-way through, the user can retry', () => {
|
|
114
|
+
it('mid-stream failure: surfaces an error after partial chunks', async () => {
|
|
115
|
+
const transport = new StubEmberAnswerTransportGateway();
|
|
116
|
+
transport.failMidStream();
|
|
117
|
+
const result = await askEmber({ question: 'Statut ?' }, {
|
|
118
|
+
transport,
|
|
119
|
+
environment: environment(false),
|
|
120
|
+
readData: reviewData(),
|
|
121
|
+
projectPath: PROJECT_PATH,
|
|
122
|
+
});
|
|
123
|
+
expect(result.status).toBe('streaming');
|
|
124
|
+
if (result.status !== 'streaming') {
|
|
125
|
+
return;
|
|
126
|
+
}
|
|
127
|
+
const statuses = [];
|
|
128
|
+
let errored = false;
|
|
129
|
+
await new Promise((resolve) => {
|
|
130
|
+
result.subscribe({
|
|
131
|
+
onStatus: (state) => statuses.push(state),
|
|
132
|
+
onChunk: () => undefined,
|
|
133
|
+
onDone: () => resolve(),
|
|
134
|
+
onError: () => {
|
|
135
|
+
errored = true;
|
|
136
|
+
resolve();
|
|
137
|
+
},
|
|
138
|
+
});
|
|
139
|
+
});
|
|
140
|
+
expect(errored).toBe(true);
|
|
141
|
+
expect(statuses).toContain('error');
|
|
142
|
+
});
|
|
143
|
+
});
|
|
144
|
+
describe('grounding is bounded by buildEmberSystemPrompt as a pure function', () => {
|
|
145
|
+
it('keeps aggregates and recent reviews while capping a huge history', () => {
|
|
146
|
+
const reviews = Array.from({ length: 500 }, (_, index) => ReviewStatsFactory.create({ mrNumber: index, score: index % 10 }));
|
|
147
|
+
const prompt = buildEmberSystemPrompt({
|
|
148
|
+
reviewScores: ProjectStatsFactory.withReviews(reviews),
|
|
149
|
+
insights: null,
|
|
150
|
+
jobHistory: null,
|
|
151
|
+
worktrees: [],
|
|
152
|
+
});
|
|
153
|
+
expect(prompt.length).toBeLessThan(60_000);
|
|
154
|
+
expect(prompt).toContain('totalReviews');
|
|
155
|
+
});
|
|
156
|
+
});
|
|
157
|
+
});
|
|
158
|
+
//# sourceMappingURL=190-ember-live-answers-subscription.acceptance.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"190-ember-live-answers-subscription.acceptance.test.js","sourceRoot":"","sources":["../../../src/tests/acceptance/190-ember-live-answers-subscription.acceptance.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import type { EmberAnswerStartOptions, EmberAnswerStartResult, EmberAnswerSubscriber, EmberAnswerTransportGateway } from '../../modules/ember-chat/entities/emberAnswer/emberAnswerTransport.gateway.js';
|
|
2
|
+
type AnswerBuilder = (question: string, systemPrompt: string) => Promise<string>;
|
|
3
|
+
export declare class StubEmberAnswerTransportGateway implements EmberAnswerTransportGateway {
|
|
4
|
+
startCount: number;
|
|
5
|
+
private shouldFailStart;
|
|
6
|
+
private shouldFailMidStream;
|
|
7
|
+
private answerBuilder;
|
|
8
|
+
failStart(): void;
|
|
9
|
+
failMidStream(): void;
|
|
10
|
+
respondWith(builder: AnswerBuilder): void;
|
|
11
|
+
/**
|
|
12
|
+
* Makes the stub answer with the system prompt it was started with. The grounding
|
|
13
|
+
* data lives in that prompt, so this proves the real path: readData → askEmber →
|
|
14
|
+
* prompt → transport, rather than the stub fabricating an answer of its own.
|
|
15
|
+
*/
|
|
16
|
+
answerFromSystemPrompt(): void;
|
|
17
|
+
start(options: EmberAnswerStartOptions, subscriber: EmberAnswerSubscriber): EmberAnswerStartResult;
|
|
18
|
+
private deliver;
|
|
19
|
+
}
|
|
20
|
+
export {};
|
|
21
|
+
//# sourceMappingURL=emberAnswerTransport.stub.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"emberAnswerTransport.stub.d.ts","sourceRoot":"","sources":["../../../src/tests/stubs/emberAnswerTransport.stub.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,uBAAuB,EACvB,sBAAsB,EACtB,qBAAqB,EACrB,2BAA2B,EAC5B,MAAM,2EAA2E,CAAC;AAEnF,KAAK,aAAa,GAAG,CAAC,QAAQ,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,KAAK,OAAO,CAAC,MAAM,CAAC,CAAC;AAEjF,qBAAa,+BAAgC,YAAW,2BAA2B;IACjF,UAAU,SAAK;IACf,OAAO,CAAC,eAAe,CAAS;IAChC,OAAO,CAAC,mBAAmB,CAAS;IACpC,OAAO,CAAC,aAAa,CAAgE;IAErF,SAAS,IAAI,IAAI;IAIjB,aAAa,IAAI,IAAI;IAIrB,WAAW,CAAC,OAAO,EAAE,aAAa,GAAG,IAAI;IAIzC;;;;OAIG;IACH,sBAAsB,IAAI,IAAI;IAI9B,KAAK,CACH,OAAO,EAAE,uBAAuB,EAChC,UAAU,EAAE,qBAAqB,GAChC,sBAAsB;YASX,OAAO;CAwBtB"}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
export class StubEmberAnswerTransportGateway {
|
|
2
|
+
startCount = 0;
|
|
3
|
+
shouldFailStart = false;
|
|
4
|
+
shouldFailMidStream = false;
|
|
5
|
+
answerBuilder = async (question) => `Réponse à : ${question}`;
|
|
6
|
+
failStart() {
|
|
7
|
+
this.shouldFailStart = true;
|
|
8
|
+
}
|
|
9
|
+
failMidStream() {
|
|
10
|
+
this.shouldFailMidStream = true;
|
|
11
|
+
}
|
|
12
|
+
respondWith(builder) {
|
|
13
|
+
this.answerBuilder = builder;
|
|
14
|
+
}
|
|
15
|
+
/**
|
|
16
|
+
* Makes the stub answer with the system prompt it was started with. The grounding
|
|
17
|
+
* data lives in that prompt, so this proves the real path: readData → askEmber →
|
|
18
|
+
* prompt → transport, rather than the stub fabricating an answer of its own.
|
|
19
|
+
*/
|
|
20
|
+
answerFromSystemPrompt() {
|
|
21
|
+
this.answerBuilder = async (_question, systemPrompt) => systemPrompt;
|
|
22
|
+
}
|
|
23
|
+
start(options, subscriber) {
|
|
24
|
+
if (this.shouldFailStart) {
|
|
25
|
+
return { status: 'failed', reason: 'stub-start-failed' };
|
|
26
|
+
}
|
|
27
|
+
this.startCount += 1;
|
|
28
|
+
void this.deliver(options, subscriber);
|
|
29
|
+
return { status: 'started', run: { cancel: () => undefined } };
|
|
30
|
+
}
|
|
31
|
+
async deliver(options, subscriber) {
|
|
32
|
+
try {
|
|
33
|
+
const answer = await this.answerBuilder(options.question, options.systemPrompt);
|
|
34
|
+
const words = answer.split(' ');
|
|
35
|
+
const emitted = this.shouldFailMidStream
|
|
36
|
+
? Math.ceil(words.length / 2)
|
|
37
|
+
: words.length;
|
|
38
|
+
for (let index = 0; index < emitted; index += 1) {
|
|
39
|
+
const fragment = index === 0 ? words[index] : ` ${words[index]}`;
|
|
40
|
+
subscriber.onChunk(fragment);
|
|
41
|
+
}
|
|
42
|
+
if (this.shouldFailMidStream) {
|
|
43
|
+
subscriber.onError('stub-mid-stream-failed');
|
|
44
|
+
return;
|
|
45
|
+
}
|
|
46
|
+
subscriber.onDone();
|
|
47
|
+
}
|
|
48
|
+
catch (error) {
|
|
49
|
+
const message = error instanceof Error ? error.message : 'stub-answer-failed';
|
|
50
|
+
subscriber.onError(message);
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
//# sourceMappingURL=emberAnswerTransport.stub.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"emberAnswerTransport.stub.js","sourceRoot":"","sources":["../../../src/tests/stubs/emberAnswerTransport.stub.ts"],"names":[],"mappings":"AASA,MAAM,OAAO,+BAA+B;IAC1C,UAAU,GAAG,CAAC,CAAC;IACP,eAAe,GAAG,KAAK,CAAC;IACxB,mBAAmB,GAAG,KAAK,CAAC;IAC5B,aAAa,GAAkB,KAAK,EAAE,QAAQ,EAAE,EAAE,CAAC,eAAe,QAAQ,EAAE,CAAC;IAErF,SAAS;QACP,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;IAC9B,CAAC;IAED,aAAa;QACX,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;IAClC,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,CAAC,aAAa,GAAG,OAAO,CAAC;IAC/B,CAAC;IAED;;;;OAIG;IACH,sBAAsB;QACpB,IAAI,CAAC,aAAa,GAAG,KAAK,EAAE,SAAS,EAAE,YAAY,EAAE,EAAE,CAAC,YAAY,CAAC;IACvE,CAAC;IAED,KAAK,CACH,OAAgC,EAChC,UAAiC;QAEjC,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;YACzB,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,mBAAmB,EAAE,CAAC;QAC3D,CAAC;QACD,IAAI,CAAC,UAAU,IAAI,CAAC,CAAC;QACrB,KAAK,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;QACvC,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,GAAG,EAAE,EAAE,MAAM,EAAE,GAAG,EAAE,CAAC,SAAS,EAAE,EAAE,CAAC;IACjE,CAAC;IAEO,KAAK,CAAC,OAAO,CACnB,OAAgC,EAChC,UAAiC;QAEjC,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,QAAQ,EAAE,OAAO,CAAC,YAAY,CAAC,CAAC;YAChF,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAChC,MAAM,OAAO,GAAG,IAAI,CAAC,mBAAmB;gBACtC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC;gBAC7B,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC;YACjB,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,OAAO,EAAE,KAAK,IAAI,CAAC,EAAE,CAAC;gBAChD,MAAM,QAAQ,GAAG,KAAK,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC;gBACjE,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;YAC/B,CAAC;YACD,IAAI,IAAI,CAAC,mBAAmB,EAAE,CAAC;gBAC7B,UAAU,CAAC,OAAO,CAAC,wBAAwB,CAAC,CAAC;gBAC7C,OAAO;YACT,CAAC;YACD,UAAU,CAAC,MAAM,EAAE,CAAC;QACtB,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,OAAO,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,oBAAoB,CAAC;YAC9E,UAAU,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QAC9B,CAAC;IACH,CAAC;CACF"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"emberAvatar.test.d.ts","sourceRoot":"","sources":["../../../../../src/tests/units/dashboard/modules/emberAvatar.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { EMBER_STATES, emberStateToVisual, buildFlameWireframe, projectFlameVertex, emberRadiusFactor, emberSwayOffset, FLAME_TIP_Y, FLAME_BASE_Y, } from '../../../../dashboard/modules/emberAvatar.js';
|
|
3
|
+
describe('EMBER_STATES', () => {
|
|
4
|
+
it('exposes exactly the three chat-driven states in narrative order', () => {
|
|
5
|
+
expect(EMBER_STATES).toEqual(['idle', 'working', 'error']);
|
|
6
|
+
});
|
|
7
|
+
});
|
|
8
|
+
describe('emberStateToVisual', () => {
|
|
9
|
+
it('makes the working state lean, flicker and glow harder than idle', () => {
|
|
10
|
+
const idle = emberStateToVisual('idle');
|
|
11
|
+
const working = emberStateToVisual('working');
|
|
12
|
+
expect(working.swayAmount).toBeGreaterThan(idle.swayAmount);
|
|
13
|
+
expect(working.flicker).toBeGreaterThan(idle.flicker);
|
|
14
|
+
expect(working.glow).toBeGreaterThan(idle.glow);
|
|
15
|
+
});
|
|
16
|
+
it('paints the error state with a distinct colour token from idle', () => {
|
|
17
|
+
expect(emberStateToVisual('error').color).not.toBe(emberStateToVisual('idle').color);
|
|
18
|
+
});
|
|
19
|
+
});
|
|
20
|
+
describe('buildFlameWireframe', () => {
|
|
21
|
+
it('has one tip vertex plus a rings × meridians body', () => {
|
|
22
|
+
const { vertices } = buildFlameWireframe({ rings: 6, meridians: 8 });
|
|
23
|
+
expect(vertices).toHaveLength(1 + 6 * 8);
|
|
24
|
+
expect(vertices[0]).toEqual([0, FLAME_TIP_Y, 0]);
|
|
25
|
+
for (const vertex of vertices) {
|
|
26
|
+
expect(vertex).toHaveLength(3);
|
|
27
|
+
}
|
|
28
|
+
});
|
|
29
|
+
it('connects the tip, the meridians and the ring loops with edges', () => {
|
|
30
|
+
const { edges } = buildFlameWireframe({ rings: 6, meridians: 8 });
|
|
31
|
+
const tipSpokes = 8;
|
|
32
|
+
const verticalLines = (6 - 1) * 8;
|
|
33
|
+
const ringLoops = 6 * 8;
|
|
34
|
+
expect(edges).toHaveLength(tipSpokes + verticalLines + ringLoops);
|
|
35
|
+
});
|
|
36
|
+
});
|
|
37
|
+
describe('projectFlameVertex', () => {
|
|
38
|
+
const projection = { tilt: 0.12, distance: 4, scale: 50, centerX: 120, centerY: 120 };
|
|
39
|
+
it('is deterministic for the same inputs', () => {
|
|
40
|
+
const vertex = [0.5, 0.2, -0.3];
|
|
41
|
+
expect(projectFlameVertex(vertex, 1.1, 0.2, projection)).toEqual(projectFlameVertex(vertex, 1.1, 0.2, projection));
|
|
42
|
+
});
|
|
43
|
+
it('leans the tip far more than the base for a positive sway', () => {
|
|
44
|
+
const tip = [0, FLAME_TIP_Y, 0];
|
|
45
|
+
const base = [0, FLAME_BASE_Y, 0];
|
|
46
|
+
const tipShift = projectFlameVertex(tip, 0, 0.4, projection).x - projectFlameVertex(tip, 0, 0, projection).x;
|
|
47
|
+
const baseShift = projectFlameVertex(base, 0, 0.4, projection).x - projectFlameVertex(base, 0, 0, projection).x;
|
|
48
|
+
expect(tipShift).toBeGreaterThan(baseShift);
|
|
49
|
+
expect(baseShift).toBeCloseTo(0, 6);
|
|
50
|
+
});
|
|
51
|
+
});
|
|
52
|
+
describe('emberRadiusFactor', () => {
|
|
53
|
+
it('returns the neutral factor of 1 at time 0', () => {
|
|
54
|
+
expect(emberRadiusFactor(emberStateToVisual('idle'), 0)).toBeCloseTo(1, 5);
|
|
55
|
+
});
|
|
56
|
+
it('stays within the flicker envelope', () => {
|
|
57
|
+
const visual = emberStateToVisual('working');
|
|
58
|
+
for (const time of [120, 900, 1750, 5000]) {
|
|
59
|
+
const factor = emberRadiusFactor(visual, time);
|
|
60
|
+
expect(factor).toBeGreaterThanOrEqual(1 - visual.flicker - 1e-6);
|
|
61
|
+
expect(factor).toBeLessThanOrEqual(1 + visual.flicker + 1e-6);
|
|
62
|
+
}
|
|
63
|
+
});
|
|
64
|
+
});
|
|
65
|
+
describe('emberSwayOffset', () => {
|
|
66
|
+
it('is zero at time 0 and bounded by the sway amplitude', () => {
|
|
67
|
+
const visual = emberStateToVisual('working');
|
|
68
|
+
expect(emberSwayOffset(visual, 0)).toBeCloseTo(0, 6);
|
|
69
|
+
for (const time of [200, 1234, 4321]) {
|
|
70
|
+
expect(Math.abs(emberSwayOffset(visual, time))).toBeLessThanOrEqual(visual.swayAmount + 1e-6);
|
|
71
|
+
}
|
|
72
|
+
});
|
|
73
|
+
});
|
|
74
|
+
//# sourceMappingURL=emberAvatar.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"emberAvatar.test.js","sourceRoot":"","sources":["../../../../../src/tests/units/dashboard/modules/emberAvatar.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"emberAvatarRenderer.test.d.ts","sourceRoot":"","sources":["../../../../../src/tests/units/dashboard/modules/emberAvatarRenderer.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { mountEmberAvatar } from '../../../../dashboard/modules/emberAvatarRenderer.js';
|
|
3
|
+
function fakeCanvas() {
|
|
4
|
+
return { width: 240, height: 240, getContext: () => null };
|
|
5
|
+
}
|
|
6
|
+
describe('mountEmberAvatar (lifecycle glue)', () => {
|
|
7
|
+
it('schedules an animation frame on mount', () => {
|
|
8
|
+
const requested = [];
|
|
9
|
+
mountEmberAvatar({
|
|
10
|
+
canvas: fakeCanvas(),
|
|
11
|
+
requestFrame: (callback) => {
|
|
12
|
+
requested.push(callback);
|
|
13
|
+
return requested.length;
|
|
14
|
+
},
|
|
15
|
+
cancelFrame: () => undefined,
|
|
16
|
+
now: () => 0,
|
|
17
|
+
});
|
|
18
|
+
expect(requested).toHaveLength(1);
|
|
19
|
+
});
|
|
20
|
+
it('cancels the scheduled frame on destroy so the loop cannot leak', () => {
|
|
21
|
+
const cancelled = [];
|
|
22
|
+
const controls = mountEmberAvatar({
|
|
23
|
+
canvas: fakeCanvas(),
|
|
24
|
+
requestFrame: () => 42,
|
|
25
|
+
cancelFrame: (handle) => {
|
|
26
|
+
cancelled.push(handle);
|
|
27
|
+
},
|
|
28
|
+
now: () => 0,
|
|
29
|
+
});
|
|
30
|
+
controls.destroy();
|
|
31
|
+
expect(cancelled).toEqual([42]);
|
|
32
|
+
});
|
|
33
|
+
it('exposes setState and destroy controls', () => {
|
|
34
|
+
const controls = mountEmberAvatar({
|
|
35
|
+
canvas: fakeCanvas(),
|
|
36
|
+
requestFrame: () => 1,
|
|
37
|
+
cancelFrame: () => undefined,
|
|
38
|
+
now: () => 0,
|
|
39
|
+
});
|
|
40
|
+
expect(typeof controls.setState).toBe('function');
|
|
41
|
+
expect(typeof controls.destroy).toBe('function');
|
|
42
|
+
});
|
|
43
|
+
});
|
|
44
|
+
//# sourceMappingURL=emberAvatarRenderer.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"emberAvatarRenderer.test.js","sourceRoot":"","sources":["../../../../../src/tests/units/dashboard/modules/emberAvatarRenderer.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAE,gBAAgB,EAAE,MAAM,4CAA4C,CAAC;AAQ9E,SAAS,UAAU;IACjB,OAAO,EAAE,KAAK,EAAE,GAAG,EAAE,MAAM,EAAE,GAAG,EAAE,UAAU,EAAE,GAAG,EAAE,CAAC,IAAI,EAAE,CAAC;AAC7D,CAAC;AAED,QAAQ,CAAC,mCAAmC,EAAE,GAAG,EAAE;IACjD,EAAE,CAAC,uCAAuC,EAAE,GAAG,EAAE;QAC/C,MAAM,SAAS,GAAkC,EAAE,CAAC;QAEpD,gBAAgB,CAAC;YACf,MAAM,EAAE,UAAU,EAAE;YACpB,YAAY,EAAE,CAAC,QAAQ,EAAE,EAAE;gBACzB,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;gBACzB,OAAO,SAAS,CAAC,MAAM,CAAC;YAC1B,CAAC;YACD,WAAW,EAAE,GAAG,EAAE,CAAC,SAAS;YAC5B,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;SACb,CAAC,CAAC;QAEH,MAAM,CAAC,SAAS,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;IACpC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gEAAgE,EAAE,GAAG,EAAE;QACxE,MAAM,SAAS,GAAa,EAAE,CAAC;QAE/B,MAAM,QAAQ,GAAG,gBAAgB,CAAC;YAChC,MAAM,EAAE,UAAU,EAAE;YACpB,YAAY,EAAE,GAAG,EAAE,CAAC,EAAE;YACtB,WAAW,EAAE,CAAC,MAAM,EAAE,EAAE;gBACtB,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YACzB,CAAC;YACD,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;SACb,CAAC,CAAC;QAEH,QAAQ,CAAC,OAAO,EAAE,CAAC;QAEnB,MAAM,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;IAClC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,uCAAuC,EAAE,GAAG,EAAE;QAC/C,MAAM,QAAQ,GAAG,gBAAgB,CAAC;YAChC,MAAM,EAAE,UAAU,EAAE;YACpB,YAAY,EAAE,GAAG,EAAE,CAAC,CAAC;YACrB,WAAW,EAAE,GAAG,EAAE,CAAC,SAAS;YAC5B,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;SACb,CAAC,CAAC;QAEH,MAAM,CAAC,OAAO,QAAQ,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAClD,MAAM,CAAC,OAAO,QAAQ,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IACnD,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -1,8 +1,7 @@
|
|
|
1
1
|
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
2
2
|
import Fastify from 'fastify';
|
|
3
3
|
import { emberChatRoutes } from '../../../../../modules/ember-chat/interface-adapters/controllers/http/emberChat.routes.js';
|
|
4
|
-
import {
|
|
5
|
-
import { StubEmberSessionTransportGateway } from '../../../../../tests/stubs/emberSessionTransport.stub.js';
|
|
4
|
+
import { StubEmberAnswerTransportGateway } from '../../../../../tests/stubs/emberAnswerTransport.stub.js';
|
|
6
5
|
import { StubEmberReadDataGateway } from '../../../../../tests/stubs/emberReadData.stub.js';
|
|
7
6
|
import { StubEnvironmentGateway } from '../../../../../tests/stubs/environment.stub.js';
|
|
8
7
|
import { createStubLogger } from '../../../../../tests/stubs/logger.stub.js';
|
|
@@ -18,22 +17,16 @@ describe('emberChat routes', () => {
|
|
|
18
17
|
let transport;
|
|
19
18
|
let environment;
|
|
20
19
|
function buildApplication() {
|
|
21
|
-
transport = new
|
|
20
|
+
transport = new StubEmberAnswerTransportGateway();
|
|
22
21
|
transport.respondWith(async (question) => `Réponse à ${question}`);
|
|
23
22
|
environment = new StubEnvironmentGateway();
|
|
24
23
|
environment.setHasAnthropicApiKey(false);
|
|
25
|
-
const registry = new EmberSessionRegistry({
|
|
26
|
-
transport,
|
|
27
|
-
now: () => new Date('2026-05-28T10:00:00Z'),
|
|
28
|
-
idleTimeoutMs: 60_000,
|
|
29
|
-
});
|
|
30
24
|
const instance = Fastify();
|
|
31
25
|
void instance.register(emberChatRoutes, {
|
|
32
|
-
|
|
26
|
+
transport,
|
|
33
27
|
environment,
|
|
34
28
|
readData: new StubEmberReadDataGateway(),
|
|
35
29
|
projectPath: PROJECT_PATH,
|
|
36
|
-
now: () => new Date('2026-05-28T10:00:00Z'),
|
|
37
30
|
logger: createStubLogger(),
|
|
38
31
|
});
|
|
39
32
|
return instance;
|
|
@@ -45,14 +38,14 @@ describe('emberChat routes', () => {
|
|
|
45
38
|
afterEach(async () => {
|
|
46
39
|
await application.close();
|
|
47
40
|
});
|
|
48
|
-
it('rejects an empty question with 400 and never
|
|
41
|
+
it('rejects an empty question with 400 and never starts', async () => {
|
|
49
42
|
const response = await application.inject({
|
|
50
43
|
method: 'POST',
|
|
51
44
|
url: '/api/ember/ask',
|
|
52
45
|
payload: { question: '' },
|
|
53
46
|
});
|
|
54
47
|
expect(response.statusCode).toBe(400);
|
|
55
|
-
expect(transport.
|
|
48
|
+
expect(transport.startCount).toBe(0);
|
|
56
49
|
});
|
|
57
50
|
it('streams chunk, status and end events for a valid question', async () => {
|
|
58
51
|
const response = await application.inject({
|
|
@@ -71,7 +64,21 @@ describe('emberChat routes', () => {
|
|
|
71
64
|
});
|
|
72
65
|
it('emits an error event when the assistant is unreachable', async () => {
|
|
73
66
|
application = buildApplication();
|
|
74
|
-
transport.
|
|
67
|
+
transport.failStart();
|
|
68
|
+
await application.ready();
|
|
69
|
+
const response = await application.inject({
|
|
70
|
+
method: 'POST',
|
|
71
|
+
url: '/api/ember/ask',
|
|
72
|
+
payload: { question: 'Quel projet a le pire score ?' },
|
|
73
|
+
});
|
|
74
|
+
const events = parseSseEvents(response.body);
|
|
75
|
+
const errors = events.filter((event) => event.type === 'error');
|
|
76
|
+
expect(errors.length).toBeGreaterThan(0);
|
|
77
|
+
expect(String(errors[0].message)).toContain('INDISPONIBLE');
|
|
78
|
+
});
|
|
79
|
+
it('emits an error event when answering fails mid-stream', async () => {
|
|
80
|
+
application = buildApplication();
|
|
81
|
+
transport.failMidStream();
|
|
75
82
|
await application.ready();
|
|
76
83
|
const response = await application.inject({
|
|
77
84
|
method: 'POST',
|
|
@@ -95,7 +102,7 @@ describe('emberChat routes', () => {
|
|
|
95
102
|
const events = parseSseEvents(response.body);
|
|
96
103
|
const errors = events.filter((event) => event.type === 'error');
|
|
97
104
|
expect(errors.length).toBeGreaterThan(0);
|
|
98
|
-
expect(transport.
|
|
105
|
+
expect(transport.startCount).toBe(0);
|
|
99
106
|
});
|
|
100
107
|
});
|
|
101
108
|
//# sourceMappingURL=emberChat.routes.test.js.map
|