gsd-pi 2.73.0-dev.e1c09f2 → 2.73.1-dev.088d28f

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.
Files changed (149) hide show
  1. package/dist/resource-loader.js +2 -2
  2. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +9 -3
  3. package/dist/resources/extensions/gsd/auto/phases.js +15 -9
  4. package/dist/resources/extensions/gsd/auto-dispatch.js +11 -3
  5. package/dist/resources/extensions/gsd/auto-model-selection.js +54 -11
  6. package/dist/resources/extensions/gsd/auto-start.js +23 -6
  7. package/dist/resources/extensions/gsd/auto.js +13 -1
  8. package/dist/resources/extensions/gsd/bootstrap/crash-log.js +31 -0
  9. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +18 -7
  10. package/dist/resources/extensions/gsd/commands-handlers.js +8 -2
  11. package/dist/resources/extensions/gsd/crash-recovery.js +51 -0
  12. package/dist/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  13. package/dist/resources/extensions/gsd/gsd-db.js +36 -2
  14. package/dist/resources/extensions/gsd/milestone-actions.js +19 -1
  15. package/dist/resources/extensions/gsd/notification-widget.js +2 -2
  16. package/dist/resources/extensions/gsd/preferences-models.js +43 -0
  17. package/dist/resources/extensions/gsd/preferences-types.js +1 -0
  18. package/dist/resources/extensions/gsd/preferences-validation.js +22 -0
  19. package/dist/resources/extensions/gsd/state.js +36 -0
  20. package/dist/update-check.d.ts +1 -0
  21. package/dist/update-check.js +13 -5
  22. package/dist/update-cmd.js +4 -3
  23. package/dist/web/standalone/.next/BUILD_ID +1 -1
  24. package/dist/web/standalone/.next/app-path-routes-manifest.json +11 -11
  25. package/dist/web/standalone/.next/build-manifest.json +2 -2
  26. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  27. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  28. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  29. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  30. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  31. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  32. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  33. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  34. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  35. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  36. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  37. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  38. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  39. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  40. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  41. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  42. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  43. package/dist/web/standalone/.next/server/app/index.html +1 -1
  44. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  45. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  46. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  47. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  48. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  49. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  50. package/dist/web/standalone/.next/server/app-paths-manifest.json +11 -11
  51. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  52. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  53. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  54. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  55. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  56. package/package.json +1 -1
  57. package/packages/pi-ai/dist/index.d.ts +1 -0
  58. package/packages/pi-ai/dist/index.d.ts.map +1 -1
  59. package/packages/pi-ai/dist/index.js +1 -0
  60. package/packages/pi-ai/dist/index.js.map +1 -1
  61. package/packages/pi-ai/dist/utils/overflow.d.ts.map +1 -1
  62. package/packages/pi-ai/dist/utils/overflow.js +12 -0
  63. package/packages/pi-ai/dist/utils/overflow.js.map +1 -1
  64. package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts +2 -0
  65. package/packages/pi-ai/dist/utils/tests/overflow.test.d.ts.map +1 -0
  66. package/packages/pi-ai/dist/utils/tests/overflow.test.js +50 -0
  67. package/packages/pi-ai/dist/utils/tests/overflow.test.js.map +1 -0
  68. package/packages/pi-ai/src/index.ts +4 -0
  69. package/packages/pi-ai/src/utils/overflow.ts +14 -1
  70. package/packages/pi-ai/src/utils/tests/overflow.test.ts +58 -0
  71. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +313 -8
  72. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  73. package/packages/pi-coding-agent/dist/core/compaction/utils.js +5 -5
  74. package/packages/pi-coding-agent/dist/core/compaction/utils.js.map +1 -1
  75. package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts +2 -0
  76. package/packages/pi-coding-agent/dist/core/compaction-utils.test.d.ts.map +1 -0
  77. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js +45 -0
  78. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js.map +1 -0
  79. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +12 -2
  80. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
  81. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +51 -26
  82. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
  83. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts +2 -1
  84. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -1
  85. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js +9 -3
  86. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js.map +1 -1
  87. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts +2 -0
  88. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.d.ts.map +1 -0
  89. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js +52 -0
  90. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js.map +1 -0
  91. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  92. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +94 -16
  93. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  94. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  95. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +11 -3
  96. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  97. package/packages/pi-coding-agent/package.json +1 -1
  98. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +355 -8
  99. package/packages/pi-coding-agent/src/core/compaction/utils.ts +5 -5
  100. package/packages/pi-coding-agent/src/core/compaction-utils.test.ts +50 -0
  101. package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +62 -26
  102. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.test.ts +73 -0
  103. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.ts +9 -3
  104. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +113 -21
  105. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +11 -3
  106. package/packages/pi-tui/dist/__tests__/tui.test.js +60 -1
  107. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  108. package/packages/pi-tui/dist/tui.d.ts +8 -0
  109. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  110. package/packages/pi-tui/dist/tui.js +32 -3
  111. package/packages/pi-tui/dist/tui.js.map +1 -1
  112. package/packages/pi-tui/src/__tests__/tui.test.ts +76 -1
  113. package/packages/pi-tui/src/tui.ts +31 -3
  114. package/pkg/package.json +1 -1
  115. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +12 -4
  116. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +23 -2
  117. package/src/resources/extensions/gsd/auto/phases.ts +22 -9
  118. package/src/resources/extensions/gsd/auto-dispatch.ts +10 -4
  119. package/src/resources/extensions/gsd/auto-model-selection.ts +85 -11
  120. package/src/resources/extensions/gsd/auto-start.ts +30 -6
  121. package/src/resources/extensions/gsd/auto.ts +10 -0
  122. package/src/resources/extensions/gsd/bootstrap/crash-log.ts +32 -0
  123. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +19 -7
  124. package/src/resources/extensions/gsd/commands-handlers.ts +8 -2
  125. package/src/resources/extensions/gsd/crash-recovery.ts +59 -0
  126. package/src/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  127. package/src/resources/extensions/gsd/gsd-db.ts +52 -2
  128. package/src/resources/extensions/gsd/milestone-actions.ts +19 -1
  129. package/src/resources/extensions/gsd/notification-widget.ts +2 -2
  130. package/src/resources/extensions/gsd/preferences-models.ts +41 -0
  131. package/src/resources/extensions/gsd/preferences-types.ts +12 -0
  132. package/src/resources/extensions/gsd/preferences-validation.ts +23 -0
  133. package/src/resources/extensions/gsd/state.ts +46 -0
  134. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +2 -2
  135. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +51 -2
  136. package/src/resources/extensions/gsd/tests/crash-handler-secondary.test.ts +235 -0
  137. package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +3 -2
  138. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +3 -2
  139. package/src/resources/extensions/gsd/tests/derive-state.test.ts +3 -3
  140. package/src/resources/extensions/gsd/tests/flat-rate-routing-guard.test.ts +137 -1
  141. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +59 -1
  142. package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +4 -2
  143. package/src/resources/extensions/gsd/tests/model-isolation.test.ts +91 -2
  144. package/src/resources/extensions/gsd/tests/park-milestone.test.ts +64 -0
  145. package/src/resources/extensions/gsd/tests/preferences.test.ts +47 -0
  146. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +5 -7
  147. package/src/resources/extensions/gsd/tests/token-profile.test.ts +1 -1
  148. /package/dist/web/standalone/.next/static/{_XD_gUDcZNBbWV5rI8RgS → nwYTvJZ1-hZIfw98d9Wfg}/_buildManifest.js +0 -0
  149. /package/dist/web/standalone/.next/static/{_XD_gUDcZNBbWV5rI8RgS → nwYTvJZ1-hZIfw98d9Wfg}/_ssgManifest.js +0 -0
@@ -307,8 +307,11 @@ export const DISPATCH_RULES: DispatchRule[] = [
307
307
  {
308
308
  name: "reassess-roadmap (post-completion)",
309
309
  match: async ({ state, mid, midTitle, basePath, prefs }) => {
310
- if (prefs?.phases?.skip_reassess || !prefs?.phases?.reassess_after_slice)
311
- return null;
310
+ if (prefs?.phases?.skip_reassess) return null;
311
+ // Default reassess_after_slice to true — reassessment after slice completion
312
+ // is essential for roadmap integrity. Opt-out via explicit `false`.
313
+ const reassessEnabled = prefs?.phases?.reassess_after_slice ?? true;
314
+ if (!reassessEnabled) return null;
312
315
  const needsReassess = await checkNeedsReassessment(basePath, mid, state);
313
316
  if (!needsReassess) return null;
314
317
  return {
@@ -877,11 +880,14 @@ export async function resolveDispatch(
877
880
  }
878
881
  }
879
882
 
880
- // No rule matched — unhandled phase
883
+ // No rule matched — unhandled phase.
884
+ // Use level "warning" so the loop pauses (resumable) instead of hard-stopping.
885
+ // Hard-stop here was causing premature termination for transient phase gaps
886
+ // (e.g. after reassessment modifies the roadmap and state needs re-derivation).
881
887
  return {
882
888
  action: "stop",
883
889
  reason: `Unhandled phase "${ctx.state.phase}" — run /gsd doctor to diagnose.`,
884
- level: "info",
890
+ level: "warning",
885
891
  matchedRule: "<no-match>",
886
892
  };
887
893
  }
@@ -15,6 +15,7 @@ import { resolveModelForComplexity, escalateTier, getEligibleModels, loadCapabil
15
15
  import { getLedger, getProjectTotals } from "./metrics.js";
16
16
  import { unitPhaseLabel } from "./auto-dashboard.js";
17
17
  import { getSessionModelOverride } from "./session-model-override.js";
18
+ import { logWarning } from "./workflow-logger.js";
18
19
 
19
20
  export interface ModelSelectionResult {
20
21
  /** Routing metadata for metrics recording */
@@ -25,9 +26,7 @@ export interface ModelSelectionResult {
25
26
 
26
27
  export function resolvePreferredModelConfig(
27
28
  unitType: string,
28
- autoModeStartModel: { provider: string; id: string } | null,
29
- /** When false, only return explicit per-phase model configs — do not
30
- * synthesize a routing ceiling from dynamic_routing.tier_models (#3962). */
29
+ autoModeStartModel: { provider: string; id: string; flatRateCtx?: FlatRateContext } | null,
31
30
  isAutoMode = true,
32
31
  ) {
33
32
  const explicitConfig = resolveModelWithFallbacksForUnit(unitType);
@@ -41,7 +40,7 @@ export function resolvePreferredModelConfig(
41
40
  if (!routingConfig.enabled || !routingConfig.tier_models) return undefined;
42
41
 
43
42
  // Don't synthesize a routing config for flat-rate providers (#3453).
44
- if (autoModeStartModel && isFlatRateProvider(autoModeStartModel.provider)) return undefined;
43
+ if (autoModeStartModel && isFlatRateProvider(autoModeStartModel.provider, autoModeStartModel.flatRateCtx)) return undefined;
45
44
 
46
45
  const ceilingModel = routingConfig.tier_models.heavy
47
46
  ?? (autoModeStartModel ? `${autoModeStartModel.provider}/${autoModeStartModel.id}` : undefined);
@@ -68,7 +67,7 @@ export async function selectAndApplyModel(
68
67
  basePath: string,
69
68
  prefs: GSDPreferences | undefined,
70
69
  verbose: boolean,
71
- autoModeStartModel: { provider: string; id: string } | null,
70
+ autoModeStartModel: { provider: string; id: string; flatRateCtx?: FlatRateContext } | null,
72
71
  retryContext?: { isRetry: boolean; previousTier?: string },
73
72
  /** When false (interactive/guided-flow), skip dynamic routing and use the session model.
74
73
  * Dynamic routing only applies in auto-mode where cost optimization is expected. (#3962) */
@@ -79,6 +78,17 @@ export async function selectAndApplyModel(
79
78
  const effectiveSessionModelOverride = sessionModelOverride === undefined
80
79
  ? getSessionModelOverride(ctx.sessionManager.getSessionId())
81
80
  : (sessionModelOverride ?? undefined);
81
+ // Enrich the start model with a flat-rate context up front so routing
82
+ // synthesis and the dispatch-time guard see the same signals (built-in
83
+ // list + user `flat_rate_providers` preference + externalCli auto-
84
+ // detection). The dispatch-time primary-model check below builds its
85
+ // own per-provider context when it has a resolved primary model.
86
+ if (autoModeStartModel) {
87
+ autoModeStartModel = {
88
+ ...autoModeStartModel,
89
+ flatRateCtx: buildFlatRateContext(autoModeStartModel.provider, ctx, prefs),
90
+ };
91
+ }
82
92
  const modelConfig = effectiveSessionModelOverride
83
93
  ? undefined
84
94
  : resolvePreferredModelConfig(unitType, autoModeStartModel, isAutoMode);
@@ -107,12 +117,16 @@ export async function selectAndApplyModel(
107
117
  if (routingConfig.enabled) {
108
118
  const primaryModel = resolveModelId(modelConfig.primary, availableModels, ctx.model?.provider);
109
119
  if (primaryModel) {
110
- if (isFlatRateProvider(primaryModel.provider)) {
120
+ const primaryFlatRateCtx = buildFlatRateContext(primaryModel.provider, ctx, prefs);
121
+ if (isFlatRateProvider(primaryModel.provider, primaryFlatRateCtx)) {
111
122
  routingConfig.enabled = false;
112
123
  }
113
124
  } else if (
114
- (autoModeStartModel && isFlatRateProvider(autoModeStartModel.provider))
115
- || (ctx.model?.provider && isFlatRateProvider(ctx.model.provider))
125
+ (autoModeStartModel && isFlatRateProvider(autoModeStartModel.provider, autoModeStartModel.flatRateCtx))
126
+ || (ctx.model?.provider && isFlatRateProvider(
127
+ ctx.model.provider,
128
+ buildFlatRateContext(ctx.model.provider, ctx, prefs),
129
+ ))
116
130
  ) {
117
131
  // Primary model unresolvable but provider signals indicate flat-rate —
118
132
  // disable routing to prevent quality degradation.
@@ -416,8 +430,68 @@ export function resolveModelId<T extends { id: string; provider: string }>(
416
430
  * Uses case-insensitive matching with alias support to prevent fail-open on
417
431
  * provider naming variations (e.g. "copilot" vs "github-copilot").
418
432
  */
419
- const FLAT_RATE_PROVIDERS = new Set(["github-copilot", "copilot", "claude-code"]);
433
+ const BUILTIN_FLAT_RATE = new Set(["github-copilot", "copilot", "claude-code"]);
434
+
435
+ /**
436
+ * Optional context that lets callers extend flat-rate detection beyond the
437
+ * hard-coded built-in list. Either signal on its own is enough to classify
438
+ * a provider as flat-rate.
439
+ */
440
+ export interface FlatRateContext {
441
+ /**
442
+ * Auth mode for the specific provider being checked, as returned by
443
+ * `ctx.modelRegistry.getProviderAuthMode(provider)`. Any provider that
444
+ * wraps a local CLI (externalCli) is, by definition, a flat-rate
445
+ * subscription wrapper — every request costs the same regardless of
446
+ * model, so dynamic routing only degrades quality.
447
+ */
448
+ authMode?: "apiKey" | "oauth" | "externalCli" | "none";
449
+ /**
450
+ * Case-insensitive list of extra provider IDs the user has declared as
451
+ * flat-rate via `preferences.flat_rate_providers`. Used for private
452
+ * subscription-backed proxies and enterprise-gated deployments that the
453
+ * built-in list doesn't know about.
454
+ */
455
+ userFlatRate?: readonly string[];
456
+ }
457
+
458
+ export function isFlatRateProvider(provider: string, opts?: FlatRateContext): boolean {
459
+ const p = provider.toLowerCase();
460
+ if (BUILTIN_FLAT_RATE.has(p)) return true;
461
+ if (opts?.userFlatRate?.some(id => id.toLowerCase() === p)) return true;
462
+ if (opts?.authMode === "externalCli") return true;
463
+ return false;
464
+ }
420
465
 
421
- export function isFlatRateProvider(provider: string): boolean {
422
- return FLAT_RATE_PROVIDERS.has(provider.toLowerCase());
466
+ /**
467
+ * Build a FlatRateContext for a given provider from live runtime state.
468
+ * Safe to call when ctx or prefs are undefined — missing pieces are
469
+ * treated as "no signal".
470
+ */
471
+ export function buildFlatRateContext(
472
+ provider: string,
473
+ ctx?: { modelRegistry?: { getProviderAuthMode?: (p: string) => string } },
474
+ prefs?: { flat_rate_providers?: readonly string[] },
475
+ ): FlatRateContext {
476
+ let authMode: FlatRateContext["authMode"];
477
+ const getAuthMode = ctx?.modelRegistry?.getProviderAuthMode;
478
+ if (typeof getAuthMode === "function") {
479
+ try {
480
+ const mode = getAuthMode(provider);
481
+ if (mode === "apiKey" || mode === "oauth" || mode === "externalCli" || mode === "none") {
482
+ authMode = mode;
483
+ }
484
+ } catch (err) {
485
+ // Registry lookup failure must never break flat-rate detection —
486
+ // fall through with authMode undefined and surface the cause.
487
+ logWarning(
488
+ "dispatch",
489
+ `flat-rate auth-mode lookup failed for ${provider}: ${err instanceof Error ? err.message : String(err)}`,
490
+ );
491
+ }
492
+ }
493
+ return {
494
+ authMode,
495
+ userFlatRate: prefs?.flat_rate_providers,
496
+ };
423
497
  }
@@ -83,7 +83,11 @@ import { join } from "node:path";
83
83
  import { sep as pathSep } from "node:path";
84
84
 
85
85
  import { resolveProjectRootDbPath } from "./bootstrap/dynamic-tools.js";
86
- import { resolveDefaultSessionModel, resolveDynamicRoutingConfig } from "./preferences-models.js";
86
+ import {
87
+ isCustomProvider,
88
+ resolveDefaultSessionModel,
89
+ resolveDynamicRoutingConfig,
90
+ } from "./preferences-models.js";
87
91
  import type { WorktreeResolver } from "./worktree-resolver.js";
88
92
  import { getSessionModelOverride } from "./session-model-override.js";
89
93
 
@@ -274,8 +278,18 @@ export async function bootstrapAutoSession(
274
278
  //
275
279
  // This preserves #3517 defaults while honoring explicit runtime model
276
280
  // selection for subsequent /gsd runs in the same session.
281
+ //
282
+ // Exception (#4122): when the session provider is a custom provider declared
283
+ // in ~/.gsd/agent/models.json (Ollama, vLLM, OpenAI-compatible proxy, etc.),
284
+ // PREFERENCES.md is skipped entirely. PREFERENCES.md cannot reference custom
285
+ // providers, so honoring it would silently reroute auto-mode to a built-in
286
+ // provider the user is not logged into and surface as "Not logged in · Please
287
+ // run /login" before pausing and resetting to claude-code/claude-sonnet-4-6.
277
288
  const manualSessionOverride = getSessionModelOverride(ctx.sessionManager.getSessionId());
278
- const preferredModel = resolveDefaultSessionModel(ctx.model?.provider);
289
+ const sessionProviderIsCustom = isCustomProvider(ctx.model?.provider);
290
+ const preferredModel = sessionProviderIsCustom
291
+ ? null
292
+ : resolveDefaultSessionModel(ctx.model?.provider);
279
293
  // Validate the preferred model against the live registry + provider auth so
280
294
  // an unconfigured PREFERENCES.md entry (no API key / OAuth) can't become the
281
295
  // start-model snapshot. Without this, every subsequent unit would try to
@@ -792,6 +806,9 @@ export async function bootstrapAutoSession(
792
806
 
793
807
  ctx.ui.setStatus("gsd-auto", s.stepMode ? "next" : "auto");
794
808
  ctx.ui.setFooter(hideFooter);
809
+ // Hide gsd-health during AUTO — gsd-progress is the single source of truth
810
+ // for last-commit / cost / health signal while auto is running.
811
+ ctx.ui.setWidget("gsd-health", undefined);
795
812
  const modeLabel = s.stepMode ? "Step-mode" : "Auto-mode";
796
813
  const pendingCount = (state.registry ?? []).filter(
797
814
  (m) => m.status !== "complete" && m.status !== "parked",
@@ -811,12 +828,19 @@ export async function bootstrapAutoSession(
811
828
  ? `${s.autoModeStartModel.provider}/${s.autoModeStartModel.id}`
812
829
  : ctx.model ? `${ctx.model.provider}/${ctx.model.id}` : "default";
813
830
 
814
- // Flat-rate providers (e.g. GitHub Copilot, claude-code) suppress routing
815
- // at dispatch time (#3453) reflect that in the banner.
816
- const { isFlatRateProvider } = await import("./auto-model-selection.js");
831
+ // Flat-rate providers (e.g. GitHub Copilot, claude-code, user-declared
832
+ // subscription proxies, externalCli CLIs) suppress routing at dispatch
833
+ // time (#3453) reflect that in the banner. Thread the same
834
+ // FlatRateContext used by selectAndApplyModel so user-declared
835
+ // flat-rate providers and externalCli auto-detection are respected.
836
+ const { isFlatRateProvider, buildFlatRateContext } = await import("./auto-model-selection.js");
837
+ const bannerPrefs = loadEffectiveGSDPreferences()?.preferences;
817
838
  const effectiveProvider = s.autoModeStartModel?.provider ?? ctx.model?.provider;
818
839
  const effectivelyEnabled = routingConfig.enabled
819
- && !(effectiveProvider && isFlatRateProvider(effectiveProvider));
840
+ && !(effectiveProvider && isFlatRateProvider(
841
+ effectiveProvider,
842
+ buildFlatRateContext(effectiveProvider, ctx, bannerPrefs),
843
+ ));
820
844
 
821
845
  // The actual ceiling may come from tier_models.heavy, not the start model.
822
846
  const effectiveCeiling = (routingConfig.enabled && routingConfig.tier_models?.heavy)
@@ -52,6 +52,7 @@ import {
52
52
  readCrashLock,
53
53
  isLockProcessAlive,
54
54
  formatCrashInfo,
55
+ emitCrashRecoveredUnitEnd,
55
56
  } from "./crash-recovery.js";
56
57
  import {
57
58
  acquireSessionLock,
@@ -198,6 +199,7 @@ import {
198
199
  postUnitPostVerification,
199
200
  } from "./auto-post-unit.js";
200
201
  import { bootstrapAutoSession, openProjectDbIfPresent, type BootstrapDeps } from "./auto-start.js";
202
+ import { initHealthWidget } from "./health-widget.js";
201
203
  import { autoLoop, resolveAgentEnd, resolveAgentEndCancelled, _resetPendingResolve, isSessionSwitchInFlight, type LoopDeps, type ErrorContext } from "./auto-loop.js";
202
204
  // Slice-level parallelism (#2340)
203
205
  import { getEligibleSlices } from "./slice-parallel-eligibility.js";
@@ -649,6 +651,7 @@ function handleLostSessionLock(
649
651
  ctx?.ui.setStatus("gsd-auto", undefined);
650
652
  ctx?.ui.setWidget("gsd-progress", undefined);
651
653
  ctx?.ui.setFooter(undefined);
654
+ if (ctx) initHealthWidget(ctx);
652
655
  }
653
656
 
654
657
  /**
@@ -683,6 +686,7 @@ function cleanupAfterLoopExit(ctx: ExtensionContext): void {
683
686
  ctx.ui.setStatus("gsd-auto", undefined);
684
687
  ctx.ui.setWidget("gsd-progress", undefined);
685
688
  ctx.ui.setFooter(undefined);
689
+ initHealthWidget(ctx);
686
690
  }
687
691
 
688
692
  // Restore CWD out of worktree back to original project root
@@ -942,6 +946,7 @@ export async function stopAuto(
942
946
  ctx?.ui.setStatus("gsd-auto", undefined);
943
947
  ctx?.ui.setWidget("gsd-progress", undefined);
944
948
  ctx?.ui.setFooter(undefined);
949
+ if (ctx) initHealthWidget(ctx);
945
950
  restoreProjectRootEnv();
946
951
  restoreMilestoneLockEnv();
947
952
 
@@ -1043,6 +1048,7 @@ export async function pauseAuto(
1043
1048
  ctx?.ui.setStatus("gsd-auto", "paused");
1044
1049
  ctx?.ui.setWidget("gsd-progress", undefined);
1045
1050
  ctx?.ui.setFooter(undefined);
1051
+ if (ctx) initHealthWidget(ctx);
1046
1052
  const resumeCmd = s.stepMode ? "/gsd next" : "/gsd auto";
1047
1053
  ctx?.ui.notify(
1048
1054
  `${s.stepMode ? "Step" : "Auto"}-mode paused (Escape). Type to interact, or ${resumeCmd} to resume.`,
@@ -1332,6 +1338,10 @@ export async function startAuto(
1332
1338
  }
1333
1339
 
1334
1340
  if (freshStartAssessment.lock) {
1341
+ // Emit a synthetic unit-end for any unit-start that has no closing event.
1342
+ // This closes the journal gap reported in #3348 where the worker wrote side
1343
+ // effects (SUMMARY.md, DB updates) but died before emitting unit-end.
1344
+ emitCrashRecoveredUnitEnd(base, freshStartAssessment.lock);
1335
1345
  clearLock(base);
1336
1346
  }
1337
1347
 
@@ -0,0 +1,32 @@
1
+ /**
2
+ * crash-log.ts — Write crash diagnostics to ~/.gsd/crash/<timestamp>.log
3
+ *
4
+ * Zero cross-dependencies: only uses Node.js built-ins so it can be imported
5
+ * safely from uncaughtException / unhandledRejection handlers and from tests
6
+ * without pulling in the full extension dependency tree.
7
+ */
8
+
9
+ import { appendFileSync, mkdirSync } from "node:fs";
10
+ import { homedir } from "node:os";
11
+ import { join } from "node:path";
12
+
13
+ /**
14
+ * Write a crash log to ~/.gsd/crash/<timestamp>.log (or $GSD_HOME/crash/).
15
+ * Never throws — must be safe to call from any error handler.
16
+ */
17
+ export function writeCrashLog(err: Error, source: string): void {
18
+ try {
19
+ const crashDir = join(process.env.GSD_HOME ?? join(homedir(), ".gsd"), "crash");
20
+ mkdirSync(crashDir, { recursive: true });
21
+ const ts = new Date().toISOString().replace(/[:.]/g, "-");
22
+ const logPath = join(crashDir, `${ts}.log`);
23
+ const lines = [
24
+ `[gsd] ${source}: ${err.message}`,
25
+ `timestamp: ${new Date().toISOString()}`,
26
+ `pid: ${process.pid}`,
27
+ err.stack ?? "(no stack trace available)",
28
+ "",
29
+ ];
30
+ appendFileSync(logPath, lines.join("\n"));
31
+ } catch { /* never throw from crash handler */ }
32
+ }
@@ -11,6 +11,9 @@ import { registerJournalTools } from "./journal-tools.js";
11
11
  import { registerQueryTools } from "./query-tools.js";
12
12
  import { registerHooks } from "./register-hooks.js";
13
13
  import { registerShortcuts } from "./register-shortcuts.js";
14
+ import { writeCrashLog } from "./crash-log.js";
15
+
16
+ export { writeCrashLog } from "./crash-log.js";
14
17
 
15
18
  export function handleRecoverableExtensionProcessError(err: Error): boolean {
16
19
  if ((err as NodeJS.ErrnoException).code === "EPIPE") {
@@ -33,16 +36,25 @@ export function handleRecoverableExtensionProcessError(err: Error): boolean {
33
36
  function installEpipeGuard(): void {
34
37
  if (!process.listeners("uncaughtException").some((listener) => listener.name === "_gsdEpipeGuard")) {
35
38
  const _gsdEpipeGuard = (err: Error): void => {
36
- if (handleRecoverableExtensionProcessError(err)) {
37
- return;
38
- }
39
- // Log unhandled errors instead of re-throwing throwing inside an
40
- // uncaughtException handler is a fatal double-fault in Node.js (#3163).
41
- process.stderr.write(`[gsd] uncaught extension error (non-fatal): ${err.message}\n`);
42
- if (err.stack) process.stderr.write(`${err.stack}\n`);
39
+ if (handleRecoverableExtensionProcessError(err)) return;
40
+ // Write crash log and exit cleanly for unrecoverable errors.
41
+ // Logging and continuing was the original double-fault fix (#3163), but
42
+ // continuing in an indeterminate state is worse than a clean exit (#3348).
43
+ writeCrashLog(err, "uncaughtException");
44
+ process.exit(1);
43
45
  };
44
46
  process.on("uncaughtException", _gsdEpipeGuard);
45
47
  }
48
+
49
+ if (!process.listeners("unhandledRejection").some((listener) => listener.name === "_gsdRejectionGuard")) {
50
+ const _gsdRejectionGuard = (reason: unknown, _promise: Promise<unknown>): void => {
51
+ const err = reason instanceof Error ? reason : new Error(String(reason));
52
+ if (handleRecoverableExtensionProcessError(err)) return;
53
+ writeCrashLog(err, "unhandledRejection");
54
+ process.exit(1);
55
+ };
56
+ process.on("unhandledRejection", _gsdRejectionGuard);
57
+ }
46
58
  }
47
59
 
48
60
  export function registerGsdExtension(pi: ExtensionAPI): void {
@@ -28,6 +28,11 @@ import { loadPrompt } from "./prompt-loader.js";
28
28
  const UPDATE_REGISTRY_URL = "https://registry.npmjs.org/gsd-pi/latest";
29
29
  const UPDATE_FETCH_TIMEOUT_MS = 5000;
30
30
 
31
+ function resolveInstallCommand(pkg: string): string {
32
+ if ('bun' in process.versions) return `bun add -g ${pkg}`;
33
+ return `npm install -g ${pkg}`;
34
+ }
35
+
31
36
  async function fetchLatestVersionForCommand(): Promise<string | null> {
32
37
  const controller = new AbortController();
33
38
  const timeout = setTimeout(() => controller.abort(), UPDATE_FETCH_TIMEOUT_MS);
@@ -431,8 +436,9 @@ export async function handleUpdate(ctx: ExtensionCommandContext): Promise<void>
431
436
 
432
437
  ctx.ui.notify(`Updating: v${current} → v${latest}...`, "info");
433
438
 
439
+ const installCmd = resolveInstallCommand(`${NPM_PACKAGE}@latest`);
434
440
  try {
435
- execSync(`npm install -g ${NPM_PACKAGE}@latest`, {
441
+ execSync(installCmd, {
436
442
  stdio: ["ignore", "pipe", "ignore"],
437
443
  });
438
444
  ctx.ui.notify(
@@ -441,7 +447,7 @@ export async function handleUpdate(ctx: ExtensionCommandContext): Promise<void>
441
447
  );
442
448
  } catch {
443
449
  ctx.ui.notify(
444
- `Update failed. Try manually: npm install -g ${NPM_PACKAGE}@latest`,
450
+ `Update failed. Try manually: ${installCmd}`,
445
451
  "error",
446
452
  );
447
453
  }
@@ -15,6 +15,7 @@ import { join } from "node:path";
15
15
  import { gsdRoot } from "./paths.js";
16
16
  import { atomicWriteSync } from "./atomic-write.js";
17
17
  import { effectiveLockFile } from "./session-lock.js";
18
+ import { emitJournalEvent, queryJournal } from "./journal.js";
18
19
 
19
20
  export interface LockData {
20
21
  pid: number;
@@ -118,3 +119,61 @@ export function formatCrashInfo(lock: LockData): string {
118
119
 
119
120
  return lines.join("\n");
120
121
  }
122
+
123
+ /**
124
+ * Emit a synthetic unit-end event for a unit that crashed without emitting its own.
125
+ *
126
+ * Queries the journal to find the most recent unit-start for the crashed unit.
127
+ * If a matching unit-end already exists (e.g. the hard timeout fired), this is a
128
+ * no-op. Called during crash recovery, before clearing the stale lock.
129
+ *
130
+ * Addresses the gap reported in #3348 where `unit-start` was emitted but no
131
+ * `unit-end` followed — side effects landed but the worker died before closeout.
132
+ */
133
+ export function emitCrashRecoveredUnitEnd(basePath: string, lock: LockData): void {
134
+ // Skip bootstrap / starting pseudo-units — they have no meaningful unit-start event.
135
+ if (!lock.unitType || !lock.unitId || lock.unitType === "starting") return;
136
+
137
+ try {
138
+ const all = queryJournal(basePath);
139
+
140
+ // Find the most recent unit-start for this unitId
141
+ const starts = all.filter(
142
+ (e) => e.eventType === "unit-start" && e.data?.unitId === lock.unitId,
143
+ );
144
+ if (starts.length === 0) return;
145
+
146
+ const lastStart = starts[starts.length - 1];
147
+
148
+ // Check if a unit-end was already emitted (e.g. hard timeout fired after the crash)
149
+ const alreadyClosed = all.some(
150
+ (e) =>
151
+ e.eventType === "unit-end" &&
152
+ e.data?.unitId === lock.unitId &&
153
+ e.causedBy?.flowId === lastStart.flowId &&
154
+ e.causedBy?.seq === lastStart.seq,
155
+ );
156
+ if (alreadyClosed) return;
157
+
158
+ // Find the highest seq in this flow for monotonic ordering
159
+ const maxSeq = all
160
+ .filter((e) => e.flowId === lastStart.flowId)
161
+ .reduce((max, e) => Math.max(max, e.seq), lastStart.seq);
162
+
163
+ emitJournalEvent(basePath, {
164
+ ts: new Date().toISOString(),
165
+ flowId: lastStart.flowId,
166
+ seq: maxSeq + 1,
167
+ eventType: "unit-end",
168
+ data: {
169
+ unitType: lock.unitType,
170
+ unitId: lock.unitId,
171
+ status: "crash-recovered",
172
+ artifactVerified: false,
173
+ },
174
+ causedBy: { flowId: lastStart.flowId, seq: lastStart.seq },
175
+ });
176
+ } catch {
177
+ // Never throw from crash recovery path — journal failure must not block recovery
178
+ }
179
+ }
@@ -157,7 +157,7 @@ Setting `prefer_skills: []` does **not** disable skill discovery — it just mea
157
157
 
158
158
  - `phases`: fine-grained control over which phases run. Usually set by `token_profile`, but can be overridden. Keys:
159
159
  - `skip_research`: boolean — skip milestone-level research. Default: `false`.
160
- - `reassess_after_slice`: boolean — run roadmap reassessment after each completed slice. Default: `false`.
160
+ - `reassess_after_slice`: boolean — run roadmap reassessment after each completed slice. Default: `true`.
161
161
  - `skip_reassess`: boolean — force-disable roadmap reassessment even if `reassess_after_slice` is enabled. Default: `false`.
162
162
  - `skip_slice_research`: boolean — skip per-slice research. Default: `false`.
163
163
 
@@ -1564,6 +1564,23 @@ export interface TaskRow {
1564
1564
  sequence: number;
1565
1565
  }
1566
1566
 
1567
+ function parseTaskArrayColumn(raw: unknown): string[] {
1568
+ if (typeof raw !== "string" || raw.trim() === "") return [];
1569
+
1570
+ try {
1571
+ const parsed = JSON.parse(raw);
1572
+ if (Array.isArray(parsed)) return parsed.map((value) => String(value));
1573
+ if (parsed === null || parsed === undefined || parsed === "") return [];
1574
+ return [String(parsed)];
1575
+ } catch {
1576
+ // Older/corrupt rows may contain comma-separated strings instead of JSON.
1577
+ return raw
1578
+ .split(",")
1579
+ .map((value) => value.trim())
1580
+ .filter(Boolean);
1581
+ }
1582
+ }
1583
+
1567
1584
  function rowToTask(row: Record<string, unknown>): TaskRow {
1568
1585
  const parseTaskArray = (value: unknown): string[] => {
1569
1586
  if (Array.isArray(value)) {
@@ -1603,8 +1620,8 @@ function rowToTask(row: Record<string, unknown>): TaskRow {
1603
1620
  blocker_discovered: (row["blocker_discovered"] as number) === 1,
1604
1621
  deviations: row["deviations"] as string,
1605
1622
  known_issues: row["known_issues"] as string,
1606
- key_files: JSON.parse((row["key_files"] as string) || "[]"),
1607
- key_decisions: JSON.parse((row["key_decisions"] as string) || "[]"),
1623
+ key_files: parseTaskArrayColumn(row["key_files"]),
1624
+ key_decisions: parseTaskArrayColumn(row["key_decisions"]),
1608
1625
  full_summary_md: row["full_summary_md"] as string,
1609
1626
  description: (row["description"] as string) ?? "",
1610
1627
  estimate: (row["estimate"] as string) ?? "",
@@ -2200,6 +2217,39 @@ export function deleteSlice(milestoneId: string, sliceId: string): void {
2200
2217
  });
2201
2218
  }
2202
2219
 
2220
+ export function deleteMilestone(milestoneId: string): void {
2221
+ if (!currentDb) throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
2222
+ transaction(() => {
2223
+ currentDb!.prepare(
2224
+ `DELETE FROM verification_evidence WHERE milestone_id = :mid`,
2225
+ ).run({ ":mid": milestoneId });
2226
+ currentDb!.prepare(
2227
+ `DELETE FROM quality_gates WHERE milestone_id = :mid`,
2228
+ ).run({ ":mid": milestoneId });
2229
+ currentDb!.prepare(
2230
+ `DELETE FROM tasks WHERE milestone_id = :mid`,
2231
+ ).run({ ":mid": milestoneId });
2232
+ currentDb!.prepare(
2233
+ `DELETE FROM slice_dependencies WHERE milestone_id = :mid`,
2234
+ ).run({ ":mid": milestoneId });
2235
+ currentDb!.prepare(
2236
+ `DELETE FROM slices WHERE milestone_id = :mid`,
2237
+ ).run({ ":mid": milestoneId });
2238
+ currentDb!.prepare(
2239
+ `DELETE FROM replan_history WHERE milestone_id = :mid`,
2240
+ ).run({ ":mid": milestoneId });
2241
+ currentDb!.prepare(
2242
+ `DELETE FROM assessments WHERE milestone_id = :mid`,
2243
+ ).run({ ":mid": milestoneId });
2244
+ currentDb!.prepare(
2245
+ `DELETE FROM artifacts WHERE milestone_id = :mid`,
2246
+ ).run({ ":mid": milestoneId });
2247
+ currentDb!.prepare(
2248
+ `DELETE FROM milestones WHERE id = :mid`,
2249
+ ).run({ ":mid": milestoneId });
2250
+ });
2251
+ }
2252
+
2203
2253
  export function updateSliceFields(milestoneId: string, sliceId: string, fields: {
2204
2254
  title?: string;
2205
2255
  risk?: string;
@@ -20,7 +20,8 @@ import {
20
20
  } from "./paths.js";
21
21
  import { invalidateAllCaches } from "./cache.js";
22
22
  import { loadQueueOrder, saveQueueOrder } from "./queue-order.js";
23
- import { getMilestone, isDbAvailable, updateMilestoneStatus } from "./gsd-db.js";
23
+ import { deleteMilestone, getMilestone, isDbAvailable, updateMilestoneStatus } from "./gsd-db.js";
24
+ import { removeWorktree } from "./worktree-manager.js";
24
25
  import { logWarning } from "./workflow-logger.js";
25
26
 
26
27
  // ─── Park ──────────────────────────────────────────────────────────────────
@@ -110,6 +111,15 @@ export function discardMilestone(basePath: string, milestoneId: string): boolean
110
111
  const mDir = resolveMilestonePath(basePath, milestoneId);
111
112
  if (!mDir || !existsSync(mDir)) return false;
112
113
 
114
+ try {
115
+ removeWorktree(basePath, milestoneId, {
116
+ branch: `milestone/${milestoneId}`,
117
+ deleteBranch: true,
118
+ });
119
+ } catch (err) {
120
+ logWarning("engine", `discardMilestone worktree cleanup failed for ${milestoneId}: ${(err as Error).message}`);
121
+ }
122
+
113
123
  rmSync(mDir, { recursive: true, force: true });
114
124
 
115
125
  // Prune from queue order if present
@@ -118,6 +128,14 @@ export function discardMilestone(basePath: string, milestoneId: string): boolean
118
128
  saveQueueOrder(basePath, order.filter(id => id !== milestoneId));
119
129
  }
120
130
 
131
+ if (isDbAvailable()) {
132
+ try {
133
+ deleteMilestone(milestoneId);
134
+ } catch (err) {
135
+ logWarning("engine", `discardMilestone DB cleanup failed for ${milestoneId}: ${(err as Error).message}`);
136
+ }
137
+ }
138
+
121
139
  invalidateAllCaches();
122
140
  return true;
123
141
  }
@@ -1,6 +1,6 @@
1
1
  // GSD Extension — Notification Widget
2
2
  // Always-on ambient widget rendered belowEditor showing unread count and
3
- // the most recent notification message. Refreshes every 5 seconds.
3
+ // the most recent notification message. Refreshes every 30 seconds.
4
4
  // Widget key: "gsd-notifications", placement: "belowEditor"
5
5
 
6
6
  import type { ExtensionContext } from "@gsd/pi-coding-agent";
@@ -19,7 +19,7 @@ export function buildNotificationWidgetLines(): string[] {
19
19
 
20
20
  // ─── Widget init ────────────────────────────────────────────────────────
21
21
 
22
- const REFRESH_INTERVAL_MS = 5_000;
22
+ const REFRESH_INTERVAL_MS = 30_000;
23
23
 
24
24
  /**
25
25
  * Initialize the always-on notification widget (belowEditor).