@kenkaiiii/ggcoder 4.3.204 → 4.3.206
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/dist/cli.js +38 -26
- package/dist/cli.js.map +1 -1
- package/dist/config.js +2 -2
- package/dist/core/compaction/compactor.d.ts.map +1 -1
- package/dist/core/compaction/compactor.js +6 -0
- package/dist/core/compaction/compactor.js.map +1 -1
- package/dist/core/compaction/compactor.test.js +16 -0
- package/dist/core/compaction/compactor.test.js.map +1 -1
- package/dist/core/goal-controller.d.ts +57 -0
- package/dist/core/goal-controller.d.ts.map +1 -0
- package/dist/core/goal-controller.js +285 -0
- package/dist/core/goal-controller.js.map +1 -0
- package/dist/core/goal-controller.test.d.ts +2 -0
- package/dist/core/goal-controller.test.d.ts.map +1 -0
- package/dist/core/goal-controller.test.js +377 -0
- package/dist/core/goal-controller.test.js.map +1 -0
- package/dist/core/goal-lifecycle-smoke.test.d.ts +2 -0
- package/dist/core/goal-lifecycle-smoke.test.d.ts.map +1 -0
- package/dist/core/goal-lifecycle-smoke.test.js +207 -0
- package/dist/core/goal-lifecycle-smoke.test.js.map +1 -0
- package/dist/core/goal-store.d.ts +164 -0
- package/dist/core/goal-store.d.ts.map +1 -0
- package/dist/core/goal-store.js +721 -0
- package/dist/core/goal-store.js.map +1 -0
- package/dist/core/goal-store.test.d.ts +2 -0
- package/dist/core/goal-store.test.d.ts.map +1 -0
- package/dist/core/goal-store.test.js +341 -0
- package/dist/core/goal-store.test.js.map +1 -0
- package/dist/core/goal-verifier.d.ts +17 -0
- package/dist/core/goal-verifier.d.ts.map +1 -0
- package/dist/core/goal-verifier.js +84 -0
- package/dist/core/goal-verifier.js.map +1 -0
- package/dist/core/goal-verifier.test.d.ts +2 -0
- package/dist/core/goal-verifier.test.d.ts.map +1 -0
- package/dist/core/goal-verifier.test.js +88 -0
- package/dist/core/goal-verifier.test.js.map +1 -0
- package/dist/core/goal-worker.d.ts +47 -0
- package/dist/core/goal-worker.d.ts.map +1 -0
- package/dist/core/goal-worker.js +329 -0
- package/dist/core/goal-worker.js.map +1 -0
- package/dist/core/goal-worker.test.d.ts +2 -0
- package/dist/core/goal-worker.test.d.ts.map +1 -0
- package/dist/core/goal-worker.test.js +206 -0
- package/dist/core/goal-worker.test.js.map +1 -0
- package/dist/core/oauth/gemini.d.ts.map +1 -1
- package/dist/core/oauth/gemini.js +138 -30
- package/dist/core/oauth/gemini.js.map +1 -1
- package/dist/core/oauth/gemini.test.d.ts +2 -0
- package/dist/core/oauth/gemini.test.d.ts.map +1 -0
- package/dist/core/oauth/gemini.test.js +154 -0
- package/dist/core/oauth/gemini.test.js.map +1 -0
- package/dist/core/prompt-commands.d.ts.map +1 -1
- package/dist/core/prompt-commands.js +124 -0
- package/dist/core/prompt-commands.js.map +1 -1
- package/dist/core/prompt-commands.test.js +36 -0
- package/dist/core/prompt-commands.test.js.map +1 -1
- package/dist/core/repomap.js +2 -0
- package/dist/core/repomap.js.map +1 -1
- package/dist/core/repomap.test.js +18 -0
- package/dist/core/repomap.test.js.map +1 -1
- package/dist/index.d.ts +1 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +1 -1
- package/dist/index.js.map +1 -1
- package/dist/interactive.d.ts.map +1 -1
- package/dist/interactive.js +20 -11
- package/dist/interactive.js.map +1 -1
- package/dist/system-prompt.d.ts.map +1 -1
- package/dist/system-prompt.js +18 -50
- package/dist/system-prompt.js.map +1 -1
- package/dist/system-prompt.test.js +124 -1
- package/dist/system-prompt.test.js.map +1 -1
- package/dist/tools/edit-diff.d.ts.map +1 -1
- package/dist/tools/edit-diff.js +71 -32
- package/dist/tools/edit-diff.js.map +1 -1
- package/dist/tools/edit-diff.test.js +14 -0
- package/dist/tools/edit-diff.test.js.map +1 -1
- package/dist/tools/edit.d.ts.map +1 -1
- package/dist/tools/edit.js +16 -6
- package/dist/tools/edit.js.map +1 -1
- package/dist/tools/edit.test.js +27 -0
- package/dist/tools/edit.test.js.map +1 -1
- package/dist/tools/enter-plan.d.ts.map +1 -1
- package/dist/tools/enter-plan.js +1 -0
- package/dist/tools/enter-plan.js.map +1 -1
- package/dist/tools/find.d.ts.map +1 -1
- package/dist/tools/find.js +2 -0
- package/dist/tools/find.js.map +1 -1
- package/dist/tools/goals.d.ts +110 -0
- package/dist/tools/goals.d.ts.map +1 -0
- package/dist/tools/goals.js +500 -0
- package/dist/tools/goals.js.map +1 -0
- package/dist/tools/goals.test.d.ts +2 -0
- package/dist/tools/goals.test.d.ts.map +1 -0
- package/dist/tools/goals.test.js +431 -0
- package/dist/tools/goals.test.js.map +1 -0
- package/dist/tools/grep.d.ts.map +1 -1
- package/dist/tools/grep.js +2 -0
- package/dist/tools/grep.js.map +1 -1
- package/dist/tools/index.d.ts +2 -0
- package/dist/tools/index.d.ts.map +1 -1
- package/dist/tools/index.js +6 -0
- package/dist/tools/index.js.map +1 -1
- package/dist/tools/prompt-hints.d.ts.map +1 -1
- package/dist/tools/prompt-hints.js +2 -0
- package/dist/tools/prompt-hints.js.map +1 -1
- package/dist/tools/source-path.d.ts +9 -0
- package/dist/tools/source-path.d.ts.map +1 -0
- package/dist/tools/source-path.js +119 -0
- package/dist/tools/source-path.js.map +1 -0
- package/dist/tools/source-path.test.d.ts +2 -0
- package/dist/tools/source-path.test.d.ts.map +1 -0
- package/dist/tools/source-path.test.js +80 -0
- package/dist/tools/source-path.test.js.map +1 -0
- package/dist/tools/subagent.js +16 -0
- package/dist/tools/subagent.js.map +1 -1
- package/dist/ui/App.d.ts +36 -3
- package/dist/ui/App.d.ts.map +1 -1
- package/dist/ui/App.js +879 -57
- package/dist/ui/App.js.map +1 -1
- package/dist/ui/activity-phrases.d.ts.map +1 -1
- package/dist/ui/activity-phrases.js +7 -3
- package/dist/ui/activity-phrases.js.map +1 -1
- package/dist/ui/app-state-persistence.test.d.ts +2 -0
- package/dist/ui/app-state-persistence.test.d.ts.map +1 -0
- package/dist/ui/app-state-persistence.test.js +56 -0
- package/dist/ui/app-state-persistence.test.js.map +1 -0
- package/dist/ui/components/BackgroundTasksBar.d.ts +16 -1
- package/dist/ui/components/BackgroundTasksBar.d.ts.map +1 -1
- package/dist/ui/components/BackgroundTasksBar.js +15 -2
- package/dist/ui/components/BackgroundTasksBar.js.map +1 -1
- package/dist/ui/components/Banner.d.ts +2 -1
- package/dist/ui/components/Banner.d.ts.map +1 -1
- package/dist/ui/components/Banner.js +3 -3
- package/dist/ui/components/Banner.js.map +1 -1
- package/dist/ui/components/GoalOverlay.d.ts +21 -0
- package/dist/ui/components/GoalOverlay.d.ts.map +1 -0
- package/dist/ui/components/GoalOverlay.js +336 -0
- package/dist/ui/components/GoalOverlay.js.map +1 -0
- package/dist/ui/components/GoalStatusBar.d.ts +24 -0
- package/dist/ui/components/GoalStatusBar.d.ts.map +1 -0
- package/dist/ui/components/GoalStatusBar.js +113 -0
- package/dist/ui/components/GoalStatusBar.js.map +1 -0
- package/dist/ui/components/InputArea.d.ts +2 -1
- package/dist/ui/components/InputArea.d.ts.map +1 -1
- package/dist/ui/components/InputArea.js +6 -1
- package/dist/ui/components/InputArea.js.map +1 -1
- package/dist/ui/components/ToolExecution.d.ts.map +1 -1
- package/dist/ui/components/ToolExecution.js +94 -1
- package/dist/ui/components/ToolExecution.js.map +1 -1
- package/dist/ui/footer-status-layout.test.d.ts +2 -0
- package/dist/ui/footer-status-layout.test.d.ts.map +1 -0
- package/dist/ui/footer-status-layout.test.js +56 -0
- package/dist/ui/footer-status-layout.test.js.map +1 -0
- package/dist/ui/goal-events.d.ts +20 -0
- package/dist/ui/goal-events.d.ts.map +1 -0
- package/dist/ui/goal-events.js +102 -0
- package/dist/ui/goal-events.js.map +1 -0
- package/dist/ui/goal-events.test.d.ts +2 -0
- package/dist/ui/goal-events.test.d.ts.map +1 -0
- package/dist/ui/goal-events.test.js +208 -0
- package/dist/ui/goal-events.test.js.map +1 -0
- package/dist/ui/goal-overlay.test.d.ts +2 -0
- package/dist/ui/goal-overlay.test.d.ts.map +1 -0
- package/dist/ui/goal-overlay.test.js +122 -0
- package/dist/ui/goal-overlay.test.js.map +1 -0
- package/dist/ui/goal-status-bar.test.d.ts +2 -0
- package/dist/ui/goal-status-bar.test.d.ts.map +1 -0
- package/dist/ui/goal-status-bar.test.js +143 -0
- package/dist/ui/goal-status-bar.test.js.map +1 -0
- package/dist/ui/live-item-flush.test.js +48 -0
- package/dist/ui/live-item-flush.test.js.map +1 -1
- package/dist/ui/render.d.ts +8 -3
- package/dist/ui/render.d.ts.map +1 -1
- package/dist/ui/render.js +10 -3
- package/dist/ui/render.js.map +1 -1
- package/dist/ui/scroll-stabilization.test.d.ts +2 -0
- package/dist/ui/scroll-stabilization.test.d.ts.map +1 -0
- package/dist/ui/scroll-stabilization.test.js +23 -0
- package/dist/ui/scroll-stabilization.test.js.map +1 -0
- package/dist/utils/format.js +44 -0
- package/dist/utils/format.js.map +1 -1
- package/package.json +5 -4
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"footer-status-layout.test.d.ts","sourceRoot":"","sources":["../../src/ui/footer-status-layout.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import { getFooterStatusLayoutDecision } from "./components/BackgroundTasksBar.js";
|
|
3
|
+
describe("footer status layout decisions", () => {
|
|
4
|
+
it("keeps a single wide row when all footer status indicators are present", () => {
|
|
5
|
+
expect(getFooterStatusLayoutDecision({
|
|
6
|
+
columns: 140,
|
|
7
|
+
backgroundTaskCount: 2,
|
|
8
|
+
eyesCount: 3,
|
|
9
|
+
updatePending: true,
|
|
10
|
+
})).toEqual({
|
|
11
|
+
hasBackgroundTasks: true,
|
|
12
|
+
hasEyesSignals: true,
|
|
13
|
+
hasUpdateNotice: true,
|
|
14
|
+
stack: false,
|
|
15
|
+
compactBackgroundTasks: false,
|
|
16
|
+
});
|
|
17
|
+
});
|
|
18
|
+
it("uses compact background task copy before stacking is needed", () => {
|
|
19
|
+
expect(getFooterStatusLayoutDecision({
|
|
20
|
+
columns: 110,
|
|
21
|
+
backgroundTaskCount: 1,
|
|
22
|
+
eyesCount: 1,
|
|
23
|
+
updatePending: true,
|
|
24
|
+
})).toMatchObject({
|
|
25
|
+
stack: false,
|
|
26
|
+
compactBackgroundTasks: true,
|
|
27
|
+
});
|
|
28
|
+
});
|
|
29
|
+
it("stacks crowded status indicators on narrow terminals to avoid collisions", () => {
|
|
30
|
+
expect(getFooterStatusLayoutDecision({
|
|
31
|
+
columns: 80,
|
|
32
|
+
backgroundTaskCount: 1,
|
|
33
|
+
eyesCount: 2,
|
|
34
|
+
updatePending: true,
|
|
35
|
+
})).toEqual({
|
|
36
|
+
hasBackgroundTasks: true,
|
|
37
|
+
hasEyesSignals: true,
|
|
38
|
+
hasUpdateNotice: true,
|
|
39
|
+
stack: true,
|
|
40
|
+
compactBackgroundTasks: true,
|
|
41
|
+
});
|
|
42
|
+
});
|
|
43
|
+
it("does not stack a lone update notice", () => {
|
|
44
|
+
expect(getFooterStatusLayoutDecision({
|
|
45
|
+
columns: 60,
|
|
46
|
+
backgroundTaskCount: 0,
|
|
47
|
+
updatePending: true,
|
|
48
|
+
})).toMatchObject({
|
|
49
|
+
hasBackgroundTasks: false,
|
|
50
|
+
hasEyesSignals: false,
|
|
51
|
+
hasUpdateNotice: true,
|
|
52
|
+
stack: false,
|
|
53
|
+
});
|
|
54
|
+
});
|
|
55
|
+
});
|
|
56
|
+
//# sourceMappingURL=footer-status-layout.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"footer-status-layout.test.js","sourceRoot":"","sources":["../../src/ui/footer-status-layout.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAE,6BAA6B,EAAE,MAAM,oCAAoC,CAAC;AAEnF,QAAQ,CAAC,gCAAgC,EAAE,GAAG,EAAE;IAC9C,EAAE,CAAC,uEAAuE,EAAE,GAAG,EAAE;QAC/E,MAAM,CACJ,6BAA6B,CAAC;YAC5B,OAAO,EAAE,GAAG;YACZ,mBAAmB,EAAE,CAAC;YACtB,SAAS,EAAE,CAAC;YACZ,aAAa,EAAE,IAAI;SACpB,CAAC,CACH,CAAC,OAAO,CAAC;YACR,kBAAkB,EAAE,IAAI;YACxB,cAAc,EAAE,IAAI;YACpB,eAAe,EAAE,IAAI;YACrB,KAAK,EAAE,KAAK;YACZ,sBAAsB,EAAE,KAAK;SAC9B,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6DAA6D,EAAE,GAAG,EAAE;QACrE,MAAM,CACJ,6BAA6B,CAAC;YAC5B,OAAO,EAAE,GAAG;YACZ,mBAAmB,EAAE,CAAC;YACtB,SAAS,EAAE,CAAC;YACZ,aAAa,EAAE,IAAI;SACpB,CAAC,CACH,CAAC,aAAa,CAAC;YACd,KAAK,EAAE,KAAK;YACZ,sBAAsB,EAAE,IAAI;SAC7B,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0EAA0E,EAAE,GAAG,EAAE;QAClF,MAAM,CACJ,6BAA6B,CAAC;YAC5B,OAAO,EAAE,EAAE;YACX,mBAAmB,EAAE,CAAC;YACtB,SAAS,EAAE,CAAC;YACZ,aAAa,EAAE,IAAI;SACpB,CAAC,CACH,CAAC,OAAO,CAAC;YACR,kBAAkB,EAAE,IAAI;YACxB,cAAc,EAAE,IAAI;YACpB,eAAe,EAAE,IAAI;YACrB,KAAK,EAAE,IAAI;YACX,sBAAsB,EAAE,IAAI;SAC7B,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,qCAAqC,EAAE,GAAG,EAAE;QAC7C,MAAM,CACJ,6BAA6B,CAAC;YAC5B,OAAO,EAAE,EAAE;YACX,mBAAmB,EAAE,CAAC;YACtB,aAAa,EAAE,IAAI;SACpB,CAAC,CACH,CAAC,aAAa,CAAC;YACd,kBAAkB,EAAE,KAAK;YACzB,cAAc,EAAE,KAAK;YACrB,eAAe,EAAE,IAAI;YACrB,KAAK,EAAE,KAAK;SACb,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { type GoalRun } from "../core/goal-store.js";
|
|
2
|
+
import type { GoalWorkerCompletion } from "../core/goal-worker.js";
|
|
3
|
+
export declare const GOAL_WORKER_EVENT_PREFIX = "[event:goal_worker_complete]";
|
|
4
|
+
export declare const GOAL_VERIFIER_EVENT_PREFIX = "[event:goal_verifier_complete]";
|
|
5
|
+
export type GoalSyntheticEventKind = "worker" | "verifier";
|
|
6
|
+
export interface GoalSyntheticEventInfo {
|
|
7
|
+
kind: GoalSyntheticEventKind;
|
|
8
|
+
runId?: string;
|
|
9
|
+
goal?: string;
|
|
10
|
+
task?: string;
|
|
11
|
+
worker?: string;
|
|
12
|
+
status?: string;
|
|
13
|
+
exitCode?: number;
|
|
14
|
+
}
|
|
15
|
+
export declare function formatGoalWorkerCompletionEvent(run: GoalRun, taskTitle: string, completion: GoalWorkerCompletion): string;
|
|
16
|
+
export declare function formatGoalVerifierCompletionEvent(run: GoalRun, status: "pass" | "fail", command: string, exitCode: number, summary: string): string;
|
|
17
|
+
export declare function isGoalSyntheticEvent(text: string): boolean;
|
|
18
|
+
export declare function parseGoalSyntheticEvent(text: string): GoalSyntheticEventInfo | null;
|
|
19
|
+
export declare function shouldContinueGoalRun(run: GoalRun): boolean;
|
|
20
|
+
//# sourceMappingURL=goal-events.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"goal-events.d.ts","sourceRoot":"","sources":["../../src/ui/goal-events.ts"],"names":[],"mappings":"AACA,OAAO,EAGL,KAAK,OAAO,EACb,MAAM,uBAAuB,CAAC;AAC/B,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,wBAAwB,CAAC;AAEnE,eAAO,MAAM,wBAAwB,iCAAiC,CAAC;AACvE,eAAO,MAAM,0BAA0B,mCAAmC,CAAC;AAE3E,MAAM,MAAM,sBAAsB,GAAG,QAAQ,GAAG,UAAU,CAAC;AAE3D,MAAM,WAAW,sBAAsB;IACrC,IAAI,EAAE,sBAAsB,CAAC;IAC7B,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB;AAoCD,wBAAgB,+BAA+B,CAC7C,GAAG,EAAE,OAAO,EACZ,SAAS,EAAE,MAAM,EACjB,UAAU,EAAE,oBAAoB,GAC/B,MAAM,CAcR;AAED,wBAAgB,iCAAiC,CAC/C,GAAG,EAAE,OAAO,EACZ,MAAM,EAAE,MAAM,GAAG,MAAM,EACvB,OAAO,EAAE,MAAM,EACf,QAAQ,EAAE,MAAM,EAChB,OAAO,EAAE,MAAM,GACd,MAAM,CAaR;AAED,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAE1D;AAYD,wBAAgB,uBAAuB,CAAC,IAAI,EAAE,MAAM,GAAG,sBAAsB,GAAG,IAAI,CAmBnF;AAED,wBAAgB,qBAAqB,CAAC,GAAG,EAAE,OAAO,GAAG,OAAO,CAW3D"}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
import { DEFAULT_GOAL_VERIFIER_FIX_LIMIT } from "../core/goal-controller.js";
|
|
2
|
+
import { formatGoalBlockingPrerequisites, goalHasBlockingPrerequisites, } from "../core/goal-store.js";
|
|
3
|
+
export const GOAL_WORKER_EVENT_PREFIX = "[event:goal_worker_complete]";
|
|
4
|
+
export const GOAL_VERIFIER_EVENT_PREFIX = "[event:goal_verifier_complete]";
|
|
5
|
+
const GOAL_ORCHESTRATOR_INSTRUCTIONS = `orchestrator_instructions:
|
|
6
|
+
1. Call goals({ action: "status", run_id }) before deciding.
|
|
7
|
+
2. Briefly say what the orchestrator is doing so the chat shows progress.
|
|
8
|
+
3. Inspect durable tasks, verifier state, blockers, and evidence.
|
|
9
|
+
4. Take exactly one next control-loop action: add/update the next Goal task, run/record verification, pause/block with evidence, or complete only if verifier evidence proves the success criteria.
|
|
10
|
+
5. Do not merely narrate and do not ask the user to open the Goal pane.`;
|
|
11
|
+
function formatGoalState(run) {
|
|
12
|
+
const tasks = run.tasks
|
|
13
|
+
.map((task) => `- ${task.id}: ${task.status}; attempts=${task.attempts}; title=${task.title}`)
|
|
14
|
+
.join("\n") || "(none)";
|
|
15
|
+
const blockers = run.blockers.length > 0 ? run.blockers.map((b) => `- ${b}`).join("\n") : "(none)";
|
|
16
|
+
const verifier = run.verifier?.command
|
|
17
|
+
? `${run.verifier.command}; last=${run.verifier.lastResult?.status ?? "none"}; output=${run.verifier.lastResult?.outputPath ?? "none"}`
|
|
18
|
+
: "(none - define an exact verifier before completion)";
|
|
19
|
+
const prerequisites = run.prerequisites.length
|
|
20
|
+
? run.prerequisites
|
|
21
|
+
.map((item) => `- ${item.id}: ${item.status}; ${item.label}${item.instructions ? `; instructions=${item.instructions}` : ""}${item.evidence ? `; evidence=${item.evidence}` : ""}`)
|
|
22
|
+
.join("\n")
|
|
23
|
+
: "(none)";
|
|
24
|
+
const userPrerequisites = goalHasBlockingPrerequisites(run)
|
|
25
|
+
? formatGoalBlockingPrerequisites(run)
|
|
26
|
+
: "(none)";
|
|
27
|
+
const evidencePlan = run.evidencePlan.length
|
|
28
|
+
? run.evidencePlan.map((item) => `- ${item.id}: ${item.status}; ${item.label}`).join("\n")
|
|
29
|
+
: "(none)";
|
|
30
|
+
return `current_goal_state:\nstatus: ${run.status}\nuser_prerequisites: ${userPrerequisites}\nverifier: ${verifier}\nblockers:\n${blockers}\nprerequisites:\n${prerequisites}\nevidence_plan:\n${evidencePlan}\ntasks:\n${tasks}`;
|
|
31
|
+
}
|
|
32
|
+
export function formatGoalWorkerCompletionEvent(run, taskTitle, completion) {
|
|
33
|
+
const summary = completion.summary.trim() || "(empty)";
|
|
34
|
+
const toolsUsed = completion.toolsUsed.length > 0
|
|
35
|
+
? completion.toolsUsed.map((tool) => `${tool.ok ? "✓" : "✗"}${tool.name}`).join(", ")
|
|
36
|
+
: "(none)";
|
|
37
|
+
const reason = completion.reason ? ` reason=${completion.reason}` : "";
|
|
38
|
+
return `${GOAL_WORKER_EVENT_PREFIX} run_id="${run.id}" goal="${run.title}" task_id="${completion.worker.goalTaskId}" task="${taskTitle}" worker="${completion.worker.id}" status=${completion.status} exit_code=${completion.exitCode}${reason}
|
|
39
|
+
tools_used: ${toolsUsed}
|
|
40
|
+
${formatGoalState(run)}
|
|
41
|
+
summary:
|
|
42
|
+
${summary}
|
|
43
|
+
|
|
44
|
+
${GOAL_ORCHESTRATOR_INSTRUCTIONS}`;
|
|
45
|
+
}
|
|
46
|
+
export function formatGoalVerifierCompletionEvent(run, status, command, exitCode, summary) {
|
|
47
|
+
const fixCount = run.tasks.filter((task) => task.title === "Fix verifier failure").length;
|
|
48
|
+
const outputPath = run.verifier?.lastResult?.outputPath ?? "not recorded";
|
|
49
|
+
return `${GOAL_VERIFIER_EVENT_PREFIX} run_id="${run.id}" goal="${run.title}" status=${status} exit_code=${exitCode}
|
|
50
|
+
command: ${command}
|
|
51
|
+
output_path: ${outputPath}
|
|
52
|
+
fix_attempts: ${fixCount}/${DEFAULT_GOAL_VERIFIER_FIX_LIMIT}
|
|
53
|
+
completion_guidance: ${status === "pass" ? "Complete only if goals(status) shows success criteria, required evidence, and verifier output match the original objective exactly." : "Create one bounded fix task with the verifier command, exit code, output path, and failure summary unless the limit or repeated-failure guard is reached."}
|
|
54
|
+
${formatGoalState(run)}
|
|
55
|
+
summary:
|
|
56
|
+
${summary.trim() || "(empty)"}
|
|
57
|
+
|
|
58
|
+
${GOAL_ORCHESTRATOR_INSTRUCTIONS}`;
|
|
59
|
+
}
|
|
60
|
+
export function isGoalSyntheticEvent(text) {
|
|
61
|
+
return text.startsWith(GOAL_WORKER_EVENT_PREFIX) || text.startsWith(GOAL_VERIFIER_EVENT_PREFIX);
|
|
62
|
+
}
|
|
63
|
+
function quotedField(text, field) {
|
|
64
|
+
const match = new RegExp(`${field}="([^"]*)"`).exec(text);
|
|
65
|
+
return match?.[1];
|
|
66
|
+
}
|
|
67
|
+
function tokenField(text, field) {
|
|
68
|
+
const match = new RegExp(`${field}=([^\\s\\n]+)`).exec(text);
|
|
69
|
+
return match?.[1];
|
|
70
|
+
}
|
|
71
|
+
export function parseGoalSyntheticEvent(text) {
|
|
72
|
+
const kind = text.startsWith(GOAL_WORKER_EVENT_PREFIX)
|
|
73
|
+
? "worker"
|
|
74
|
+
: text.startsWith(GOAL_VERIFIER_EVENT_PREFIX)
|
|
75
|
+
? "verifier"
|
|
76
|
+
: null;
|
|
77
|
+
if (kind === null)
|
|
78
|
+
return null;
|
|
79
|
+
const exitCodeRaw = tokenField(text, "exit_code");
|
|
80
|
+
const exitCode = exitCodeRaw === undefined ? undefined : Number(exitCodeRaw);
|
|
81
|
+
return {
|
|
82
|
+
kind,
|
|
83
|
+
...(quotedField(text, "run_id") ? { runId: quotedField(text, "run_id") } : {}),
|
|
84
|
+
...(quotedField(text, "goal") ? { goal: quotedField(text, "goal") } : {}),
|
|
85
|
+
...(quotedField(text, "task") ? { task: quotedField(text, "task") } : {}),
|
|
86
|
+
...(quotedField(text, "worker") ? { worker: quotedField(text, "worker") } : {}),
|
|
87
|
+
...(tokenField(text, "status") ? { status: tokenField(text, "status") } : {}),
|
|
88
|
+
...(exitCode !== undefined && Number.isFinite(exitCode) ? { exitCode } : {}),
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
export function shouldContinueGoalRun(run) {
|
|
92
|
+
if (run.status === "blocked" ||
|
|
93
|
+
run.status === "paused" ||
|
|
94
|
+
run.status === "passed" ||
|
|
95
|
+
run.status === "failed") {
|
|
96
|
+
return false;
|
|
97
|
+
}
|
|
98
|
+
if (run.activeWorkerId)
|
|
99
|
+
return false;
|
|
100
|
+
return !run.tasks.some((task) => task.status === "running");
|
|
101
|
+
}
|
|
102
|
+
//# sourceMappingURL=goal-events.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"goal-events.js","sourceRoot":"","sources":["../../src/ui/goal-events.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"goal-events.test.d.ts","sourceRoot":"","sources":["../../src/ui/goal-events.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,208 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import { GOAL_VERIFIER_EVENT_PREFIX, GOAL_WORKER_EVENT_PREFIX, formatGoalVerifierCompletionEvent, formatGoalWorkerCompletionEvent, isGoalSyntheticEvent, parseGoalSyntheticEvent, shouldContinueGoalRun, } from "./goal-events.js";
|
|
3
|
+
function goalRun(overrides = {}) {
|
|
4
|
+
return {
|
|
5
|
+
id: "goal-12345678",
|
|
6
|
+
title: "Fix messy web output",
|
|
7
|
+
goal: "Fix web search and fetch output clarity",
|
|
8
|
+
status: "ready",
|
|
9
|
+
createdAt: "2024-01-01T00:00:00.000Z",
|
|
10
|
+
updatedAt: "2024-01-01T00:00:00.000Z",
|
|
11
|
+
projectPath: "/tmp/project",
|
|
12
|
+
successCriteria: ["Verifier passes"],
|
|
13
|
+
prerequisites: [],
|
|
14
|
+
harness: [],
|
|
15
|
+
evidencePlan: [],
|
|
16
|
+
tasks: [],
|
|
17
|
+
evidence: [],
|
|
18
|
+
blockers: [],
|
|
19
|
+
...overrides,
|
|
20
|
+
};
|
|
21
|
+
}
|
|
22
|
+
function workerCompletion(overrides = {}) {
|
|
23
|
+
return {
|
|
24
|
+
worker: {
|
|
25
|
+
id: "worker-a",
|
|
26
|
+
pid: 123,
|
|
27
|
+
goalRunId: "goal-12345678",
|
|
28
|
+
goalTaskId: "task-a",
|
|
29
|
+
cwd: "/tmp/project",
|
|
30
|
+
provider: "anthropic",
|
|
31
|
+
model: "claude-test",
|
|
32
|
+
startedAt: "2024-01-01T00:00:00.000Z",
|
|
33
|
+
logFile: "/tmp/worker.ndjson",
|
|
34
|
+
status: "done",
|
|
35
|
+
exitCode: 0,
|
|
36
|
+
},
|
|
37
|
+
summary: "Changed: web output cleaner\nVerified: npm run typecheck\nStatus: DONE",
|
|
38
|
+
status: "done",
|
|
39
|
+
exitCode: 0,
|
|
40
|
+
toolsUsed: [
|
|
41
|
+
{ name: "read", ok: true },
|
|
42
|
+
{ name: "edit", ok: true },
|
|
43
|
+
{ name: "bash", ok: true },
|
|
44
|
+
],
|
|
45
|
+
...overrides,
|
|
46
|
+
};
|
|
47
|
+
}
|
|
48
|
+
describe("goal event formatting", () => {
|
|
49
|
+
it("formats worker completion as a synthetic orchestrator event", () => {
|
|
50
|
+
const event = formatGoalWorkerCompletionEvent(goalRun(), "Implement cleaner web output", workerCompletion());
|
|
51
|
+
expect(event).toContain(GOAL_WORKER_EVENT_PREFIX);
|
|
52
|
+
expect(event).toContain('run_id="goal-12345678"');
|
|
53
|
+
expect(event).toContain('task_id="task-a"');
|
|
54
|
+
expect(event).toContain('worker="worker-a"');
|
|
55
|
+
expect(event).toContain("status=done");
|
|
56
|
+
expect(event).toContain("exit_code=0");
|
|
57
|
+
expect(event).toContain("tools_used: ✓read, ✓edit, ✓bash");
|
|
58
|
+
expect(event).toContain("current_goal_state:");
|
|
59
|
+
expect(event).toContain("user_prerequisites: (none)");
|
|
60
|
+
expect(event).toContain("tasks:\n(none)");
|
|
61
|
+
expect(event).toContain("summary:\nChanged: web output cleaner");
|
|
62
|
+
expect(event).toContain("orchestrator_instructions:");
|
|
63
|
+
expect(event).toContain('Call goals({ action: "status", run_id }) before deciding.');
|
|
64
|
+
expect(event).toContain("Briefly say what the orchestrator is doing");
|
|
65
|
+
expect(isGoalSyntheticEvent(event)).toBe(true);
|
|
66
|
+
expect(parseGoalSyntheticEvent(event)).toEqual({
|
|
67
|
+
kind: "worker",
|
|
68
|
+
runId: "goal-12345678",
|
|
69
|
+
goal: "Fix messy web output",
|
|
70
|
+
task: "Implement cleaner web output",
|
|
71
|
+
worker: "worker-a",
|
|
72
|
+
status: "done",
|
|
73
|
+
exitCode: 0,
|
|
74
|
+
});
|
|
75
|
+
});
|
|
76
|
+
it("includes exact user prerequisite instructions in synthetic state", () => {
|
|
77
|
+
const event = formatGoalWorkerCompletionEvent(goalRun({
|
|
78
|
+
status: "blocked",
|
|
79
|
+
prerequisites: [
|
|
80
|
+
{
|
|
81
|
+
id: "supabase-token",
|
|
82
|
+
label: "Supabase token",
|
|
83
|
+
status: "missing",
|
|
84
|
+
instructions: "Provide SUPABASE_ACCESS_TOKEN in chat or the local environment.",
|
|
85
|
+
},
|
|
86
|
+
],
|
|
87
|
+
}), "Collect prerequisites", workerCompletion());
|
|
88
|
+
expect(event).toContain("user_prerequisites: Supabase token: Provide SUPABASE_ACCESS_TOKEN in chat or the local environment.");
|
|
89
|
+
expect(event).toContain("- supabase-token: missing; Supabase token; instructions=Provide SUPABASE_ACCESS_TOKEN in chat or the local environment.");
|
|
90
|
+
});
|
|
91
|
+
it("formats failed worker completion with failed tool markers", () => {
|
|
92
|
+
const event = formatGoalWorkerCompletionEvent(goalRun(), "Fix verifier failure", workerCompletion({
|
|
93
|
+
status: "failed",
|
|
94
|
+
exitCode: 1,
|
|
95
|
+
summary: "Status: BLOCKED",
|
|
96
|
+
toolsUsed: [{ name: "bash", ok: false }],
|
|
97
|
+
}));
|
|
98
|
+
expect(event).toContain("status=failed");
|
|
99
|
+
expect(event).toContain("exit_code=1");
|
|
100
|
+
expect(event).toContain("tools_used: ✗bash");
|
|
101
|
+
expect(isGoalSyntheticEvent(event)).toBe(true);
|
|
102
|
+
expect(parseGoalSyntheticEvent(event)).toMatchObject({
|
|
103
|
+
kind: "worker",
|
|
104
|
+
status: "failed",
|
|
105
|
+
exitCode: 1,
|
|
106
|
+
});
|
|
107
|
+
});
|
|
108
|
+
it("formats crashed worker completion as failed synthetic event", () => {
|
|
109
|
+
const event = formatGoalWorkerCompletionEvent(goalRun(), "Run disposable worker", workerCompletion({
|
|
110
|
+
status: "failed",
|
|
111
|
+
exitCode: 1,
|
|
112
|
+
reason: "spawn_error",
|
|
113
|
+
summary: "Failed to spawn Goal worker: spawn exploded",
|
|
114
|
+
toolsUsed: [],
|
|
115
|
+
}));
|
|
116
|
+
expect(event).toContain("status=failed");
|
|
117
|
+
expect(event).toContain("reason=spawn_error");
|
|
118
|
+
expect(event).toContain("Failed to spawn Goal worker: spawn exploded");
|
|
119
|
+
expect(parseGoalSyntheticEvent(event)).toMatchObject({
|
|
120
|
+
kind: "worker",
|
|
121
|
+
runId: "goal-12345678",
|
|
122
|
+
status: "failed",
|
|
123
|
+
exitCode: 1,
|
|
124
|
+
});
|
|
125
|
+
});
|
|
126
|
+
it("formats stopped worker completion as failed synthetic event when a stop is summarized", () => {
|
|
127
|
+
const event = formatGoalWorkerCompletionEvent(goalRun(), "Stop disposable worker", workerCompletion({
|
|
128
|
+
status: "failed",
|
|
129
|
+
exitCode: 1,
|
|
130
|
+
summary: "Worker stopped by user.",
|
|
131
|
+
worker: { ...workerCompletion().worker, status: "stopped", exitCode: null },
|
|
132
|
+
toolsUsed: [],
|
|
133
|
+
}));
|
|
134
|
+
expect(event).toContain("status=failed");
|
|
135
|
+
expect(event).toContain("Worker stopped by user.");
|
|
136
|
+
expect(parseGoalSyntheticEvent(event)).toMatchObject({
|
|
137
|
+
kind: "worker",
|
|
138
|
+
status: "failed",
|
|
139
|
+
exitCode: 1,
|
|
140
|
+
});
|
|
141
|
+
});
|
|
142
|
+
it("formats verifier completion as a synthetic orchestrator event", () => {
|
|
143
|
+
const event = formatGoalVerifierCompletionEvent(goalRun(), "pass", "npm run typecheck && node scripts/test-web-tools.mjs", 0, "Verifier passed");
|
|
144
|
+
expect(event).toContain(GOAL_VERIFIER_EVENT_PREFIX);
|
|
145
|
+
expect(event).toContain('run_id="goal-12345678"');
|
|
146
|
+
expect(event).toContain("status=pass");
|
|
147
|
+
expect(event).toContain("exit_code=0");
|
|
148
|
+
expect(event).toContain("command: npm run typecheck && node scripts/test-web-tools.mjs");
|
|
149
|
+
expect(event).toContain("summary:\nVerifier passed");
|
|
150
|
+
expect(event).toContain("current_goal_state:");
|
|
151
|
+
expect(event).toContain("verifier: (none - define an exact verifier before completion)");
|
|
152
|
+
expect(event).toContain("Complete only if goals(status) shows success criteria");
|
|
153
|
+
expect(event).toContain("fix_attempts: 0/5");
|
|
154
|
+
expect(isGoalSyntheticEvent(event)).toBe(true);
|
|
155
|
+
expect(parseGoalSyntheticEvent(event)).toEqual({
|
|
156
|
+
kind: "verifier",
|
|
157
|
+
runId: "goal-12345678",
|
|
158
|
+
goal: "Fix messy web output",
|
|
159
|
+
status: "pass",
|
|
160
|
+
exitCode: 0,
|
|
161
|
+
});
|
|
162
|
+
});
|
|
163
|
+
it("formats verifier failure as corrective orchestrator input", () => {
|
|
164
|
+
const event = formatGoalVerifierCompletionEvent(goalRun(), "fail", "pnpm test", 1, "Verifier failed");
|
|
165
|
+
expect(event).toContain("status=fail");
|
|
166
|
+
expect(event).toContain("exit_code=1");
|
|
167
|
+
expect(event).toContain("Verifier failed");
|
|
168
|
+
expect(event).toContain("Inspect durable tasks, verifier state, blockers, and evidence.");
|
|
169
|
+
expect(parseGoalSyntheticEvent(event)).toEqual({
|
|
170
|
+
kind: "verifier",
|
|
171
|
+
runId: "goal-12345678",
|
|
172
|
+
goal: "Fix messy web output",
|
|
173
|
+
status: "fail",
|
|
174
|
+
exitCode: 1,
|
|
175
|
+
});
|
|
176
|
+
});
|
|
177
|
+
it("classifies only prefixed worker/verifier pings as synthetic events so display can hide them", () => {
|
|
178
|
+
expect(isGoalSyntheticEvent(formatGoalWorkerCompletionEvent(goalRun(), "Task", workerCompletion()))).toBe(true);
|
|
179
|
+
expect(isGoalSyntheticEvent(formatGoalVerifierCompletionEvent(goalRun(), "fail", "pnpm test", 1, "Verifier failed"))).toBe(true);
|
|
180
|
+
expect(isGoalSyntheticEvent("Please inspect the goal")).toBe(false);
|
|
181
|
+
expect(isGoalSyntheticEvent(`note ${GOAL_WORKER_EVENT_PREFIX}`)).toBe(false);
|
|
182
|
+
expect(parseGoalSyntheticEvent("Please inspect the goal")).toBeNull();
|
|
183
|
+
});
|
|
184
|
+
it("continues non-terminal goals that have no active worker or running task", () => {
|
|
185
|
+
expect(shouldContinueGoalRun(goalRun({ status: "ready" }))).toBe(true);
|
|
186
|
+
expect(shouldContinueGoalRun(goalRun({ status: "running" }))).toBe(true);
|
|
187
|
+
expect(shouldContinueGoalRun(goalRun({ status: "verifying" }))).toBe(true);
|
|
188
|
+
});
|
|
189
|
+
it("does not continue terminal or already-active goals", () => {
|
|
190
|
+
expect(shouldContinueGoalRun(goalRun({ status: "blocked" }))).toBe(false);
|
|
191
|
+
expect(shouldContinueGoalRun(goalRun({ status: "paused" }))).toBe(false);
|
|
192
|
+
expect(shouldContinueGoalRun(goalRun({ status: "passed" }))).toBe(false);
|
|
193
|
+
expect(shouldContinueGoalRun(goalRun({ status: "failed" }))).toBe(false);
|
|
194
|
+
expect(shouldContinueGoalRun(goalRun({ activeWorkerId: "worker-a" }))).toBe(false);
|
|
195
|
+
expect(shouldContinueGoalRun(goalRun({
|
|
196
|
+
tasks: [
|
|
197
|
+
{
|
|
198
|
+
id: "task-a",
|
|
199
|
+
title: "Running task",
|
|
200
|
+
prompt: "Do work",
|
|
201
|
+
status: "running",
|
|
202
|
+
attempts: 1,
|
|
203
|
+
},
|
|
204
|
+
],
|
|
205
|
+
}))).toBe(false);
|
|
206
|
+
});
|
|
207
|
+
});
|
|
208
|
+
//# sourceMappingURL=goal-events.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"goal-events.test.js","sourceRoot":"","sources":["../../src/ui/goal-events.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"goal-overlay.test.d.ts","sourceRoot":"","sources":["../../src/ui/goal-overlay.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,122 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import { clampGoalSelectedIndex, formatGoalPrerequisiteSummary, formatGoalTaskDetailSummary, formatGoalTaskSummary, formatGoalVerifierSummary, getGoalDetailTaskHeading, getGoalStatusCountsText, getGoalUserPrerequisiteHeading, shouldPersistGoalOverlayRuns, sortGoalRunsForOverlay, } from "./components/GoalOverlay.js";
|
|
3
|
+
function goalRun(overrides) {
|
|
4
|
+
return {
|
|
5
|
+
id: "goal-1",
|
|
6
|
+
title: "Goal",
|
|
7
|
+
goal: "Goal text",
|
|
8
|
+
status: "ready",
|
|
9
|
+
createdAt: "2024-01-01T00:00:00.000Z",
|
|
10
|
+
updatedAt: "2024-01-01T00:00:00.000Z",
|
|
11
|
+
projectPath: "/tmp/project",
|
|
12
|
+
successCriteria: [],
|
|
13
|
+
prerequisites: [],
|
|
14
|
+
harness: [],
|
|
15
|
+
evidencePlan: [],
|
|
16
|
+
tasks: [],
|
|
17
|
+
evidence: [],
|
|
18
|
+
blockers: [],
|
|
19
|
+
...overrides,
|
|
20
|
+
};
|
|
21
|
+
}
|
|
22
|
+
describe("goal overlay helpers", () => {
|
|
23
|
+
it("sorts runs newest first", () => {
|
|
24
|
+
const oldRun = goalRun({ id: "old", updatedAt: "2024-01-01T00:00:00.000Z" });
|
|
25
|
+
const newRun = goalRun({ id: "new", updatedAt: "2024-02-01T00:00:00.000Z" });
|
|
26
|
+
expect(sortGoalRunsForOverlay([oldRun, newRun]).map((run) => run.id)).toEqual(["new", "old"]);
|
|
27
|
+
});
|
|
28
|
+
it("clamps selected index", () => {
|
|
29
|
+
expect(clampGoalSelectedIndex(3, 0)).toBe(0);
|
|
30
|
+
expect(clampGoalSelectedIndex(-1, 3)).toBe(0);
|
|
31
|
+
expect(clampGoalSelectedIndex(4, 3)).toBe(2);
|
|
32
|
+
expect(clampGoalSelectedIndex(1, 3)).toBe(1);
|
|
33
|
+
});
|
|
34
|
+
it("summarizes prerequisites including blocking states", () => {
|
|
35
|
+
const run = goalRun({
|
|
36
|
+
prerequisites: [
|
|
37
|
+
{ id: "cli", label: "CLI", status: "met" },
|
|
38
|
+
{ id: "sim", label: "Simulator", status: "missing" },
|
|
39
|
+
{ id: "data", label: "Fixture data", status: "unknown" },
|
|
40
|
+
],
|
|
41
|
+
});
|
|
42
|
+
expect(formatGoalPrerequisiteSummary(run)).toBe("1/3 prereqs met (1 missing, 1 unknown)");
|
|
43
|
+
});
|
|
44
|
+
it("puts user prerequisites before worker tasks in detail headings", () => {
|
|
45
|
+
const run = goalRun({
|
|
46
|
+
prerequisites: [
|
|
47
|
+
{
|
|
48
|
+
id: "supabase-token",
|
|
49
|
+
label: "Supabase token",
|
|
50
|
+
status: "missing",
|
|
51
|
+
instructions: "Provide SUPABASE_ACCESS_TOKEN.",
|
|
52
|
+
},
|
|
53
|
+
],
|
|
54
|
+
});
|
|
55
|
+
expect(getGoalUserPrerequisiteHeading(run)).toBe("1. User prerequisites");
|
|
56
|
+
expect(getGoalDetailTaskHeading(run)).toBe("2. Worker tasks");
|
|
57
|
+
expect(getGoalUserPrerequisiteHeading(goalRun({}))).toBeNull();
|
|
58
|
+
expect(getGoalDetailTaskHeading(goalRun({}))).toBe("Worker tasks");
|
|
59
|
+
});
|
|
60
|
+
it("summarizes task states", () => {
|
|
61
|
+
const run = goalRun({
|
|
62
|
+
tasks: [
|
|
63
|
+
{ id: "a", title: "A", prompt: "A", status: "done", attempts: 1 },
|
|
64
|
+
{ id: "b", title: "B", prompt: "B", status: "running", attempts: 2 },
|
|
65
|
+
{ id: "c", title: "C", prompt: "C", status: "failed", attempts: 1 },
|
|
66
|
+
{ id: "d", title: "D", prompt: "D", status: "blocked", attempts: 0 },
|
|
67
|
+
],
|
|
68
|
+
});
|
|
69
|
+
expect(formatGoalTaskSummary(run)).toBe("1/4 tasks done (1 running, 1 failed, 1 blocked)");
|
|
70
|
+
});
|
|
71
|
+
it("summarizes task detail with only the first concise line", () => {
|
|
72
|
+
expect(formatGoalTaskDetailSummary("\nChanged the harness.\nVerified tests.")).toBe("Changed the harness.");
|
|
73
|
+
expect(formatGoalTaskDetailSummary("\n\n")).toBe("");
|
|
74
|
+
expect(formatGoalTaskDetailSummary("x".repeat(220))).toBe(`${"x".repeat(177)}…`);
|
|
75
|
+
});
|
|
76
|
+
it("summarizes verifier state", () => {
|
|
77
|
+
expect(formatGoalVerifierSummary(goalRun({}))).toBe("no verifier");
|
|
78
|
+
expect(formatGoalVerifierSummary(goalRun({ verifier: { description: "Run tests", command: "pnpm test" } }))).toBe("verifier command ready");
|
|
79
|
+
expect(formatGoalVerifierSummary(goalRun({
|
|
80
|
+
verifier: {
|
|
81
|
+
description: "Run tests",
|
|
82
|
+
lastResult: {
|
|
83
|
+
status: "pass",
|
|
84
|
+
summary: "passed",
|
|
85
|
+
checkedAt: "2024-01-01T00:00:00.000Z",
|
|
86
|
+
},
|
|
87
|
+
},
|
|
88
|
+
}))).toBe("verifier pass");
|
|
89
|
+
});
|
|
90
|
+
it("formats status counts for header", () => {
|
|
91
|
+
const runs = [
|
|
92
|
+
goalRun({ id: "passed", status: "passed" }),
|
|
93
|
+
goalRun({ id: "running", status: "running" }),
|
|
94
|
+
goalRun({ id: "paused", status: "paused" }),
|
|
95
|
+
goalRun({ id: "blocked", status: "blocked" }),
|
|
96
|
+
];
|
|
97
|
+
expect(getGoalStatusCountsText(runs)).toBe("1 passed · 1 running · 1 pending · 1 blocked");
|
|
98
|
+
});
|
|
99
|
+
it("refuses to persist a transient empty overlay state while active Goal work exists", () => {
|
|
100
|
+
const activeRuns = [
|
|
101
|
+
goalRun({
|
|
102
|
+
id: "active",
|
|
103
|
+
status: "running",
|
|
104
|
+
activeWorkerId: "worker-a",
|
|
105
|
+
tasks: [
|
|
106
|
+
{
|
|
107
|
+
id: "task-a",
|
|
108
|
+
title: "Active work",
|
|
109
|
+
prompt: "Do work",
|
|
110
|
+
status: "running",
|
|
111
|
+
attempts: 1,
|
|
112
|
+
workerId: "worker-a",
|
|
113
|
+
},
|
|
114
|
+
],
|
|
115
|
+
}),
|
|
116
|
+
];
|
|
117
|
+
expect(shouldPersistGoalOverlayRuns(activeRuns, [])).toBe(false);
|
|
118
|
+
expect(shouldPersistGoalOverlayRuns(activeRuns, [goalRun({ id: "next" })])).toBe(true);
|
|
119
|
+
expect(shouldPersistGoalOverlayRuns([goalRun({ id: "done", status: "passed" })], [])).toBe(true);
|
|
120
|
+
});
|
|
121
|
+
});
|
|
122
|
+
//# sourceMappingURL=goal-overlay.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"goal-overlay.test.js","sourceRoot":"","sources":["../../src/ui/goal-overlay.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"goal-status-bar.test.d.ts","sourceRoot":"","sources":["../../src/ui/goal-status-bar.test.ts"],"names":[],"mappings":""}
|