gsd-pi 2.81.0-dev.72a81bdf3 → 2.82.0-dev.2841a1e44
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/README.md +49 -30
- package/dist/resources/.managed-resources-content-hash +1 -1
- package/dist/resources/GSD-WORKFLOW.md +3 -1
- package/dist/resources/extensions/browser-tools/tools/screenshot.js +1 -0
- package/dist/resources/extensions/browser-tools/tools/zoom.js +1 -0
- package/dist/resources/extensions/cmux/index.js +5 -0
- package/dist/resources/extensions/gsd/auto/orchestrator.js +113 -6
- package/dist/resources/extensions/gsd/auto/phases.js +9 -0
- package/dist/resources/extensions/gsd/auto-post-unit.js +169 -124
- package/dist/resources/extensions/gsd/auto-prompts.js +13 -5
- package/dist/resources/extensions/gsd/auto-verification.js +28 -22
- package/dist/resources/extensions/gsd/auto.js +128 -52
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +5 -0
- package/dist/resources/extensions/gsd/bootstrap/subagent-input.js +16 -7
- package/dist/resources/extensions/gsd/bootstrap/system-context.js +55 -12
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +3 -1
- package/dist/resources/extensions/gsd/clean-root-preflight.js +170 -8
- package/dist/resources/extensions/gsd/commands/catalog.js +4 -1
- package/dist/resources/extensions/gsd/commands/handlers/core.js +22 -1
- package/dist/resources/extensions/gsd/commands-bootstrap.js +5 -0
- package/dist/resources/extensions/gsd/commands-handlers.js +15 -2
- package/dist/resources/extensions/gsd/context-store.js +112 -0
- package/dist/resources/extensions/gsd/db-writer.js +150 -84
- package/dist/resources/extensions/gsd/docs/preferences-reference.md +1 -1
- package/dist/resources/extensions/gsd/doctor-git-checks.js +41 -6
- package/dist/resources/extensions/gsd/knowledge-backfill.js +144 -0
- package/dist/resources/extensions/gsd/knowledge-capture.js +136 -0
- package/dist/resources/extensions/gsd/knowledge-parser.js +154 -0
- package/dist/resources/extensions/gsd/knowledge-projection.js +210 -0
- package/dist/resources/extensions/gsd/markdown-renderer.js +6 -1
- package/dist/resources/extensions/gsd/md-importer.js +1 -1
- package/dist/resources/extensions/gsd/memory-backfill.js +73 -17
- package/dist/resources/extensions/gsd/memory-consolidation-scanner.js +222 -0
- package/dist/resources/extensions/gsd/migrate/command.js +5 -0
- package/dist/resources/extensions/gsd/migrate/preview.js +9 -0
- package/dist/resources/extensions/gsd/migrate/transformer.js +51 -4
- package/dist/resources/extensions/gsd/migrate/writer.js +11 -1
- package/dist/resources/extensions/gsd/prompts/system.md +2 -2
- package/dist/resources/extensions/gsd/provider-switch-observer.js +146 -0
- package/dist/resources/extensions/gsd/templates/knowledge.md +2 -2
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +119 -0
- package/dist/resources/extensions/gsd/unit-context-manifest.js +25 -2
- package/dist/resources/extensions/gsd/verification-verdict.js +26 -0
- package/dist/resources/extensions/gsd/worktree-lifecycle.js +21 -2
- package/dist/resources/extensions/subagent/index.js +448 -78
- package/dist/resources/extensions/subagent/launch.js +77 -0
- package/dist/resources/extensions/subagent/run-store.js +148 -0
- package/dist/resources/extensions/visual-brief/artifact-policy.js +29 -0
- package/dist/resources/extensions/visual-brief/extension-manifest.json +8 -0
- package/dist/resources/extensions/visual-brief/index.js +5 -0
- package/dist/resources/extensions/visual-brief/page-contract.js +122 -0
- package/dist/resources/extensions/visual-brief/prompts.js +111 -0
- package/dist/tsconfig.extensions.tsbuildinfo +1 -1
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +15 -15
- package/dist/web/standalone/.next/build-manifest.json +3 -3
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/react-loadable-manifest.json +3 -3
- package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +2 -2
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +2 -2
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +15 -15
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +1 -1
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/dist/web/standalone/.next/static/chunks/2973.33f26573894b6153.js +2 -0
- package/dist/web/standalone/.next/static/chunks/{8359.e059d86b255fce1c.js → 8359.7eb3bb8f8ecf4c01.js} +2 -2
- package/dist/web/standalone/.next/static/chunks/{webpack-de742b64187e13fe.js → webpack-6a95bc41e0f7ec89.js} +1 -1
- package/dist/web/standalone/.next/static/css/0262768ec1b89d34.css +1 -0
- package/package.json +5 -4
- package/packages/contracts/dist/rpc.test.js +7 -0
- package/packages/contracts/dist/rpc.test.js.map +1 -1
- package/packages/contracts/dist/workflow.d.ts +21 -0
- package/packages/contracts/dist/workflow.d.ts.map +1 -1
- package/packages/contracts/dist/workflow.js +24 -0
- package/packages/contracts/dist/workflow.js.map +1 -1
- package/packages/contracts/src/rpc.test.ts +8 -0
- package/packages/contracts/src/workflow.ts +24 -0
- package/packages/daemon/package.json +2 -2
- package/packages/mcp-server/README.md +14 -3
- package/packages/mcp-server/dist/workflow-tools.d.ts +0 -3
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +80 -0
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/package.json +2 -2
- package/packages/mcp-server/src/workflow-tools-parity.test.ts +244 -0
- package/packages/mcp-server/src/workflow-tools.test.ts +22 -0
- package/packages/mcp-server/src/workflow-tools.ts +168 -0
- package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
- package/packages/native/package.json +1 -1
- package/packages/pi-agent-core/package.json +1 -1
- package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-ai/dist/index.d.ts +2 -2
- package/packages/pi-ai/dist/index.d.ts.map +1 -1
- package/packages/pi-ai/dist/index.js +1 -1
- package/packages/pi-ai/dist/index.js.map +1 -1
- package/packages/pi-ai/dist/providers/transform-messages.d.ts +11 -0
- package/packages/pi-ai/dist/providers/transform-messages.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/transform-messages.js +20 -0
- package/packages/pi-ai/dist/providers/transform-messages.js.map +1 -1
- package/packages/pi-ai/package.json +1 -1
- package/packages/pi-ai/src/index.ts +7 -2
- package/packages/pi-ai/src/providers/transform-messages.ts +24 -0
- package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-coding-agent/dist/core/system-prompt.js +4 -4
- package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
- package/packages/pi-coding-agent/dist/tests/system-prompt-file-safety.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/tests/system-prompt-file-safety.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/tests/system-prompt-file-safety.test.js +17 -0
- package/packages/pi-coding-agent/dist/tests/system-prompt-file-safety.test.js.map +1 -0
- package/packages/pi-coding-agent/package.json +1 -1
- package/packages/pi-coding-agent/src/core/system-prompt.ts +4 -4
- package/packages/pi-coding-agent/src/tests/system-prompt-file-safety.test.ts +22 -0
- package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-tui/dist/tui.d.ts.map +1 -1
- package/packages/pi-tui/dist/tui.js +5 -0
- package/packages/pi-tui/dist/tui.js.map +1 -1
- package/packages/pi-tui/package.json +1 -1
- package/packages/pi-tui/src/tui.ts +6 -0
- package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
- package/packages/rpc-client/package.json +1 -1
- package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
- package/pkg/package.json +1 -1
- package/src/resources/GSD-WORKFLOW.md +3 -1
- package/src/resources/extensions/browser-tools/tools/screenshot.ts +1 -0
- package/src/resources/extensions/browser-tools/tools/zoom.ts +1 -0
- package/src/resources/extensions/cmux/index.ts +6 -0
- package/src/resources/extensions/gsd/auto/contracts.ts +46 -11
- package/src/resources/extensions/gsd/auto/orchestrator.ts +118 -6
- package/src/resources/extensions/gsd/auto/phases.ts +14 -0
- package/src/resources/extensions/gsd/auto-post-unit.ts +194 -137
- package/src/resources/extensions/gsd/auto-prompts.ts +13 -5
- package/src/resources/extensions/gsd/auto-verification.ts +36 -34
- package/src/resources/extensions/gsd/auto.ts +136 -51
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +6 -0
- package/src/resources/extensions/gsd/bootstrap/subagent-input.ts +16 -6
- package/src/resources/extensions/gsd/bootstrap/system-context.ts +58 -15
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +3 -2
- package/src/resources/extensions/gsd/clean-root-preflight.ts +174 -8
- package/src/resources/extensions/gsd/commands/catalog.ts +4 -1
- package/src/resources/extensions/gsd/commands/handlers/core.ts +25 -1
- package/src/resources/extensions/gsd/commands-bootstrap.ts +10 -0
- package/src/resources/extensions/gsd/commands-handlers.ts +19 -2
- package/src/resources/extensions/gsd/context-store.ts +120 -1
- package/src/resources/extensions/gsd/db-writer.ts +167 -84
- package/src/resources/extensions/gsd/docs/preferences-reference.md +1 -1
- package/src/resources/extensions/gsd/doctor-git-checks.ts +44 -6
- package/src/resources/extensions/gsd/doctor-types.ts +2 -0
- package/src/resources/extensions/gsd/knowledge-backfill.ts +164 -0
- package/src/resources/extensions/gsd/knowledge-capture.ts +160 -0
- package/src/resources/extensions/gsd/knowledge-parser.ts +174 -0
- package/src/resources/extensions/gsd/knowledge-projection.ts +241 -0
- package/src/resources/extensions/gsd/markdown-renderer.ts +6 -1
- package/src/resources/extensions/gsd/md-importer.ts +1 -1
- package/src/resources/extensions/gsd/memory-backfill.ts +89 -17
- package/src/resources/extensions/gsd/memory-consolidation-scanner.ts +277 -0
- package/src/resources/extensions/gsd/migrate/command.ts +5 -0
- package/src/resources/extensions/gsd/migrate/preview.ts +10 -0
- package/src/resources/extensions/gsd/migrate/transformer.ts +58 -4
- package/src/resources/extensions/gsd/migrate/writer.ts +14 -1
- package/src/resources/extensions/gsd/prompts/system.md +2 -2
- package/src/resources/extensions/gsd/provider-switch-observer.ts +185 -0
- package/src/resources/extensions/gsd/templates/knowledge.md +2 -2
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +75 -0
- package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +408 -4
- package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +6 -5
- package/src/resources/extensions/gsd/tests/auto-runtime-state.test.ts +4 -4
- package/src/resources/extensions/gsd/tests/brief-command.test.ts +89 -0
- package/src/resources/extensions/gsd/tests/browser-tools-compatibility-declarations.test.ts +62 -0
- package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +107 -2
- package/src/resources/extensions/gsd/tests/closeout-git-deferral.test.ts +16 -0
- package/src/resources/extensions/gsd/tests/context-store-decisions-from-memories.test.ts +312 -0
- package/src/resources/extensions/gsd/tests/db-writer.test.ts +13 -8
- package/src/resources/extensions/gsd/tests/decisions-projection-from-memories.test.ts +453 -0
- package/src/resources/extensions/gsd/tests/decisions-stop-table-writes.test.ts +348 -0
- package/src/resources/extensions/gsd/tests/evidence-cross-ref.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/freeform-decisions.test.ts +8 -4
- package/src/resources/extensions/gsd/tests/gsd-tools.test.ts +11 -7
- package/src/resources/extensions/gsd/tests/integration/doctor-git.test.ts +44 -0
- package/src/resources/extensions/gsd/tests/integration/integration-lifecycle.test.ts +13 -5
- package/src/resources/extensions/gsd/tests/integration/migrate-command.test.ts +48 -3
- package/src/resources/extensions/gsd/tests/knowledge-backfill-projection.test.ts +323 -0
- package/src/resources/extensions/gsd/tests/knowledge-capture.test.ts +242 -0
- package/src/resources/extensions/gsd/tests/knowledge.test.ts +47 -2
- package/src/resources/extensions/gsd/tests/load-knowledge-block-rules-only.test.ts +209 -0
- package/src/resources/extensions/gsd/tests/memory-consolidation-scanner.test.ts +316 -0
- package/src/resources/extensions/gsd/tests/migrate-transformer.test.ts +5 -1
- package/src/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +6 -1
- package/src/resources/extensions/gsd/tests/plan-milestone-sketch-render.test.ts +157 -0
- package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
- package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +8 -0
- package/src/resources/extensions/gsd/tests/provider-switch-observer.test.ts +252 -0
- package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +16 -4
- package/src/resources/extensions/gsd/tests/state-corruption-2945.test.ts +6 -0
- package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +21 -0
- package/src/resources/extensions/gsd/tests/verification-verdict.test.ts +78 -0
- package/src/resources/extensions/gsd/tests/worktree-lifecycle.test.ts +25 -0
- package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +16 -0
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +135 -0
- package/src/resources/extensions/gsd/unit-context-manifest.ts +35 -2
- package/src/resources/extensions/gsd/verification-verdict.ts +47 -0
- package/src/resources/extensions/gsd/workflow-logger.ts +4 -0
- package/src/resources/extensions/gsd/worktree-lifecycle.ts +20 -2
- package/src/resources/extensions/subagent/index.ts +567 -103
- package/src/resources/extensions/subagent/launch.ts +131 -0
- package/src/resources/extensions/subagent/run-store.ts +218 -0
- package/src/resources/extensions/subagent/tests/launch.test.ts +115 -0
- package/src/resources/extensions/subagent/tests/run-store.test.ts +111 -0
- package/src/resources/extensions/visual-brief/artifact-policy.ts +41 -0
- package/src/resources/extensions/visual-brief/extension-manifest.json +8 -0
- package/src/resources/extensions/visual-brief/index.ts +8 -0
- package/src/resources/extensions/visual-brief/page-contract.ts +134 -0
- package/src/resources/extensions/visual-brief/prompts.ts +147 -0
- package/src/resources/extensions/visual-brief/tests/visual-brief.test.ts +172 -0
- package/dist/web/standalone/.next/static/chunks/2556.0527fea66e123b7f.js +0 -1
- package/dist/web/standalone/.next/static/css/54ec2745c1da488b.css +0 -1
- /package/dist/web/standalone/.next/static/{rIkMv4YSNlfSeqmGqWVns → Qgr2B_MRhPxC0z8fwv4vT}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{rIkMv4YSNlfSeqmGqWVns → Qgr2B_MRhPxC0z8fwv4vT}/_ssgManifest.js +0 -0
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
// GSD-2 — ADR-005 Phase 3b: surface ProviderSwitchReport from pi-ai.
|
|
2
|
+
//
|
|
3
|
+
// pi-ai builds a ProviderSwitchReport on every cross-provider transform but
|
|
4
|
+
// only logs it to stderr when GSD_VERBOSE=1. This module installs a
|
|
5
|
+
// single-subscriber observer that surfaces non-empty reports through GSD's
|
|
6
|
+
// three usual telemetry surfaces:
|
|
7
|
+
//
|
|
8
|
+
// 1. UOK audit event (category model-policy, type provider-switch) — only
|
|
9
|
+
// when an auto trace is active.
|
|
10
|
+
// 2. Persistent notification (.gsd/notifications.jsonl, severity warning) —
|
|
11
|
+
// whenever the GSD basePath is known, so users see the loss in the
|
|
12
|
+
// dashboard / status surface without GSD_VERBOSE.
|
|
13
|
+
// 3. In-memory counter, exposed via getProviderSwitchStats() so any
|
|
14
|
+
// caller (dashboard, doctor, tests) can read the rollup.
|
|
15
|
+
import { setProviderSwitchObserver } from "@gsd/pi-ai";
|
|
16
|
+
import { autoSession } from "./auto-runtime-state.js";
|
|
17
|
+
import { appendNotification } from "./notification-store.js";
|
|
18
|
+
import { buildAuditEnvelope, emitUokAuditEvent } from "./uok/audit.js";
|
|
19
|
+
const INTERACTIVE_TRACE_KEY = "interactive";
|
|
20
|
+
let installed = false;
|
|
21
|
+
let totalSwitches = 0;
|
|
22
|
+
const totals = {
|
|
23
|
+
thinkingBlocksDropped: 0,
|
|
24
|
+
thinkingBlocksDowngraded: 0,
|
|
25
|
+
toolCallIdsRemapped: 0,
|
|
26
|
+
syntheticToolResultsInserted: 0,
|
|
27
|
+
thoughtSignaturesDropped: 0,
|
|
28
|
+
};
|
|
29
|
+
const byTrace = new Map();
|
|
30
|
+
let lastReport = null;
|
|
31
|
+
let lastAt = null;
|
|
32
|
+
/** Format a one-line summary suitable for a notification message. */
|
|
33
|
+
function summarize(report) {
|
|
34
|
+
const parts = [];
|
|
35
|
+
if (report.thinkingBlocksDropped > 0)
|
|
36
|
+
parts.push(`${report.thinkingBlocksDropped} thinking dropped`);
|
|
37
|
+
if (report.thinkingBlocksDowngraded > 0)
|
|
38
|
+
parts.push(`${report.thinkingBlocksDowngraded} thinking downgraded`);
|
|
39
|
+
if (report.toolCallIdsRemapped > 0)
|
|
40
|
+
parts.push(`${report.toolCallIdsRemapped} tool ids remapped`);
|
|
41
|
+
if (report.syntheticToolResultsInserted > 0)
|
|
42
|
+
parts.push(`${report.syntheticToolResultsInserted} synthetic tool results`);
|
|
43
|
+
if (report.thoughtSignaturesDropped > 0)
|
|
44
|
+
parts.push(`${report.thoughtSignaturesDropped} thought signatures dropped`);
|
|
45
|
+
return `Provider switch ${report.fromApi} → ${report.toApi}: ${parts.join(", ")}`;
|
|
46
|
+
}
|
|
47
|
+
function recordReport(report) {
|
|
48
|
+
const now = new Date().toISOString();
|
|
49
|
+
totalSwitches += 1;
|
|
50
|
+
totals.thinkingBlocksDropped += report.thinkingBlocksDropped;
|
|
51
|
+
totals.thinkingBlocksDowngraded += report.thinkingBlocksDowngraded;
|
|
52
|
+
totals.toolCallIdsRemapped += report.toolCallIdsRemapped;
|
|
53
|
+
totals.syntheticToolResultsInserted += report.syntheticToolResultsInserted;
|
|
54
|
+
totals.thoughtSignaturesDropped += report.thoughtSignaturesDropped;
|
|
55
|
+
lastReport = report;
|
|
56
|
+
lastAt = now;
|
|
57
|
+
const traceKey = autoSession.currentTraceId ?? INTERACTIVE_TRACE_KEY;
|
|
58
|
+
const existing = byTrace.get(traceKey);
|
|
59
|
+
if (existing) {
|
|
60
|
+
existing.switches += 1;
|
|
61
|
+
existing.lastReport = report;
|
|
62
|
+
existing.lastAt = now;
|
|
63
|
+
}
|
|
64
|
+
else {
|
|
65
|
+
byTrace.set(traceKey, { switches: 1, lastReport: report, lastAt: now });
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
function emitAudit(report) {
|
|
69
|
+
const traceId = autoSession.currentTraceId;
|
|
70
|
+
const basePath = autoSession.basePath;
|
|
71
|
+
if (!traceId || !basePath)
|
|
72
|
+
return;
|
|
73
|
+
try {
|
|
74
|
+
emitUokAuditEvent(basePath, buildAuditEnvelope({
|
|
75
|
+
traceId,
|
|
76
|
+
category: "model-policy",
|
|
77
|
+
type: "provider-switch",
|
|
78
|
+
payload: {
|
|
79
|
+
fromApi: report.fromApi,
|
|
80
|
+
toApi: report.toApi,
|
|
81
|
+
thinkingBlocksDropped: report.thinkingBlocksDropped,
|
|
82
|
+
thinkingBlocksDowngraded: report.thinkingBlocksDowngraded,
|
|
83
|
+
toolCallIdsRemapped: report.toolCallIdsRemapped,
|
|
84
|
+
syntheticToolResultsInserted: report.syntheticToolResultsInserted,
|
|
85
|
+
thoughtSignaturesDropped: report.thoughtSignaturesDropped,
|
|
86
|
+
},
|
|
87
|
+
}));
|
|
88
|
+
}
|
|
89
|
+
catch {
|
|
90
|
+
// Audit emission is best-effort. Counter + notification still fire.
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
function emitNotification(report) {
|
|
94
|
+
try {
|
|
95
|
+
appendNotification(summarize(report), "warning", "workflow-logger");
|
|
96
|
+
}
|
|
97
|
+
catch {
|
|
98
|
+
// Notification persistence is best-effort.
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
function handleReport(report) {
|
|
102
|
+
recordReport(report);
|
|
103
|
+
emitAudit(report);
|
|
104
|
+
emitNotification(report);
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* Install the pi-ai observer. Idempotent — calling more than once is a no-op
|
|
108
|
+
* after the first install.
|
|
109
|
+
*/
|
|
110
|
+
export function installProviderSwitchObserver() {
|
|
111
|
+
if (installed)
|
|
112
|
+
return;
|
|
113
|
+
setProviderSwitchObserver(handleReport);
|
|
114
|
+
installed = true;
|
|
115
|
+
}
|
|
116
|
+
/** Uninstall the observer. Intended for tests. */
|
|
117
|
+
export function uninstallProviderSwitchObserver() {
|
|
118
|
+
setProviderSwitchObserver(undefined);
|
|
119
|
+
installed = false;
|
|
120
|
+
}
|
|
121
|
+
/** Read-only snapshot of the in-memory rollup. */
|
|
122
|
+
export function getProviderSwitchStats() {
|
|
123
|
+
const trace = {};
|
|
124
|
+
for (const [key, value] of byTrace) {
|
|
125
|
+
trace[key] = { switches: value.switches, lastReport: { ...value.lastReport }, lastAt: value.lastAt };
|
|
126
|
+
}
|
|
127
|
+
return {
|
|
128
|
+
totalSwitches,
|
|
129
|
+
totals: { ...totals },
|
|
130
|
+
byTrace: trace,
|
|
131
|
+
lastReport: lastReport ? { ...lastReport } : null,
|
|
132
|
+
lastAt,
|
|
133
|
+
};
|
|
134
|
+
}
|
|
135
|
+
/** Reset the in-memory rollup. Intended for tests. */
|
|
136
|
+
export function _resetProviderSwitchStats() {
|
|
137
|
+
totalSwitches = 0;
|
|
138
|
+
totals.thinkingBlocksDropped = 0;
|
|
139
|
+
totals.thinkingBlocksDowngraded = 0;
|
|
140
|
+
totals.toolCallIdsRemapped = 0;
|
|
141
|
+
totals.syntheticToolResultsInserted = 0;
|
|
142
|
+
totals.thoughtSignaturesDropped = 0;
|
|
143
|
+
byTrace.clear();
|
|
144
|
+
lastReport = null;
|
|
145
|
+
lastAt = null;
|
|
146
|
+
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
# Project Knowledge
|
|
2
2
|
|
|
3
|
-
|
|
4
|
-
|
|
3
|
+
Project-specific rules plus projected patterns and lessons learned.
|
|
4
|
+
Rules are maintained in this file. Patterns and lessons are persisted to the memories table and rendered here on the next session-start projection.
|
|
5
5
|
|
|
6
6
|
## Rules
|
|
7
7
|
|
|
@@ -1,3 +1,5 @@
|
|
|
1
|
+
// Project/App: GSD-2
|
|
2
|
+
// File Purpose: Adapts shared GSD workflow handlers for MCP executor calls.
|
|
1
3
|
import { ensureDbOpen } from "../bootstrap/dynamic-tools.js";
|
|
2
4
|
import { sanitizeCompleteMilestoneParams } from "../bootstrap/sanitize-complete-milestone.js";
|
|
3
5
|
import { loadWriteGateSnapshot, shouldBlockContextArtifactSaveInSnapshot, shouldBlockRootArtifactSaveInSnapshot } from "../bootstrap/write-gate.js";
|
|
@@ -12,6 +14,9 @@ import { handleCompleteSlice } from "./complete-slice.js";
|
|
|
12
14
|
import { handlePlanMilestone } from "./plan-milestone.js";
|
|
13
15
|
import { handlePlanSlice } from "./plan-slice.js";
|
|
14
16
|
import { handleReplanSlice } from "./replan-slice.js";
|
|
17
|
+
import { handleReopenMilestone } from "./reopen-milestone.js";
|
|
18
|
+
import { handleReopenSlice } from "./reopen-slice.js";
|
|
19
|
+
import { handleReopenTask } from "./reopen-task.js";
|
|
15
20
|
import { handleReassessRoadmap } from "./reassess-roadmap.js";
|
|
16
21
|
import { handleValidateMilestone } from "./validate-milestone.js";
|
|
17
22
|
import { logError, logWarning } from "../workflow-logger.js";
|
|
@@ -274,6 +279,120 @@ export async function executeTaskComplete(params, basePath = process.cwd()) {
|
|
|
274
279
|
};
|
|
275
280
|
}
|
|
276
281
|
}
|
|
282
|
+
export async function executeTaskReopen(params, basePath = process.cwd()) {
|
|
283
|
+
const dbAvailable = await ensureDbOpen(basePath);
|
|
284
|
+
if (!dbAvailable) {
|
|
285
|
+
return {
|
|
286
|
+
content: [{ type: "text", text: "Error: GSD database is not available. Cannot reopen task." }],
|
|
287
|
+
details: { operation: "reopen_task", error: "db_unavailable" },
|
|
288
|
+
isError: true,
|
|
289
|
+
};
|
|
290
|
+
}
|
|
291
|
+
try {
|
|
292
|
+
const result = await handleReopenTask(params, basePath);
|
|
293
|
+
if ("error" in result) {
|
|
294
|
+
return {
|
|
295
|
+
content: [{ type: "text", text: `Error reopening task: ${result.error}` }],
|
|
296
|
+
details: { operation: "reopen_task", error: result.error },
|
|
297
|
+
isError: true,
|
|
298
|
+
};
|
|
299
|
+
}
|
|
300
|
+
return {
|
|
301
|
+
content: [{ type: "text", text: `Reopened task ${result.taskId} (${result.sliceId}/${result.milestoneId})` }],
|
|
302
|
+
details: {
|
|
303
|
+
operation: "reopen_task",
|
|
304
|
+
taskId: result.taskId,
|
|
305
|
+
sliceId: result.sliceId,
|
|
306
|
+
milestoneId: result.milestoneId,
|
|
307
|
+
},
|
|
308
|
+
};
|
|
309
|
+
}
|
|
310
|
+
catch (err) {
|
|
311
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
312
|
+
logError("tool", `reopen_task tool failed: ${msg}`, { tool: "gsd_task_reopen", error: String(err) });
|
|
313
|
+
return {
|
|
314
|
+
content: [{ type: "text", text: `Error reopening task: ${msg}` }],
|
|
315
|
+
details: { operation: "reopen_task", error: msg },
|
|
316
|
+
isError: true,
|
|
317
|
+
};
|
|
318
|
+
}
|
|
319
|
+
}
|
|
320
|
+
export async function executeSliceReopen(params, basePath = process.cwd()) {
|
|
321
|
+
const dbAvailable = await ensureDbOpen(basePath);
|
|
322
|
+
if (!dbAvailable) {
|
|
323
|
+
return {
|
|
324
|
+
content: [{ type: "text", text: "Error: GSD database is not available. Cannot reopen slice." }],
|
|
325
|
+
details: { operation: "reopen_slice", error: "db_unavailable" },
|
|
326
|
+
isError: true,
|
|
327
|
+
};
|
|
328
|
+
}
|
|
329
|
+
try {
|
|
330
|
+
const result = await handleReopenSlice(params, basePath);
|
|
331
|
+
if ("error" in result) {
|
|
332
|
+
return {
|
|
333
|
+
content: [{ type: "text", text: `Error reopening slice: ${result.error}` }],
|
|
334
|
+
details: { operation: "reopen_slice", error: result.error },
|
|
335
|
+
isError: true,
|
|
336
|
+
};
|
|
337
|
+
}
|
|
338
|
+
return {
|
|
339
|
+
content: [{ type: "text", text: `Reopened slice ${result.sliceId} (${result.milestoneId})` }],
|
|
340
|
+
details: {
|
|
341
|
+
operation: "reopen_slice",
|
|
342
|
+
sliceId: result.sliceId,
|
|
343
|
+
milestoneId: result.milestoneId,
|
|
344
|
+
tasksReset: result.tasksReset,
|
|
345
|
+
},
|
|
346
|
+
};
|
|
347
|
+
}
|
|
348
|
+
catch (err) {
|
|
349
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
350
|
+
logError("tool", `reopen_slice tool failed: ${msg}`, { tool: "gsd_slice_reopen", error: String(err) });
|
|
351
|
+
return {
|
|
352
|
+
content: [{ type: "text", text: `Error reopening slice: ${msg}` }],
|
|
353
|
+
details: { operation: "reopen_slice", error: msg },
|
|
354
|
+
isError: true,
|
|
355
|
+
};
|
|
356
|
+
}
|
|
357
|
+
}
|
|
358
|
+
export async function executeMilestoneReopen(params, basePath = process.cwd()) {
|
|
359
|
+
const dbAvailable = await ensureDbOpen(basePath);
|
|
360
|
+
if (!dbAvailable) {
|
|
361
|
+
return {
|
|
362
|
+
content: [{ type: "text", text: "Error: GSD database is not available. Cannot reopen milestone." }],
|
|
363
|
+
details: { operation: "reopen_milestone", error: "db_unavailable" },
|
|
364
|
+
isError: true,
|
|
365
|
+
};
|
|
366
|
+
}
|
|
367
|
+
try {
|
|
368
|
+
const result = await handleReopenMilestone(params, basePath);
|
|
369
|
+
if ("error" in result) {
|
|
370
|
+
return {
|
|
371
|
+
content: [{ type: "text", text: `Error reopening milestone: ${result.error}` }],
|
|
372
|
+
details: { operation: "reopen_milestone", error: result.error },
|
|
373
|
+
isError: true,
|
|
374
|
+
};
|
|
375
|
+
}
|
|
376
|
+
return {
|
|
377
|
+
content: [{ type: "text", text: `Reopened milestone ${result.milestoneId}` }],
|
|
378
|
+
details: {
|
|
379
|
+
operation: "reopen_milestone",
|
|
380
|
+
milestoneId: result.milestoneId,
|
|
381
|
+
slicesReset: result.slicesReset,
|
|
382
|
+
tasksReset: result.tasksReset,
|
|
383
|
+
},
|
|
384
|
+
};
|
|
385
|
+
}
|
|
386
|
+
catch (err) {
|
|
387
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
388
|
+
logError("tool", `reopen_milestone tool failed: ${msg}`, { tool: "gsd_milestone_reopen", error: String(err) });
|
|
389
|
+
return {
|
|
390
|
+
content: [{ type: "text", text: `Error reopening milestone: ${msg}` }],
|
|
391
|
+
details: { operation: "reopen_milestone", error: msg },
|
|
392
|
+
isError: true,
|
|
393
|
+
};
|
|
394
|
+
}
|
|
395
|
+
}
|
|
277
396
|
export async function executeSliceComplete(params, basePath = process.cwd()) {
|
|
278
397
|
const dbAvailable = await ensureDbOpen(basePath);
|
|
279
398
|
if (!dbAvailable) {
|
|
@@ -227,7 +227,9 @@ export const UNIT_MANIFESTS = {
|
|
|
227
227
|
codebaseMap: true,
|
|
228
228
|
preferences: "active-only",
|
|
229
229
|
contextMode: "research",
|
|
230
|
-
|
|
230
|
+
// Multi-slice research dispatches use the research-slice unit contract to
|
|
231
|
+
// fan out scout subagents that write .gsd research artifacts.
|
|
232
|
+
tools: TOOLS_PLANNING_DISPATCH_RECON,
|
|
231
233
|
artifacts: {
|
|
232
234
|
inline: ["roadmap", "milestone-research", "dependency-summaries", "templates"],
|
|
233
235
|
excerpt: [],
|
|
@@ -384,7 +386,9 @@ export const UNIT_MANIFESTS = {
|
|
|
384
386
|
codebaseMap: false,
|
|
385
387
|
preferences: "active-only",
|
|
386
388
|
contextMode: "verification",
|
|
387
|
-
|
|
389
|
+
// Gate evaluation fans out tester-style subagents, which read the slice
|
|
390
|
+
// plan and report via the DB-backed gate-result tool.
|
|
391
|
+
tools: TOOLS_PLANNING_DISPATCH_REVIEW,
|
|
388
392
|
artifacts: {
|
|
389
393
|
inline: ["slice-plan", "prior-task-summaries"],
|
|
390
394
|
excerpt: [],
|
|
@@ -509,3 +513,22 @@ export const UNIT_MANIFESTS = {
|
|
|
509
513
|
export function resolveManifest(unitType) {
|
|
510
514
|
return UNIT_MANIFESTS[unitType] ?? null;
|
|
511
515
|
}
|
|
516
|
+
export function compileSubagentPermissionContract(policy) {
|
|
517
|
+
if (!policy) {
|
|
518
|
+
return { allowed: false, allowedSubagents: [], toolsMode: "unknown" };
|
|
519
|
+
}
|
|
520
|
+
if (policy.mode === "all") {
|
|
521
|
+
return { allowed: true, allowedSubagents: ["*"], toolsMode: policy.mode };
|
|
522
|
+
}
|
|
523
|
+
if (policy.mode === "planning-dispatch") {
|
|
524
|
+
return {
|
|
525
|
+
allowed: true,
|
|
526
|
+
allowedSubagents: [...policy.allowedSubagents],
|
|
527
|
+
toolsMode: policy.mode,
|
|
528
|
+
};
|
|
529
|
+
}
|
|
530
|
+
return { allowed: false, allowedSubagents: [], toolsMode: policy.mode };
|
|
531
|
+
}
|
|
532
|
+
export function resolveSubagentPermissionContract(unitType) {
|
|
533
|
+
return compileSubagentPermissionContract(resolveManifest(unitType)?.tools);
|
|
534
|
+
}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
// Project/App: GSD-2
|
|
2
|
+
// File Purpose: Host-owned verification verdict policy for auto-mode units.
|
|
3
|
+
export function decideVerificationVerdict(unitType, result) {
|
|
4
|
+
if (unitType === "execute-task" && result.discoverySource === "none" && result.checks.length === 0) {
|
|
5
|
+
return {
|
|
6
|
+
passed: false,
|
|
7
|
+
reason: "no-host-checks",
|
|
8
|
+
retryable: false,
|
|
9
|
+
failureContext: "No runnable host-owned verification command was discovered. Add project verification_commands or a runnable task-plan Verify command before completing this execute-task.",
|
|
10
|
+
};
|
|
11
|
+
}
|
|
12
|
+
if (!result.passed) {
|
|
13
|
+
return {
|
|
14
|
+
passed: false,
|
|
15
|
+
reason: "checks-failed",
|
|
16
|
+
retryable: true,
|
|
17
|
+
failureContext: "",
|
|
18
|
+
};
|
|
19
|
+
}
|
|
20
|
+
return {
|
|
21
|
+
passed: true,
|
|
22
|
+
reason: "passed",
|
|
23
|
+
retryable: false,
|
|
24
|
+
failureContext: "",
|
|
25
|
+
};
|
|
26
|
+
}
|
|
@@ -1198,17 +1198,36 @@ export class WorktreeLifecycle {
|
|
|
1198
1198
|
}
|
|
1199
1199
|
}
|
|
1200
1200
|
/**
|
|
1201
|
-
* Restore `s.basePath` to `s.originalBasePath
|
|
1202
|
-
* No-op when `originalBasePath` is empty (fresh
|
|
1201
|
+
* Restore `s.basePath` to `s.originalBasePath`, chdir process cwd, and
|
|
1202
|
+
* rebuild `s.gitService`. No-op when `originalBasePath` is empty (fresh
|
|
1203
|
+
* sessions).
|
|
1203
1204
|
*
|
|
1204
1205
|
* Used by error/cleanup paths that need the session to behave as if the
|
|
1205
1206
|
* worktree was never entered. Does NOT teardown the worktree directory —
|
|
1206
1207
|
* callers that need teardown go through `exitMilestone({ merge: false })`.
|
|
1208
|
+
*
|
|
1209
|
+
* ADR-016 phase 3 (#5693): chdir lives inside the verb so callers do not
|
|
1210
|
+
* pair `restoreToProjectRoot()` with a redundant `process.chdir`. The
|
|
1211
|
+
* chdir runs BEFORE the throwable work (`rebuildGitService`, cache
|
|
1212
|
+
* invalidation) so that cleanup-path cwd is restored even if the
|
|
1213
|
+
* downstream rebuild throws. The chdir itself is best-effort; failure is
|
|
1214
|
+
* logged via debugLog and swallowed.
|
|
1207
1215
|
*/
|
|
1208
1216
|
restoreToProjectRoot() {
|
|
1209
1217
|
if (!this.s.originalBasePath)
|
|
1210
1218
|
return;
|
|
1211
1219
|
this.s.basePath = this.s.originalBasePath;
|
|
1220
|
+
try {
|
|
1221
|
+
process.chdir(this.s.basePath);
|
|
1222
|
+
}
|
|
1223
|
+
catch (err) {
|
|
1224
|
+
debugLog("WorktreeLifecycle", {
|
|
1225
|
+
action: "restoreToProjectRoot",
|
|
1226
|
+
result: "chdir-failed",
|
|
1227
|
+
basePath: this.s.basePath,
|
|
1228
|
+
error: err instanceof Error ? err.message : String(err),
|
|
1229
|
+
});
|
|
1230
|
+
}
|
|
1212
1231
|
rebuildGitService(this.s, this.deps);
|
|
1213
1232
|
invalidateAllCaches();
|
|
1214
1233
|
}
|