gsd-pi 2.41.0-dev.cac69f9 → 2.42.0-dev.1df898f

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 (263) hide show
  1. package/README.md +23 -0
  2. package/dist/cli.js +18 -3
  3. package/dist/loader.js +3 -1
  4. package/dist/resource-loader.js +39 -6
  5. package/dist/resources/extensions/async-jobs/async-bash-tool.js +52 -4
  6. package/dist/resources/extensions/async-jobs/await-tool.js +5 -0
  7. package/dist/resources/extensions/async-jobs/index.js +2 -0
  8. package/dist/resources/extensions/gsd/auto/loop.js +80 -0
  9. package/dist/resources/extensions/gsd/auto/phases.js +3 -5
  10. package/dist/resources/extensions/gsd/auto/session.js +6 -0
  11. package/dist/resources/extensions/gsd/auto-dashboard.js +2 -0
  12. package/dist/resources/extensions/gsd/auto-prompts.js +3 -16
  13. package/dist/resources/extensions/gsd/auto-start.js +8 -11
  14. package/dist/resources/extensions/gsd/auto.js +28 -1
  15. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +11 -5
  16. package/dist/resources/extensions/gsd/bootstrap/tool-call-loop-guard.js +7 -2
  17. package/dist/resources/extensions/gsd/commands/catalog.js +32 -0
  18. package/dist/resources/extensions/gsd/commands/handlers/workflow.js +146 -0
  19. package/dist/resources/extensions/gsd/context-injector.js +74 -0
  20. package/dist/resources/extensions/gsd/custom-execution-policy.js +47 -0
  21. package/dist/resources/extensions/gsd/custom-verification.js +145 -0
  22. package/dist/resources/extensions/gsd/custom-workflow-engine.js +164 -0
  23. package/dist/resources/extensions/gsd/dashboard-overlay.js +1 -0
  24. package/dist/resources/extensions/gsd/definition-loader.js +352 -0
  25. package/dist/resources/extensions/gsd/detection.js +19 -0
  26. package/dist/resources/extensions/gsd/dev-execution-policy.js +24 -0
  27. package/dist/resources/extensions/gsd/dev-workflow-engine.js +82 -0
  28. package/dist/resources/extensions/gsd/doctor-checks.js +31 -1
  29. package/dist/resources/extensions/gsd/doctor-providers.js +10 -0
  30. package/dist/resources/extensions/gsd/engine-resolver.js +40 -0
  31. package/dist/resources/extensions/gsd/engine-types.js +8 -0
  32. package/dist/resources/extensions/gsd/execution-policy.js +8 -0
  33. package/dist/resources/extensions/gsd/forensics.js +84 -0
  34. package/dist/resources/extensions/gsd/git-constants.js +1 -0
  35. package/dist/resources/extensions/gsd/git-service.js +1 -1
  36. package/dist/resources/extensions/gsd/graph.js +225 -0
  37. package/dist/resources/extensions/gsd/native-git-bridge.js +1 -0
  38. package/dist/resources/extensions/gsd/preferences-types.js +1 -0
  39. package/dist/resources/extensions/gsd/preferences.js +59 -8
  40. package/dist/resources/extensions/gsd/prompts/forensics.md +12 -5
  41. package/dist/resources/extensions/gsd/repo-identity.js +46 -5
  42. package/dist/resources/extensions/gsd/run-manager.js +134 -0
  43. package/dist/resources/extensions/gsd/service-tier.js +13 -4
  44. package/dist/resources/extensions/gsd/session-lock.js +2 -2
  45. package/dist/resources/extensions/gsd/workflow-engine.js +7 -0
  46. package/dist/resources/extensions/gsd/worktree-resolver.js +2 -2
  47. package/dist/resources/extensions/gsd/worktree.js +2 -2
  48. package/dist/resources/extensions/mcp-client/index.js +2 -1
  49. package/dist/resources/extensions/search-the-web/tool-search.js +3 -3
  50. package/dist/resources/skills/create-workflow/SKILL.md +103 -0
  51. package/dist/resources/skills/create-workflow/references/feature-patterns.md +128 -0
  52. package/dist/resources/skills/create-workflow/references/verification-policies.md +76 -0
  53. package/dist/resources/skills/create-workflow/references/yaml-schema-v1.md +46 -0
  54. package/dist/resources/skills/create-workflow/templates/blog-post-pipeline.yaml +60 -0
  55. package/dist/resources/skills/create-workflow/templates/code-audit.yaml +60 -0
  56. package/dist/resources/skills/create-workflow/templates/release-checklist.yaml +66 -0
  57. package/dist/resources/skills/create-workflow/templates/workflow-definition.yaml +32 -0
  58. package/dist/resources/skills/create-workflow/workflows/create-from-scratch.md +104 -0
  59. package/dist/resources/skills/create-workflow/workflows/create-from-template.md +72 -0
  60. package/dist/web/standalone/.next/BUILD_ID +1 -1
  61. package/dist/web/standalone/.next/app-path-routes-manifest.json +15 -15
  62. package/dist/web/standalone/.next/build-manifest.json +2 -2
  63. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  64. package/dist/web/standalone/.next/server/app/_global-error.html +2 -2
  65. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  66. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  67. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  68. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  69. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  70. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  71. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  72. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  73. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  74. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  75. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  76. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  77. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  78. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  79. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  80. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  81. package/dist/web/standalone/.next/server/app/index.html +1 -1
  82. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  83. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  84. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  85. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  86. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  87. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  88. package/dist/web/standalone/.next/server/app-paths-manifest.json +15 -15
  89. package/dist/web/standalone/.next/server/chunks/229.js +2 -2
  90. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  91. package/dist/web/standalone/.next/server/pages/500.html +2 -2
  92. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  93. package/dist/web-mode.d.ts +2 -0
  94. package/dist/web-mode.js +40 -4
  95. package/package.json +1 -1
  96. package/packages/pi-agent-core/dist/agent.d.ts.map +1 -1
  97. package/packages/pi-agent-core/dist/agent.js +2 -0
  98. package/packages/pi-agent-core/dist/agent.js.map +1 -1
  99. package/packages/pi-agent-core/dist/types.d.ts +6 -0
  100. package/packages/pi-agent-core/dist/types.d.ts.map +1 -1
  101. package/packages/pi-agent-core/dist/types.js.map +1 -1
  102. package/packages/pi-agent-core/src/agent.test.ts +53 -0
  103. package/packages/pi-agent-core/src/agent.ts +3 -0
  104. package/packages/pi-agent-core/src/types.ts +6 -0
  105. package/packages/pi-agent-core/tsconfig.json +1 -1
  106. package/packages/pi-ai/dist/models.d.ts +5 -3
  107. package/packages/pi-ai/dist/models.d.ts.map +1 -1
  108. package/packages/pi-ai/dist/models.generated.d.ts +801 -1468
  109. package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
  110. package/packages/pi-ai/dist/models.generated.js +1135 -1588
  111. package/packages/pi-ai/dist/models.generated.js.map +1 -1
  112. package/packages/pi-ai/dist/models.js.map +1 -1
  113. package/packages/pi-ai/dist/utils/oauth/github-copilot.d.ts.map +1 -1
  114. package/packages/pi-ai/dist/utils/oauth/github-copilot.js +60 -2
  115. package/packages/pi-ai/dist/utils/oauth/github-copilot.js.map +1 -1
  116. package/packages/pi-ai/scripts/generate-models.ts +1543 -0
  117. package/packages/pi-ai/src/models.generated.ts +1140 -1593
  118. package/packages/pi-ai/src/models.ts +7 -4
  119. package/packages/pi-ai/src/utils/oauth/github-copilot.ts +74 -2
  120. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  121. package/packages/pi-coding-agent/dist/core/agent-session.js +8 -1
  122. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  123. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +7 -0
  124. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
  125. package/packages/pi-coding-agent/dist/core/auth-storage.js +29 -2
  126. package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
  127. package/packages/pi-coding-agent/dist/core/auth-storage.test.js +60 -0
  128. package/packages/pi-coding-agent/dist/core/auth-storage.test.js.map +1 -1
  129. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  130. package/packages/pi-coding-agent/dist/core/extensions/loader.js +18 -0
  131. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  132. package/packages/pi-coding-agent/dist/core/lsp/client.d.ts.map +1 -1
  133. package/packages/pi-coding-agent/dist/core/lsp/client.js +23 -0
  134. package/packages/pi-coding-agent/dist/core/lsp/client.js.map +1 -1
  135. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  136. package/packages/pi-coding-agent/dist/core/model-registry.js +2 -0
  137. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  138. package/packages/pi-coding-agent/dist/core/package-manager.d.ts +6 -0
  139. package/packages/pi-coding-agent/dist/core/package-manager.d.ts.map +1 -1
  140. package/packages/pi-coding-agent/dist/core/package-manager.js +63 -11
  141. package/packages/pi-coding-agent/dist/core/package-manager.js.map +1 -1
  142. package/packages/pi-coding-agent/dist/core/resource-loader.d.ts +9 -0
  143. package/packages/pi-coding-agent/dist/core/resource-loader.d.ts.map +1 -1
  144. package/packages/pi-coding-agent/dist/core/resource-loader.js +20 -6
  145. package/packages/pi-coding-agent/dist/core/resource-loader.js.map +1 -1
  146. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
  147. package/packages/pi-coding-agent/dist/core/system-prompt.js +6 -5
  148. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  149. package/packages/pi-coding-agent/dist/modes/interactive/components/extension-editor.d.ts.map +1 -1
  150. package/packages/pi-coding-agent/dist/modes/interactive/components/extension-editor.js +3 -0
  151. package/packages/pi-coding-agent/dist/modes/interactive/components/extension-editor.js.map +1 -1
  152. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts.map +1 -1
  153. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js +9 -6
  154. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js.map +1 -1
  155. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  156. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +30 -10
  157. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  158. package/packages/pi-coding-agent/package.json +1 -1
  159. package/packages/pi-coding-agent/src/core/agent-session.ts +7 -1
  160. package/packages/pi-coding-agent/src/core/auth-storage.test.ts +68 -0
  161. package/packages/pi-coding-agent/src/core/auth-storage.ts +30 -2
  162. package/packages/pi-coding-agent/src/core/extensions/loader.ts +18 -0
  163. package/packages/pi-coding-agent/src/core/lsp/client.ts +29 -0
  164. package/packages/pi-coding-agent/src/core/model-registry.ts +3 -0
  165. package/packages/pi-coding-agent/src/core/package-manager.ts +99 -58
  166. package/packages/pi-coding-agent/src/core/resource-loader.ts +24 -6
  167. package/packages/pi-coding-agent/src/core/system-prompt.ts +6 -5
  168. package/packages/pi-coding-agent/src/modes/interactive/components/extension-editor.ts +3 -0
  169. package/packages/pi-coding-agent/src/modes/interactive/components/footer.ts +10 -6
  170. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +31 -11
  171. package/pkg/package.json +1 -1
  172. package/src/resources/extensions/async-jobs/async-bash-timeout.test.ts +122 -0
  173. package/src/resources/extensions/async-jobs/async-bash-tool.ts +40 -4
  174. package/src/resources/extensions/async-jobs/await-tool.test.ts +47 -0
  175. package/src/resources/extensions/async-jobs/await-tool.ts +5 -0
  176. package/src/resources/extensions/async-jobs/index.ts +1 -0
  177. package/src/resources/extensions/async-jobs/job-manager.ts +2 -0
  178. package/src/resources/extensions/gsd/auto/loop-deps.ts +0 -1
  179. package/src/resources/extensions/gsd/auto/loop.ts +91 -0
  180. package/src/resources/extensions/gsd/auto/phases.ts +3 -5
  181. package/src/resources/extensions/gsd/auto/session.ts +6 -0
  182. package/src/resources/extensions/gsd/auto-dashboard.ts +2 -0
  183. package/src/resources/extensions/gsd/auto-prompts.ts +2 -18
  184. package/src/resources/extensions/gsd/auto-start.ts +7 -10
  185. package/src/resources/extensions/gsd/auto.ts +31 -1
  186. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +13 -5
  187. package/src/resources/extensions/gsd/bootstrap/tool-call-loop-guard.ts +9 -2
  188. package/src/resources/extensions/gsd/commands/catalog.ts +32 -0
  189. package/src/resources/extensions/gsd/commands/handlers/workflow.ts +164 -0
  190. package/src/resources/extensions/gsd/context-injector.ts +100 -0
  191. package/src/resources/extensions/gsd/custom-execution-policy.ts +73 -0
  192. package/src/resources/extensions/gsd/custom-verification.ts +180 -0
  193. package/src/resources/extensions/gsd/custom-workflow-engine.ts +216 -0
  194. package/src/resources/extensions/gsd/dashboard-overlay.ts +1 -0
  195. package/src/resources/extensions/gsd/definition-loader.ts +462 -0
  196. package/src/resources/extensions/gsd/detection.ts +19 -0
  197. package/src/resources/extensions/gsd/dev-execution-policy.ts +51 -0
  198. package/src/resources/extensions/gsd/dev-workflow-engine.ts +110 -0
  199. package/src/resources/extensions/gsd/doctor-checks.ts +32 -1
  200. package/src/resources/extensions/gsd/doctor-providers.ts +13 -0
  201. package/src/resources/extensions/gsd/doctor-types.ts +1 -0
  202. package/src/resources/extensions/gsd/engine-resolver.ts +57 -0
  203. package/src/resources/extensions/gsd/engine-types.ts +71 -0
  204. package/src/resources/extensions/gsd/execution-policy.ts +43 -0
  205. package/src/resources/extensions/gsd/forensics.ts +92 -0
  206. package/src/resources/extensions/gsd/git-constants.ts +1 -0
  207. package/src/resources/extensions/gsd/git-service.ts +0 -1
  208. package/src/resources/extensions/gsd/gitignore.ts +1 -1
  209. package/src/resources/extensions/gsd/graph.ts +312 -0
  210. package/src/resources/extensions/gsd/native-git-bridge.ts +1 -0
  211. package/src/resources/extensions/gsd/preferences-types.ts +3 -0
  212. package/src/resources/extensions/gsd/preferences.ts +62 -6
  213. package/src/resources/extensions/gsd/prompts/forensics.md +12 -5
  214. package/src/resources/extensions/gsd/repo-identity.ts +48 -5
  215. package/src/resources/extensions/gsd/run-manager.ts +180 -0
  216. package/src/resources/extensions/gsd/service-tier.ts +17 -4
  217. package/src/resources/extensions/gsd/session-lock.ts +2 -2
  218. package/src/resources/extensions/gsd/tests/activity-log.test.ts +31 -69
  219. package/src/resources/extensions/gsd/tests/bundled-workflow-defs.test.ts +180 -0
  220. package/src/resources/extensions/gsd/tests/commands-workflow-custom.test.ts +283 -0
  221. package/src/resources/extensions/gsd/tests/context-injector.test.ts +313 -0
  222. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +540 -0
  223. package/src/resources/extensions/gsd/tests/custom-verification.test.ts +382 -0
  224. package/src/resources/extensions/gsd/tests/custom-workflow-engine.test.ts +339 -0
  225. package/src/resources/extensions/gsd/tests/dashboard-custom-engine.test.ts +87 -0
  226. package/src/resources/extensions/gsd/tests/definition-loader.test.ts +778 -0
  227. package/src/resources/extensions/gsd/tests/dev-engine-wrapper.test.ts +318 -0
  228. package/src/resources/extensions/gsd/tests/e2e-workflow-pipeline-integration.test.ts +476 -0
  229. package/src/resources/extensions/gsd/tests/engine-interfaces-contract.test.ts +271 -0
  230. package/src/resources/extensions/gsd/tests/forensics-dedup.test.ts +48 -0
  231. package/src/resources/extensions/gsd/tests/forensics-issue-routing.test.ts +43 -0
  232. package/src/resources/extensions/gsd/tests/git-locale.test.ts +133 -0
  233. package/src/resources/extensions/gsd/tests/git-service.test.ts +44 -0
  234. package/src/resources/extensions/gsd/tests/graph-operations.test.ts +599 -0
  235. package/src/resources/extensions/gsd/tests/iterate-engine-integration.test.ts +429 -0
  236. package/src/resources/extensions/gsd/tests/journal.test.ts +82 -127
  237. package/src/resources/extensions/gsd/tests/manifest-status.test.ts +73 -82
  238. package/src/resources/extensions/gsd/tests/run-manager.test.ts +229 -0
  239. package/src/resources/extensions/gsd/tests/service-tier.test.ts +30 -1
  240. package/src/resources/extensions/gsd/tests/skill-activation.test.ts +56 -3
  241. package/src/resources/extensions/gsd/tests/symlink-numbered-variants.test.ts +151 -0
  242. package/src/resources/extensions/gsd/tests/tool-call-loop-guard.test.ts +45 -0
  243. package/src/resources/extensions/gsd/tests/verification-gate.test.ts +156 -263
  244. package/src/resources/extensions/gsd/tests/worktree-health-dispatch.test.ts +35 -78
  245. package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +81 -74
  246. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +1 -2
  247. package/src/resources/extensions/gsd/workflow-engine.ts +38 -0
  248. package/src/resources/extensions/gsd/worktree-resolver.ts +2 -3
  249. package/src/resources/extensions/gsd/worktree.ts +2 -2
  250. package/src/resources/extensions/mcp-client/index.ts +5 -1
  251. package/src/resources/extensions/search-the-web/tool-search.ts +3 -3
  252. package/src/resources/skills/create-workflow/SKILL.md +103 -0
  253. package/src/resources/skills/create-workflow/references/feature-patterns.md +128 -0
  254. package/src/resources/skills/create-workflow/references/verification-policies.md +76 -0
  255. package/src/resources/skills/create-workflow/references/yaml-schema-v1.md +46 -0
  256. package/src/resources/skills/create-workflow/templates/blog-post-pipeline.yaml +60 -0
  257. package/src/resources/skills/create-workflow/templates/code-audit.yaml +60 -0
  258. package/src/resources/skills/create-workflow/templates/release-checklist.yaml +66 -0
  259. package/src/resources/skills/create-workflow/templates/workflow-definition.yaml +32 -0
  260. package/src/resources/skills/create-workflow/workflows/create-from-scratch.md +104 -0
  261. package/src/resources/skills/create-workflow/workflows/create-from-template.md +72 -0
  262. /package/dist/web/standalone/.next/static/{EnGUNqHeGbE0tuuUkTJVA → qw8qDHXOTLUXBq1vEknSz}/_buildManifest.js +0 -0
  263. /package/dist/web/standalone/.next/static/{EnGUNqHeGbE0tuuUkTJVA → qw8qDHXOTLUXBq1vEknSz}/_ssgManifest.js +0 -0
@@ -0,0 +1,382 @@
1
+ /**
2
+ * custom-verification.test.ts — Tests for runCustomVerification().
3
+ *
4
+ * Tests all four verification policies (content-heuristic, shell-command,
5
+ * prompt-verify, human-review) plus edge cases (no policy, missing file).
6
+ * Each test creates a temp run directory with a DEFINITION.yaml and
7
+ * optional test artifacts.
8
+ */
9
+
10
+ import { describe, it } from "node:test";
11
+ import assert from "node:assert/strict";
12
+ import { mkdtempSync, writeFileSync, mkdirSync } from "node:fs";
13
+ import { join } from "node:path";
14
+ import { tmpdir } from "node:os";
15
+ import { stringify } from "yaml";
16
+ import { runCustomVerification } from "../custom-verification.ts";
17
+ import type { WorkflowDefinition } from "../definition-loader.ts";
18
+
19
+ /** Create a temp run directory with the given definition and optional files. */
20
+ function makeTempRun(
21
+ def: WorkflowDefinition,
22
+ files?: Record<string, string>,
23
+ ): string {
24
+ const runDir = mkdtempSync(join(tmpdir(), "cv-test-"));
25
+ writeFileSync(join(runDir, "DEFINITION.yaml"), stringify(def), "utf-8");
26
+
27
+ if (files) {
28
+ for (const [relPath, content] of Object.entries(files)) {
29
+ const absPath = join(runDir, relPath);
30
+ // Ensure parent directories exist
31
+ const parentDir = join(absPath, "..");
32
+ mkdirSync(parentDir, { recursive: true });
33
+ writeFileSync(absPath, content, "utf-8");
34
+ }
35
+ }
36
+
37
+ return runDir;
38
+ }
39
+
40
+ /** Minimal valid workflow definition factory. */
41
+ function makeDef(
42
+ steps: WorkflowDefinition["steps"],
43
+ ): WorkflowDefinition {
44
+ return {
45
+ version: 1,
46
+ name: "test-workflow",
47
+ steps,
48
+ };
49
+ }
50
+
51
+ // ─── content-heuristic tests ────────────────────────────────────────────
52
+
53
+ describe("content-heuristic policy", () => {
54
+ it("returns 'continue' when file exists and meets size/pattern", () => {
55
+ const def = makeDef([
56
+ {
57
+ id: "step-1",
58
+ name: "Generate report",
59
+ prompt: "Generate a report",
60
+ requires: [],
61
+ produces: ["report.md"],
62
+ verify: {
63
+ policy: "content-heuristic",
64
+ minSize: 10,
65
+ pattern: "# Report",
66
+ },
67
+ },
68
+ ]);
69
+
70
+ const runDir = makeTempRun(def, {
71
+ "report.md": "# Report\n\nThis is a valid report with sufficient content.",
72
+ });
73
+
74
+ const result = runCustomVerification(runDir, "step-1");
75
+ assert.equal(result, "continue");
76
+ });
77
+
78
+ it("returns 'pause' when produces file is missing", () => {
79
+ const def = makeDef([
80
+ {
81
+ id: "step-1",
82
+ name: "Generate report",
83
+ prompt: "Generate a report",
84
+ requires: [],
85
+ produces: ["report.md"],
86
+ verify: { policy: "content-heuristic" },
87
+ },
88
+ ]);
89
+
90
+ // No files created — report.md doesn't exist
91
+ const runDir = makeTempRun(def);
92
+
93
+ const result = runCustomVerification(runDir, "step-1");
94
+ assert.equal(result, "pause");
95
+ });
96
+
97
+ it("returns 'pause' when file exists but below minSize", () => {
98
+ const def = makeDef([
99
+ {
100
+ id: "step-1",
101
+ name: "Generate report",
102
+ prompt: "Generate a report",
103
+ requires: [],
104
+ produces: ["report.md"],
105
+ verify: {
106
+ policy: "content-heuristic",
107
+ minSize: 1000,
108
+ },
109
+ },
110
+ ]);
111
+
112
+ const runDir = makeTempRun(def, {
113
+ "report.md": "tiny",
114
+ });
115
+
116
+ const result = runCustomVerification(runDir, "step-1");
117
+ assert.equal(result, "pause");
118
+ });
119
+
120
+ it("returns 'pause' when file exists but pattern does not match", () => {
121
+ const def = makeDef([
122
+ {
123
+ id: "step-1",
124
+ name: "Generate report",
125
+ prompt: "Generate a report",
126
+ requires: [],
127
+ produces: ["report.md"],
128
+ verify: {
129
+ policy: "content-heuristic",
130
+ pattern: "^# Summary",
131
+ },
132
+ },
133
+ ]);
134
+
135
+ const runDir = makeTempRun(def, {
136
+ "report.md": "This has no heading at all.",
137
+ });
138
+
139
+ const result = runCustomVerification(runDir, "step-1");
140
+ assert.equal(result, "pause");
141
+ });
142
+
143
+ it("returns 'continue' when produces is empty", () => {
144
+ const def = makeDef([
145
+ {
146
+ id: "step-1",
147
+ name: "Think step",
148
+ prompt: "Think about the problem",
149
+ requires: [],
150
+ produces: [],
151
+ verify: { policy: "content-heuristic" },
152
+ },
153
+ ]);
154
+
155
+ const runDir = makeTempRun(def);
156
+
157
+ const result = runCustomVerification(runDir, "step-1");
158
+ assert.equal(result, "continue");
159
+ });
160
+
161
+ it("returns 'continue' when file exists with no minSize or pattern checks", () => {
162
+ const def = makeDef([
163
+ {
164
+ id: "step-1",
165
+ name: "Generate output",
166
+ prompt: "Generate output",
167
+ requires: [],
168
+ produces: ["output.txt"],
169
+ verify: { policy: "content-heuristic" },
170
+ },
171
+ ]);
172
+
173
+ const runDir = makeTempRun(def, {
174
+ "output.txt": "",
175
+ });
176
+
177
+ const result = runCustomVerification(runDir, "step-1");
178
+ assert.equal(result, "continue");
179
+ });
180
+ });
181
+
182
+ // ─── shell-command tests ────────────────────────────────────────────────
183
+
184
+ describe("shell-command policy", () => {
185
+ it("returns 'continue' when command exits 0", () => {
186
+ const def = makeDef([
187
+ {
188
+ id: "step-1",
189
+ name: "Build artifact",
190
+ prompt: "Build the artifact",
191
+ requires: [],
192
+ produces: ["artifact.txt"],
193
+ verify: {
194
+ policy: "shell-command",
195
+ command: "test -f artifact.txt",
196
+ },
197
+ },
198
+ ]);
199
+
200
+ const runDir = makeTempRun(def, {
201
+ "artifact.txt": "content",
202
+ });
203
+
204
+ const result = runCustomVerification(runDir, "step-1");
205
+ assert.equal(result, "continue");
206
+ });
207
+
208
+ it("returns 'retry' when command exits non-zero", () => {
209
+ const def = makeDef([
210
+ {
211
+ id: "step-1",
212
+ name: "Build artifact",
213
+ prompt: "Build the artifact",
214
+ requires: [],
215
+ produces: ["artifact.txt"],
216
+ verify: {
217
+ policy: "shell-command",
218
+ command: "test -f nonexistent-file.txt",
219
+ },
220
+ },
221
+ ]);
222
+
223
+ const runDir = makeTempRun(def);
224
+
225
+ const result = runCustomVerification(runDir, "step-1");
226
+ assert.equal(result, "retry");
227
+ });
228
+ });
229
+
230
+ // ─── prompt-verify tests ────────────────────────────────────────────────
231
+
232
+ describe("prompt-verify policy", () => {
233
+ it("returns 'pause'", () => {
234
+ const def = makeDef([
235
+ {
236
+ id: "step-1",
237
+ name: "Creative step",
238
+ prompt: "Write something creative",
239
+ requires: [],
240
+ produces: ["creative.md"],
241
+ verify: {
242
+ policy: "prompt-verify",
243
+ prompt: "Does the creative output meet the brief?",
244
+ },
245
+ },
246
+ ]);
247
+
248
+ const runDir = makeTempRun(def);
249
+
250
+ const result = runCustomVerification(runDir, "step-1");
251
+ assert.equal(result, "pause");
252
+ });
253
+ });
254
+
255
+ // ─── human-review tests ─────────────────────────────────────────────────
256
+
257
+ describe("human-review policy", () => {
258
+ it("returns 'pause'", () => {
259
+ const def = makeDef([
260
+ {
261
+ id: "step-1",
262
+ name: "Review step",
263
+ prompt: "Prepare for review",
264
+ requires: [],
265
+ produces: ["review-doc.md"],
266
+ verify: { policy: "human-review" },
267
+ },
268
+ ]);
269
+
270
+ const runDir = makeTempRun(def);
271
+
272
+ const result = runCustomVerification(runDir, "step-1");
273
+ assert.equal(result, "pause");
274
+ });
275
+ });
276
+
277
+ // ─── no verify policy tests ─────────────────────────────────────────────
278
+
279
+ describe("no verify policy", () => {
280
+ it("returns 'continue' when step has no verify field", () => {
281
+ const def = makeDef([
282
+ {
283
+ id: "step-1",
284
+ name: "Simple step",
285
+ prompt: "Do something simple",
286
+ requires: [],
287
+ produces: [],
288
+ // No verify field
289
+ },
290
+ ]);
291
+
292
+ const runDir = makeTempRun(def);
293
+
294
+ const result = runCustomVerification(runDir, "step-1");
295
+ assert.equal(result, "continue");
296
+ });
297
+
298
+ it("returns 'continue' when step ID is not found in definition", () => {
299
+ const def = makeDef([
300
+ {
301
+ id: "step-1",
302
+ name: "Only step",
303
+ prompt: "Only step",
304
+ requires: [],
305
+ produces: [],
306
+ },
307
+ ]);
308
+
309
+ const runDir = makeTempRun(def);
310
+
311
+ const result = runCustomVerification(runDir, "nonexistent-step");
312
+ assert.equal(result, "continue");
313
+ });
314
+ });
315
+
316
+ // ─── missing DEFINITION.yaml ────────────────────────────────────────────
317
+
318
+ describe("error handling", () => {
319
+ it("throws when DEFINITION.yaml is missing", () => {
320
+ const runDir = mkdtempSync(join(tmpdir(), "cv-test-nodef-"));
321
+ // No DEFINITION.yaml written
322
+
323
+ assert.throws(
324
+ () => runCustomVerification(runDir, "step-1"),
325
+ /ENOENT/,
326
+ );
327
+ });
328
+ });
329
+
330
+ // ─── CustomExecutionPolicy integration ──────────────────────────────────
331
+
332
+ describe("CustomExecutionPolicy.verify() integration", () => {
333
+ it("extracts stepId from unitId and calls runCustomVerification", async () => {
334
+ // Import the policy class
335
+ const { CustomExecutionPolicy } = await import("../custom-execution-policy.ts");
336
+
337
+ const def = makeDef([
338
+ {
339
+ id: "analyze",
340
+ name: "Analyze",
341
+ prompt: "Analyze the data",
342
+ requires: [],
343
+ produces: ["analysis.md"],
344
+ verify: { policy: "content-heuristic" },
345
+ },
346
+ ]);
347
+
348
+ const runDir = makeTempRun(def, {
349
+ "analysis.md": "Analysis complete.",
350
+ });
351
+
352
+ const policy = new CustomExecutionPolicy(runDir);
353
+ const result = await policy.verify("custom-step", "my-workflow/analyze", {
354
+ basePath: "/tmp",
355
+ });
356
+ assert.equal(result, "continue");
357
+ });
358
+
359
+ it("returns 'pause' when content-heuristic fails via policy", async () => {
360
+ const { CustomExecutionPolicy } = await import("../custom-execution-policy.ts");
361
+
362
+ const def = makeDef([
363
+ {
364
+ id: "generate",
365
+ name: "Generate",
366
+ prompt: "Generate output",
367
+ requires: [],
368
+ produces: ["output.md"],
369
+ verify: { policy: "content-heuristic" },
370
+ },
371
+ ]);
372
+
373
+ // No output.md created
374
+ const runDir = makeTempRun(def);
375
+
376
+ const policy = new CustomExecutionPolicy(runDir);
377
+ const result = await policy.verify("custom-step", "my-workflow/generate", {
378
+ basePath: "/tmp",
379
+ });
380
+ assert.equal(result, "pause");
381
+ });
382
+ });