gsd-pi 2.74.0-dev.2b524c3 → 2.74.0-dev.6e23363

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 (275) hide show
  1. package/dist/cli.js +85 -0
  2. package/dist/headless-query.js +4 -1
  3. package/dist/help-text.js +23 -0
  4. package/dist/resources/extensions/gsd/activity-log.js +16 -0
  5. package/dist/resources/extensions/gsd/auto/detect-stuck.js +11 -4
  6. package/dist/resources/extensions/gsd/auto/loop.js +147 -10
  7. package/dist/resources/extensions/gsd/auto/phases.js +158 -4
  8. package/dist/resources/extensions/gsd/auto/session.js +10 -0
  9. package/dist/resources/extensions/gsd/auto-dispatch.js +11 -1
  10. package/dist/resources/extensions/gsd/auto-model-selection.js +51 -5
  11. package/dist/resources/extensions/gsd/auto-post-unit.js +213 -14
  12. package/dist/resources/extensions/gsd/auto-prompts.js +12 -0
  13. package/dist/resources/extensions/gsd/auto-unit-closeout.js +18 -0
  14. package/dist/resources/extensions/gsd/auto-verification.js +100 -2
  15. package/dist/resources/extensions/gsd/auto.js +36 -4
  16. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +21 -8
  17. package/dist/resources/extensions/gsd/commands/catalog.js +26 -1
  18. package/dist/resources/extensions/gsd/commands/handlers/ops.js +20 -0
  19. package/dist/resources/extensions/gsd/commands/handlers/workflow.js +68 -9
  20. package/dist/resources/extensions/gsd/commands-add-tests.js +111 -0
  21. package/dist/resources/extensions/gsd/commands-backlog.js +140 -0
  22. package/dist/resources/extensions/gsd/commands-do.js +79 -0
  23. package/dist/resources/extensions/gsd/commands-maintenance.js +6 -6
  24. package/dist/resources/extensions/gsd/commands-pr-branch.js +180 -0
  25. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +1 -1
  26. package/dist/resources/extensions/gsd/commands-session-report.js +82 -0
  27. package/dist/resources/extensions/gsd/commands-ship.js +187 -0
  28. package/dist/resources/extensions/gsd/db-writer.js +3 -5
  29. package/dist/resources/extensions/gsd/docs/preferences-reference.md +14 -1
  30. package/dist/resources/extensions/gsd/git-service.js +49 -1
  31. package/dist/resources/extensions/gsd/graph-context.js +157 -0
  32. package/dist/resources/extensions/gsd/gsd-db.js +581 -2
  33. package/dist/resources/extensions/gsd/guided-flow.js +23 -0
  34. package/dist/resources/extensions/gsd/index.js +15 -2
  35. package/dist/resources/extensions/gsd/init-wizard.js +1 -0
  36. package/dist/resources/extensions/gsd/journal.js +27 -0
  37. package/dist/resources/extensions/gsd/md-importer.js +3 -4
  38. package/dist/resources/extensions/gsd/memory-store.js +19 -51
  39. package/dist/resources/extensions/gsd/metrics.js +19 -0
  40. package/dist/resources/extensions/gsd/milestone-validation-gates.js +13 -12
  41. package/dist/resources/extensions/gsd/native-git-bridge.js +7 -4
  42. package/dist/resources/extensions/gsd/parallel-orchestrator.js +33 -1
  43. package/dist/resources/extensions/gsd/preferences-models.js +20 -3
  44. package/dist/resources/extensions/gsd/preferences-types.js +1 -0
  45. package/dist/resources/extensions/gsd/preferences-validation.js +108 -2
  46. package/dist/resources/extensions/gsd/preferences.js +26 -0
  47. package/dist/resources/extensions/gsd/prompts/add-tests.md +35 -0
  48. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +12 -2
  49. package/dist/resources/extensions/gsd/state.js +5 -1
  50. package/dist/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
  51. package/dist/resources/extensions/gsd/tools/complete-slice.js +20 -0
  52. package/dist/resources/extensions/gsd/tools/validate-milestone.js +39 -4
  53. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +3 -14
  54. package/dist/resources/extensions/gsd/triage-resolution.js +2 -5
  55. package/dist/resources/extensions/gsd/unit-ownership.js +1 -1
  56. package/dist/resources/extensions/gsd/uok/audit-toggle.js +7 -0
  57. package/dist/resources/extensions/gsd/uok/audit.js +40 -0
  58. package/dist/resources/extensions/gsd/uok/contracts.js +1 -0
  59. package/dist/resources/extensions/gsd/uok/execution-graph.js +179 -0
  60. package/dist/resources/extensions/gsd/uok/flags.js +29 -0
  61. package/dist/resources/extensions/gsd/uok/gate-runner.js +109 -0
  62. package/dist/resources/extensions/gsd/uok/gitops.js +53 -0
  63. package/dist/resources/extensions/gsd/uok/kernel.js +80 -0
  64. package/dist/resources/extensions/gsd/uok/loop-adapter.js +133 -0
  65. package/dist/resources/extensions/gsd/uok/model-policy.js +66 -0
  66. package/dist/resources/extensions/gsd/uok/plan-v2.js +132 -0
  67. package/dist/resources/extensions/gsd/workflow-logger.js +22 -0
  68. package/dist/resources/extensions/gsd/workflow-manifest.js +8 -69
  69. package/dist/resources/extensions/gsd/workflow-migration.js +21 -22
  70. package/dist/resources/extensions/gsd/workflow-projections.js +4 -1
  71. package/dist/resources/extensions/gsd/workflow-reconcile.js +14 -11
  72. package/dist/resources/extensions/ttsr/ttsr-manager.js +3 -1
  73. package/dist/tsconfig.extensions.tsbuildinfo +1 -0
  74. package/dist/web/standalone/.next/BUILD_ID +1 -1
  75. package/dist/web/standalone/.next/app-path-routes-manifest.json +9 -9
  76. package/dist/web/standalone/.next/build-manifest.json +2 -2
  77. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  78. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  79. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  80. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  81. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  82. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  83. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  84. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  85. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  86. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  87. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  88. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  89. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  90. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  91. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  92. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  93. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  94. package/dist/web/standalone/.next/server/app/index.html +1 -1
  95. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  96. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  97. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  98. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  99. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  100. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  101. package/dist/web/standalone/.next/server/app-paths-manifest.json +9 -9
  102. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  103. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  104. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  105. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  106. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  107. package/package.json +3 -2
  108. package/packages/daemon/package.json +2 -2
  109. package/packages/mcp-server/dist/index.d.ts +3 -0
  110. package/packages/mcp-server/dist/index.d.ts.map +1 -1
  111. package/packages/mcp-server/dist/index.js +3 -0
  112. package/packages/mcp-server/dist/index.js.map +1 -1
  113. package/packages/mcp-server/dist/readers/graph.d.ts +87 -0
  114. package/packages/mcp-server/dist/readers/graph.d.ts.map +1 -0
  115. package/packages/mcp-server/dist/readers/graph.js +548 -0
  116. package/packages/mcp-server/dist/readers/graph.js.map +1 -0
  117. package/packages/mcp-server/dist/readers/index.d.ts +2 -0
  118. package/packages/mcp-server/dist/readers/index.d.ts.map +1 -1
  119. package/packages/mcp-server/dist/readers/index.js +1 -0
  120. package/packages/mcp-server/dist/readers/index.js.map +1 -1
  121. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  122. package/packages/mcp-server/dist/server.js +65 -0
  123. package/packages/mcp-server/dist/server.js.map +1 -1
  124. package/packages/mcp-server/package.json +2 -2
  125. package/packages/mcp-server/src/index.ts +15 -0
  126. package/packages/mcp-server/src/readers/graph.test.ts +426 -0
  127. package/packages/mcp-server/src/readers/graph.ts +708 -0
  128. package/packages/mcp-server/src/readers/index.ts +12 -0
  129. package/packages/mcp-server/src/server.ts +83 -0
  130. package/packages/mcp-server/tsconfig.json +1 -0
  131. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -0
  132. package/packages/native/package.json +2 -2
  133. package/packages/native/tsconfig.tsbuildinfo +1 -0
  134. package/packages/pi-agent-core/package.json +1 -1
  135. package/packages/pi-agent-core/tsconfig.json +1 -0
  136. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -0
  137. package/packages/pi-ai/package.json +1 -1
  138. package/packages/pi-ai/tsconfig.json +1 -0
  139. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -0
  140. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +120 -0
  141. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  142. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts +2 -0
  143. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts.map +1 -0
  144. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js +52 -0
  145. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js.map +1 -0
  146. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  147. package/packages/pi-coding-agent/dist/core/model-registry.js +2 -2
  148. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  149. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  150. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +48 -4
  151. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  152. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +166 -0
  153. package/packages/pi-coding-agent/src/core/model-registry-env-fallback.test.ts +59 -0
  154. package/packages/pi-coding-agent/src/core/model-registry.ts +2 -1
  155. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +53 -4
  156. package/packages/pi-coding-agent/src/types/ambient-modules.d.ts +69 -0
  157. package/packages/pi-coding-agent/tsconfig.json +3 -2
  158. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -0
  159. package/packages/pi-tui/package.json +1 -1
  160. package/packages/pi-tui/tsconfig.json +1 -0
  161. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -0
  162. package/packages/rpc-client/package.json +1 -1
  163. package/packages/rpc-client/tsconfig.json +1 -0
  164. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -0
  165. package/src/resources/extensions/gsd/activity-log.ts +21 -0
  166. package/src/resources/extensions/gsd/auto/detect-stuck.ts +12 -4
  167. package/src/resources/extensions/gsd/auto/loop-deps.ts +10 -0
  168. package/src/resources/extensions/gsd/auto/loop.ts +159 -10
  169. package/src/resources/extensions/gsd/auto/phases.ts +191 -4
  170. package/src/resources/extensions/gsd/auto/session.ts +10 -0
  171. package/src/resources/extensions/gsd/auto-dispatch.ts +16 -6
  172. package/src/resources/extensions/gsd/auto-model-selection.ts +66 -5
  173. package/src/resources/extensions/gsd/auto-post-unit.ts +231 -15
  174. package/src/resources/extensions/gsd/auto-prompts.ts +13 -0
  175. package/src/resources/extensions/gsd/auto-unit-closeout.ts +25 -1
  176. package/src/resources/extensions/gsd/auto-verification.ts +129 -2
  177. package/src/resources/extensions/gsd/auto.ts +41 -2
  178. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +24 -8
  179. package/src/resources/extensions/gsd/commands/catalog.ts +26 -1
  180. package/src/resources/extensions/gsd/commands/handlers/ops.ts +20 -0
  181. package/src/resources/extensions/gsd/commands/handlers/workflow.ts +74 -9
  182. package/src/resources/extensions/gsd/commands-add-tests.ts +137 -0
  183. package/src/resources/extensions/gsd/commands-backlog.ts +182 -0
  184. package/src/resources/extensions/gsd/commands-do.ts +109 -0
  185. package/src/resources/extensions/gsd/commands-maintenance.ts +6 -6
  186. package/src/resources/extensions/gsd/commands-pr-branch.ts +234 -0
  187. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +1 -1
  188. package/src/resources/extensions/gsd/commands-session-report.ts +101 -0
  189. package/src/resources/extensions/gsd/commands-ship.ts +219 -0
  190. package/src/resources/extensions/gsd/db-writer.ts +3 -5
  191. package/src/resources/extensions/gsd/docs/preferences-reference.md +14 -1
  192. package/src/resources/extensions/gsd/git-service.ts +68 -0
  193. package/src/resources/extensions/gsd/graph-context.ts +212 -0
  194. package/src/resources/extensions/gsd/gsd-db.ts +788 -3
  195. package/src/resources/extensions/gsd/guided-flow.ts +32 -0
  196. package/src/resources/extensions/gsd/index.ts +18 -2
  197. package/src/resources/extensions/gsd/init-wizard.ts +3 -2
  198. package/src/resources/extensions/gsd/journal.ts +30 -0
  199. package/src/resources/extensions/gsd/md-importer.ts +3 -5
  200. package/src/resources/extensions/gsd/memory-store.ts +31 -62
  201. package/src/resources/extensions/gsd/metrics.ts +26 -0
  202. package/src/resources/extensions/gsd/milestone-validation-gates.ts +13 -14
  203. package/src/resources/extensions/gsd/native-git-bridge.ts +11 -12
  204. package/src/resources/extensions/gsd/parallel-orchestrator.ts +40 -1
  205. package/src/resources/extensions/gsd/preferences-models.ts +20 -3
  206. package/src/resources/extensions/gsd/preferences-types.ts +32 -0
  207. package/src/resources/extensions/gsd/preferences-validation.ts +107 -2
  208. package/src/resources/extensions/gsd/preferences.ts +28 -0
  209. package/src/resources/extensions/gsd/prompts/add-tests.md +35 -0
  210. package/src/resources/extensions/gsd/session-lock.ts +14 -2
  211. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +20 -1
  212. package/src/resources/extensions/gsd/state.ts +9 -2
  213. package/src/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
  214. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +7 -3
  215. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +20 -0
  216. package/src/resources/extensions/gsd/tests/auto-project-root-env.test.ts +7 -3
  217. package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +6 -2
  218. package/src/resources/extensions/gsd/tests/commands-backlog.test.ts +158 -0
  219. package/src/resources/extensions/gsd/tests/commands-do.test.ts +127 -0
  220. package/src/resources/extensions/gsd/tests/commands-pr-branch.test.ts +68 -0
  221. package/src/resources/extensions/gsd/tests/commands-session-report.test.ts +82 -0
  222. package/src/resources/extensions/gsd/tests/commands-ship.test.ts +71 -0
  223. package/src/resources/extensions/gsd/tests/commands-workflow-custom.test.ts +14 -0
  224. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
  225. package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
  226. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +154 -0
  227. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +10 -7
  228. package/src/resources/extensions/gsd/tests/graph-context.test.ts +337 -0
  229. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +1 -1
  230. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +68 -1
  231. package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -2
  232. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -3
  233. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +140 -0
  234. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
  235. package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +2 -1
  236. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +40 -1
  237. package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +180 -0
  238. package/src/resources/extensions/gsd/tests/token-profile.test.ts +8 -5
  239. package/src/resources/extensions/gsd/tests/uok-audit-unified.test.ts +101 -0
  240. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +85 -0
  241. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +69 -0
  242. package/src/resources/extensions/gsd/tests/uok-flags.test.ts +39 -0
  243. package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +70 -0
  244. package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +85 -0
  245. package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +35 -0
  246. package/src/resources/extensions/gsd/tests/uok-model-policy.test.ts +89 -0
  247. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +167 -0
  248. package/src/resources/extensions/gsd/tests/uok-preferences.test.ts +42 -0
  249. package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +39 -0
  250. package/src/resources/extensions/gsd/tests/workflow-logger-wiring.test.ts +223 -0
  251. package/src/resources/extensions/gsd/tools/complete-slice.ts +26 -0
  252. package/src/resources/extensions/gsd/tools/validate-milestone.ts +48 -3
  253. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +3 -11
  254. package/src/resources/extensions/gsd/triage-resolution.ts +2 -7
  255. package/src/resources/extensions/gsd/types.ts +1 -1
  256. package/src/resources/extensions/gsd/unit-ownership.ts +2 -2
  257. package/src/resources/extensions/gsd/uok/audit-toggle.ts +9 -0
  258. package/src/resources/extensions/gsd/uok/audit.ts +51 -0
  259. package/src/resources/extensions/gsd/uok/contracts.ts +135 -0
  260. package/src/resources/extensions/gsd/uok/execution-graph.ts +241 -0
  261. package/src/resources/extensions/gsd/uok/flags.ts +45 -0
  262. package/src/resources/extensions/gsd/uok/gate-runner.ts +146 -0
  263. package/src/resources/extensions/gsd/uok/gitops.ts +75 -0
  264. package/src/resources/extensions/gsd/uok/kernel.ts +105 -0
  265. package/src/resources/extensions/gsd/uok/loop-adapter.ts +162 -0
  266. package/src/resources/extensions/gsd/uok/model-policy.ts +112 -0
  267. package/src/resources/extensions/gsd/uok/plan-v2.ts +156 -0
  268. package/src/resources/extensions/gsd/workflow-logger.ts +25 -0
  269. package/src/resources/extensions/gsd/workflow-manifest.ts +9 -104
  270. package/src/resources/extensions/gsd/workflow-migration.ts +21 -29
  271. package/src/resources/extensions/gsd/workflow-projections.ts +8 -1
  272. package/src/resources/extensions/gsd/workflow-reconcile.ts +15 -15
  273. package/src/resources/extensions/ttsr/ttsr-manager.ts +10 -5
  274. /package/dist/web/standalone/.next/static/{YzIEI9sxJy4t5xgClF08g → bc2gRVFTgD7j--BsJE7vP}/_buildManifest.js +0 -0
  275. /package/dist/web/standalone/.next/static/{YzIEI9sxJy4t5xgClF08g → bc2gRVFTgD7j--BsJE7vP}/_ssgManifest.js +0 -0
@@ -0,0 +1,241 @@
1
+ import type { UokGraphNode } from "./contracts.js";
2
+ import type { DerivedTaskNode } from "../types.js";
3
+ import type { SidecarItem } from "../auto/session.js";
4
+
5
+ export interface ExecutionGraphRunOptions {
6
+ parallel?: boolean;
7
+ maxWorkers?: number;
8
+ }
9
+
10
+ export interface ExecutionGraphResult {
11
+ order: string[];
12
+ conflicts: Array<{ nodeA: string; nodeB: string; file: string }>;
13
+ }
14
+
15
+ export type ExecutionNodeHandler = (node: UokGraphNode) => Promise<void>;
16
+
17
+ export interface ConflictFreeBatchInput {
18
+ orderedIds: string[];
19
+ maxParallel: number;
20
+ hasConflict: (leftId: string, rightId: string) => boolean;
21
+ }
22
+
23
+ export interface ReactiveDispatchSelectionInput {
24
+ graph: Array<Pick<DerivedTaskNode, "id" | "dependsOn" | "outputFiles">>;
25
+ readyIds: string[];
26
+ maxParallel: number;
27
+ inFlightOutputs?: Set<string>;
28
+ }
29
+
30
+ export interface ReactiveDispatchSelectionResult {
31
+ selected: string[];
32
+ conflicts: Array<{ nodeA: string; nodeB: string; file: string }>;
33
+ }
34
+
35
+ export function selectConflictFreeBatch({
36
+ orderedIds,
37
+ maxParallel,
38
+ hasConflict,
39
+ }: ConflictFreeBatchInput): string[] {
40
+ if (maxParallel <= 0 || orderedIds.length === 0) return [];
41
+ const selected: string[] = [];
42
+ for (const candidate of orderedIds) {
43
+ if (selected.length >= maxParallel) break;
44
+ const conflictsExisting = selected.some((existing) => hasConflict(candidate, existing));
45
+ if (conflictsExisting) continue;
46
+ selected.push(candidate);
47
+ }
48
+ return selected;
49
+ }
50
+
51
+ function buildReactiveNodes(
52
+ graph: Array<Pick<DerivedTaskNode, "id" | "dependsOn" | "outputFiles">>,
53
+ ): UokGraphNode[] {
54
+ return graph.map((node) => ({
55
+ id: node.id,
56
+ kind: "unit",
57
+ dependsOn: [...node.dependsOn],
58
+ writes: [...node.outputFiles],
59
+ }));
60
+ }
61
+
62
+ export function selectReactiveDispatchBatch(
63
+ input: ReactiveDispatchSelectionInput,
64
+ ): ReactiveDispatchSelectionResult {
65
+ const nodeMap = new Map(buildReactiveNodes(input.graph).map((n) => [n.id, n]));
66
+ const readyNodes = input.readyIds
67
+ .map((id) => nodeMap.get(id))
68
+ .filter((node): node is UokGraphNode => !!node);
69
+ const conflicts = detectFileConflicts(readyNodes);
70
+ if (readyNodes.length === 0 || input.maxParallel <= 0) {
71
+ return { selected: [], conflicts };
72
+ }
73
+
74
+ const claimed = new Set(input.inFlightOutputs ?? []);
75
+ const selected: string[] = [];
76
+ const selectedSet = new Set<string>();
77
+ const readySet = new Set(input.readyIds);
78
+
79
+ for (const id of input.readyIds) {
80
+ if (selected.length >= input.maxParallel) break;
81
+ const node = nodeMap.get(id);
82
+ if (!node) continue;
83
+
84
+ const hasUnmetReadyDependency = node.dependsOn.some(
85
+ (dep) => readySet.has(dep) && !selectedSet.has(dep),
86
+ );
87
+ if (hasUnmetReadyDependency) continue;
88
+
89
+ const writes = node.writes ?? [];
90
+ const conflictsWithClaimed = writes.some((file) => claimed.has(file));
91
+ if (conflictsWithClaimed) continue;
92
+
93
+ selected.push(node.id);
94
+ selectedSet.add(node.id);
95
+ for (const file of writes) claimed.add(file);
96
+ }
97
+
98
+ return { selected, conflicts };
99
+ }
100
+
101
+ function sidecarToNodeKind(kind: SidecarItem["kind"]): UokGraphNode["kind"] {
102
+ if (kind === "hook") return "hook";
103
+ if (kind === "triage") return "verification";
104
+ return "team-worker";
105
+ }
106
+
107
+ export function buildSidecarQueueNodes(queue: SidecarItem[]): UokGraphNode[] {
108
+ return queue.map((item, index) => ({
109
+ id: `sidecar-${String(index).padStart(4, "0")}:${item.kind}:${item.unitType}:${item.unitId}`,
110
+ kind: sidecarToNodeKind(item.kind),
111
+ dependsOn: index > 0 ? [`sidecar-${String(index - 1).padStart(4, "0")}:${queue[index - 1].kind}:${queue[index - 1].unitType}:${queue[index - 1].unitId}`] : [],
112
+ metadata: { index },
113
+ }));
114
+ }
115
+
116
+ export async function scheduleSidecarQueue(queue: SidecarItem[]): Promise<SidecarItem[]> {
117
+ if (queue.length <= 1) return [...queue];
118
+ const nodes = buildSidecarQueueNodes(queue);
119
+ const scheduler = new ExecutionGraphScheduler();
120
+ const orderedIndexes: number[] = [];
121
+ const seenKinds = new Set<UokGraphNode["kind"]>(nodes.map((n) => n.kind));
122
+
123
+ for (const kind of seenKinds) {
124
+ scheduler.registerHandler(kind, async (node) => {
125
+ const idx = Number(node.metadata?.index);
126
+ if (Number.isInteger(idx) && idx >= 0) orderedIndexes.push(idx);
127
+ });
128
+ }
129
+
130
+ await scheduler.run(nodes, { parallel: false });
131
+ return orderedIndexes.map((idx) => queue[idx]).filter((item): item is SidecarItem => !!item);
132
+ }
133
+
134
+ export class ExecutionGraphScheduler {
135
+ private readonly handlers = new Map<string, ExecutionNodeHandler>();
136
+
137
+ registerHandler(kind: UokGraphNode["kind"], handler: ExecutionNodeHandler): void {
138
+ this.handlers.set(kind, handler);
139
+ }
140
+
141
+ async run(nodes: UokGraphNode[], options?: ExecutionGraphRunOptions): Promise<ExecutionGraphResult> {
142
+ const sorted = topologicalSort(nodes);
143
+ const conflicts = detectFileConflicts(nodes);
144
+
145
+ // Default deterministic serial execution remains the reference path.
146
+ if (!options?.parallel) {
147
+ for (const node of sorted) {
148
+ const handler = this.handlers.get(node.kind);
149
+ if (handler) await handler(node);
150
+ }
151
+ return { order: sorted.map((n) => n.id), conflicts };
152
+ }
153
+
154
+ // Parallel mode only for nodes whose dependencies are already satisfied.
155
+ const maxWorkers = Math.max(1, Math.min(8, options.maxWorkers ?? 2));
156
+ const remaining = new Map(nodes.map((n) => [n.id, n]));
157
+ const done = new Set<string>();
158
+ const order: string[] = [];
159
+
160
+ while (remaining.size > 0) {
161
+ const ready = Array.from(remaining.values()).filter((node) =>
162
+ node.dependsOn.every((dep) => done.has(dep)),
163
+ );
164
+ ready.sort((a, b) => a.id.localeCompare(b.id));
165
+ if (ready.length === 0) {
166
+ throw new Error("Execution graph deadlock detected: no ready nodes and graph not complete");
167
+ }
168
+
169
+ const batch = ready.slice(0, maxWorkers);
170
+ await Promise.all(
171
+ batch.map(async (node) => {
172
+ const handler = this.handlers.get(node.kind);
173
+ if (handler) await handler(node);
174
+ done.add(node.id);
175
+ order.push(node.id);
176
+ remaining.delete(node.id);
177
+ }),
178
+ );
179
+ }
180
+
181
+ return { order, conflicts };
182
+ }
183
+ }
184
+
185
+ function topologicalSort(nodes: UokGraphNode[]): UokGraphNode[] {
186
+ const nodeMap = new Map(nodes.map((n) => [n.id, n]));
187
+ const inDegree = new Map(nodes.map((n) => [n.id, 0]));
188
+
189
+ for (const node of nodes) {
190
+ for (const dep of node.dependsOn) {
191
+ if (nodeMap.has(dep)) {
192
+ inDegree.set(node.id, (inDegree.get(node.id) ?? 0) + 1);
193
+ }
194
+ }
195
+ }
196
+
197
+ const queue = nodes
198
+ .filter((n) => (inDegree.get(n.id) ?? 0) === 0)
199
+ .sort((a, b) => a.id.localeCompare(b.id));
200
+ const ordered: UokGraphNode[] = [];
201
+
202
+ while (queue.length > 0) {
203
+ const current = queue.shift()!;
204
+ ordered.push(current);
205
+
206
+ for (const next of nodes) {
207
+ if (!next.dependsOn.includes(current.id)) continue;
208
+ const deg = (inDegree.get(next.id) ?? 0) - 1;
209
+ inDegree.set(next.id, deg);
210
+ if (deg === 0) {
211
+ queue.push(next);
212
+ queue.sort((a, b) => a.id.localeCompare(b.id));
213
+ }
214
+ }
215
+ }
216
+
217
+ if (ordered.length !== nodes.length) {
218
+ throw new Error("Execution graph has cyclic dependencies");
219
+ }
220
+
221
+ return ordered;
222
+ }
223
+
224
+ function detectFileConflicts(nodes: UokGraphNode[]): Array<{ nodeA: string; nodeB: string; file: string }> {
225
+ const conflicts: Array<{ nodeA: string; nodeB: string; file: string }> = [];
226
+ for (let i = 0; i < nodes.length; i++) {
227
+ const a = nodes[i];
228
+ const writesA = new Set(a.writes ?? []);
229
+ if (writesA.size === 0) continue;
230
+
231
+ for (let j = i + 1; j < nodes.length; j++) {
232
+ const b = nodes[j];
233
+ for (const file of b.writes ?? []) {
234
+ if (writesA.has(file)) {
235
+ conflicts.push({ nodeA: a.id, nodeB: b.id, file });
236
+ }
237
+ }
238
+ }
239
+ }
240
+ return conflicts;
241
+ }
@@ -0,0 +1,45 @@
1
+ import type { GSDPreferences } from "../preferences.js";
2
+ import { loadEffectiveGSDPreferences } from "../preferences.js";
3
+
4
+ export interface UokFlags {
5
+ enabled: boolean;
6
+ legacyFallback: boolean;
7
+ gates: boolean;
8
+ modelPolicy: boolean;
9
+ executionGraph: boolean;
10
+ gitops: boolean;
11
+ gitopsTurnAction: "commit" | "snapshot" | "status-only";
12
+ gitopsTurnPush: boolean;
13
+ auditUnified: boolean;
14
+ planV2: boolean;
15
+ }
16
+
17
+ function envForcesLegacyFallback(): boolean {
18
+ const raw = process.env.GSD_UOK_FORCE_LEGACY ?? process.env.GSD_UOK_LEGACY_FALLBACK;
19
+ if (!raw) return false;
20
+ const normalized = raw.trim().toLowerCase();
21
+ return normalized === "1" || normalized === "true" || normalized === "yes" || normalized === "on";
22
+ }
23
+
24
+ export function resolveUokFlags(prefs: GSDPreferences | undefined): UokFlags {
25
+ const uok = prefs?.uok;
26
+ const legacyFallback = uok?.legacy_fallback?.enabled === true || envForcesLegacyFallback();
27
+ const enabledByPreference = uok?.enabled ?? true;
28
+ return {
29
+ enabled: enabledByPreference && !legacyFallback,
30
+ legacyFallback,
31
+ gates: uok?.gates?.enabled === true,
32
+ modelPolicy: uok?.model_policy?.enabled === true,
33
+ executionGraph: uok?.execution_graph?.enabled === true,
34
+ gitops: uok?.gitops?.enabled === true,
35
+ gitopsTurnAction: uok?.gitops?.turn_action ?? "status-only",
36
+ gitopsTurnPush: uok?.gitops?.turn_push === true,
37
+ auditUnified: uok?.audit_unified?.enabled === true,
38
+ planV2: uok?.plan_v2?.enabled === true,
39
+ };
40
+ }
41
+
42
+ export function loadUokFlags(): UokFlags {
43
+ const prefs = loadEffectiveGSDPreferences()?.preferences;
44
+ return resolveUokFlags(prefs);
45
+ }
@@ -0,0 +1,146 @@
1
+ import type { FailureClass, GateResult } from "./contracts.js";
2
+ import { insertGateRun } from "../gsd-db.js";
3
+ import { buildAuditEnvelope, emitUokAuditEvent } from "./audit.js";
4
+
5
+ export interface GateRunnerContext {
6
+ basePath: string;
7
+ traceId: string;
8
+ turnId: string;
9
+ milestoneId?: string;
10
+ sliceId?: string;
11
+ taskId?: string;
12
+ unitType?: string;
13
+ unitId?: string;
14
+ }
15
+
16
+ export interface GateExecutionInput {
17
+ id: string;
18
+ type: string;
19
+ execute: (ctx: GateRunnerContext, attempt: number) => Promise<{
20
+ outcome: "pass" | "fail" | "retry" | "manual-attention";
21
+ rationale?: string;
22
+ findings?: string;
23
+ failureClass?: FailureClass;
24
+ }>;
25
+ }
26
+
27
+ const RETRY_MATRIX: Record<FailureClass, number> = {
28
+ none: 0,
29
+ policy: 0,
30
+ input: 0,
31
+ execution: 1,
32
+ artifact: 1,
33
+ verification: 1,
34
+ closeout: 1,
35
+ git: 1,
36
+ timeout: 2,
37
+ "manual-attention": 0,
38
+ unknown: 0,
39
+ };
40
+
41
+ export class UokGateRunner {
42
+ private readonly registry = new Map<string, GateExecutionInput>();
43
+
44
+ register(gate: GateExecutionInput): void {
45
+ this.registry.set(gate.id, gate);
46
+ }
47
+
48
+ list(): GateExecutionInput[] {
49
+ return Array.from(this.registry.values());
50
+ }
51
+
52
+ async run(id: string, ctx: GateRunnerContext): Promise<GateResult> {
53
+ const gate = this.registry.get(id);
54
+ if (!gate) {
55
+ return {
56
+ gateId: id,
57
+ gateType: "unknown",
58
+ outcome: "manual-attention",
59
+ failureClass: "unknown",
60
+ rationale: `Gate ${id} not registered`,
61
+ attempt: 1,
62
+ maxAttempts: 1,
63
+ retryable: false,
64
+ evaluatedAt: new Date().toISOString(),
65
+ };
66
+ }
67
+
68
+ let attempt = 0;
69
+ let final: GateResult | null = null;
70
+ const maxAttemptsByFailureClass = RETRY_MATRIX;
71
+
72
+ while (attempt < 3) {
73
+ attempt += 1;
74
+ const now = new Date().toISOString();
75
+ const result = await gate.execute(ctx, attempt);
76
+ const failureClass = result.failureClass ?? (result.outcome === "pass" ? "none" : "unknown");
77
+ const retryBudget = maxAttemptsByFailureClass[failureClass] ?? 0;
78
+ const retryable = result.outcome !== "pass" && attempt <= retryBudget;
79
+
80
+ final = {
81
+ gateId: gate.id,
82
+ gateType: gate.type,
83
+ outcome: retryable ? "retry" : result.outcome,
84
+ failureClass,
85
+ rationale: result.rationale,
86
+ findings: result.findings,
87
+ attempt,
88
+ maxAttempts: Math.max(1, retryBudget),
89
+ retryable,
90
+ evaluatedAt: now,
91
+ };
92
+
93
+ insertGateRun({
94
+ traceId: ctx.traceId,
95
+ turnId: ctx.turnId,
96
+ gateId: final.gateId,
97
+ gateType: final.gateType,
98
+ unitType: ctx.unitType,
99
+ unitId: ctx.unitId,
100
+ milestoneId: ctx.milestoneId,
101
+ sliceId: ctx.sliceId,
102
+ taskId: ctx.taskId,
103
+ outcome: final.outcome,
104
+ failureClass: final.failureClass,
105
+ rationale: final.rationale,
106
+ findings: final.findings,
107
+ attempt: final.attempt,
108
+ maxAttempts: final.maxAttempts,
109
+ retryable: final.retryable,
110
+ evaluatedAt: final.evaluatedAt,
111
+ });
112
+
113
+ emitUokAuditEvent(
114
+ ctx.basePath,
115
+ buildAuditEnvelope({
116
+ traceId: ctx.traceId,
117
+ turnId: ctx.turnId,
118
+ category: "gate",
119
+ type: "gate-run",
120
+ payload: {
121
+ gateId: final.gateId,
122
+ gateType: final.gateType,
123
+ outcome: final.outcome,
124
+ failureClass: final.failureClass,
125
+ attempt: final.attempt,
126
+ maxAttempts: final.maxAttempts,
127
+ retryable: final.retryable,
128
+ },
129
+ }),
130
+ );
131
+
132
+ if (!retryable) break;
133
+ }
134
+
135
+ return final ?? {
136
+ gateId: gate.id,
137
+ gateType: gate.type,
138
+ outcome: "manual-attention",
139
+ failureClass: "unknown",
140
+ attempt: 1,
141
+ maxAttempts: 1,
142
+ retryable: false,
143
+ evaluatedAt: new Date().toISOString(),
144
+ };
145
+ }
146
+ }
@@ -0,0 +1,75 @@
1
+ import { isDbAvailable, upsertTurnGitTransaction } from "../gsd-db.js";
2
+ import type { TurnCloseoutRecord } from "./contracts.js";
3
+ import { buildAuditEnvelope, emitUokAuditEvent } from "./audit.js";
4
+
5
+ export type TurnGitStage = "turn-start" | "stage" | "checkpoint" | "publish" | "record";
6
+
7
+ interface GitTxArgs {
8
+ basePath: string;
9
+ traceId: string;
10
+ turnId: string;
11
+ unitType?: string;
12
+ unitId?: string;
13
+ stage: TurnGitStage;
14
+ action: "commit" | "snapshot" | "status-only";
15
+ push: boolean;
16
+ status: "ok" | "failed";
17
+ error?: string;
18
+ metadata?: Record<string, unknown>;
19
+ }
20
+
21
+ export function writeTurnGitTransaction(args: GitTxArgs): void {
22
+ if (!isDbAvailable()) return;
23
+ upsertTurnGitTransaction({
24
+ traceId: args.traceId,
25
+ turnId: args.turnId,
26
+ unitType: args.unitType,
27
+ unitId: args.unitId,
28
+ stage: args.stage,
29
+ action: args.action,
30
+ push: args.push,
31
+ status: args.status,
32
+ error: args.error,
33
+ metadata: args.metadata,
34
+ updatedAt: new Date().toISOString(),
35
+ });
36
+
37
+ emitUokAuditEvent(
38
+ args.basePath,
39
+ buildAuditEnvelope({
40
+ traceId: args.traceId,
41
+ turnId: args.turnId,
42
+ category: "gitops",
43
+ type: `turn-git-${args.stage}`,
44
+ payload: {
45
+ unitType: args.unitType,
46
+ unitId: args.unitId,
47
+ action: args.action,
48
+ push: args.push,
49
+ status: args.status,
50
+ error: args.error,
51
+ ...(args.metadata ?? {}),
52
+ },
53
+ }),
54
+ );
55
+ }
56
+
57
+ export function writeTurnCloseoutGitRecord(basePath: string, record: TurnCloseoutRecord): void {
58
+ writeTurnGitTransaction({
59
+ basePath,
60
+ traceId: record.traceId,
61
+ turnId: record.turnId,
62
+ unitType: record.unitType,
63
+ unitId: record.unitId,
64
+ stage: "record",
65
+ action: record.gitAction,
66
+ push: record.gitPushed,
67
+ status: record.failureClass === "git" ? "failed" : "ok",
68
+ error: record.failureClass === "git" ? "git closeout failure" : undefined,
69
+ metadata: {
70
+ turnStatus: record.status,
71
+ finishedAt: record.finishedAt,
72
+ activityFile: record.activityFile,
73
+ },
74
+ });
75
+ }
@@ -0,0 +1,105 @@
1
+ import type { ExtensionAPI, ExtensionContext } from "@gsd/pi-coding-agent";
2
+ import { appendFileSync, mkdirSync } from "node:fs";
3
+ import { join } from "node:path";
4
+
5
+ import type { AutoSession } from "../auto/session.js";
6
+ import type { LoopDeps } from "../auto/loop-deps.js";
7
+ import { gsdRoot } from "../paths.js";
8
+ import { buildAuditEnvelope, emitUokAuditEvent } from "./audit.js";
9
+ import { setUnifiedAuditEnabled } from "./audit-toggle.js";
10
+ import { resolveUokFlags } from "./flags.js";
11
+ import { createTurnObserver } from "./loop-adapter.js";
12
+
13
+ interface RunAutoLoopWithUokArgs {
14
+ ctx: ExtensionContext;
15
+ pi: ExtensionAPI;
16
+ s: AutoSession;
17
+ deps: LoopDeps;
18
+ runLegacyLoop: (
19
+ ctx: ExtensionContext,
20
+ pi: ExtensionAPI,
21
+ s: AutoSession,
22
+ deps: LoopDeps,
23
+ ) => Promise<void>;
24
+ }
25
+
26
+ function parityLogPath(basePath: string): string {
27
+ return join(gsdRoot(basePath), "runtime", "uok-parity.jsonl");
28
+ }
29
+
30
+ function writeParityEvent(basePath: string, event: Record<string, unknown>): void {
31
+ try {
32
+ mkdirSync(join(gsdRoot(basePath), "runtime"), { recursive: true });
33
+ appendFileSync(parityLogPath(basePath), `${JSON.stringify(event)}\n`, "utf-8");
34
+ } catch {
35
+ // parity telemetry must never block orchestration
36
+ }
37
+ }
38
+
39
+ function resolveKernelPathLabel(flags: ReturnType<typeof resolveUokFlags>): "uok-wrapper" | "legacy-wrapper" | "legacy-fallback" {
40
+ if (flags.legacyFallback) return "legacy-fallback";
41
+ return flags.enabled ? "uok-wrapper" : "legacy-wrapper";
42
+ }
43
+
44
+ export async function runAutoLoopWithUok(args: RunAutoLoopWithUokArgs): Promise<void> {
45
+ const { ctx, pi, s, deps, runLegacyLoop } = args;
46
+ const prefs = deps.loadEffectiveGSDPreferences()?.preferences;
47
+ const flags = resolveUokFlags(prefs);
48
+ setUnifiedAuditEnabled(flags.auditUnified);
49
+
50
+ writeParityEvent(s.basePath, {
51
+ ts: new Date().toISOString(),
52
+ path: resolveKernelPathLabel(flags),
53
+ flags,
54
+ phase: "enter",
55
+ });
56
+
57
+ if (flags.auditUnified) {
58
+ emitUokAuditEvent(
59
+ s.basePath,
60
+ buildAuditEnvelope({
61
+ traceId: `session:${String(s.autoStartTime || Date.now())}`,
62
+ category: "orchestration",
63
+ type: "uok-kernel-enter",
64
+ payload: {
65
+ flags,
66
+ sessionId: ctx.sessionManager?.getSessionId?.(),
67
+ },
68
+ }),
69
+ );
70
+ }
71
+
72
+ const decoratedDeps: LoopDeps = flags.enabled
73
+ ? {
74
+ ...deps,
75
+ uokObserver: createTurnObserver({
76
+ basePath: s.basePath,
77
+ gitAction: flags.gitopsTurnAction,
78
+ gitPush: flags.gitopsTurnPush,
79
+ enableAudit: flags.auditUnified,
80
+ enableGitops: flags.gitops,
81
+ }),
82
+ }
83
+ : deps;
84
+
85
+ try {
86
+ await runLegacyLoop(ctx, pi, s, decoratedDeps);
87
+ writeParityEvent(s.basePath, {
88
+ ts: new Date().toISOString(),
89
+ path: resolveKernelPathLabel(flags),
90
+ flags,
91
+ phase: "exit",
92
+ status: "ok",
93
+ });
94
+ } catch (err) {
95
+ writeParityEvent(s.basePath, {
96
+ ts: new Date().toISOString(),
97
+ path: resolveKernelPathLabel(flags),
98
+ flags,
99
+ phase: "exit",
100
+ status: "error",
101
+ error: err instanceof Error ? err.message : String(err),
102
+ });
103
+ throw err;
104
+ }
105
+ }