@kirrosh/zond 0.21.0 → 0.23.0

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 (256) hide show
  1. package/CHANGELOG.md +758 -3
  2. package/README.md +78 -15
  3. package/package.json +17 -10
  4. package/src/cli/argv.ts +122 -0
  5. package/src/cli/commands/add-api.ts +134 -0
  6. package/src/cli/commands/api/annotate/idempotency.ts +59 -0
  7. package/src/cli/commands/api/annotate/index.ts +525 -0
  8. package/src/cli/commands/api/annotate/lifecycle.ts +74 -0
  9. package/src/cli/commands/api/annotate/overlay.ts +206 -0
  10. package/src/cli/commands/api/annotate/pagination.ts +60 -0
  11. package/src/cli/commands/api/annotate/prompts.ts +183 -0
  12. package/src/cli/commands/api/annotate/readback.ts +58 -0
  13. package/src/cli/commands/api/annotate/resources.ts +91 -0
  14. package/src/cli/commands/api/annotate/seed-bodies.ts +61 -0
  15. package/src/cli/commands/audit.ts +480 -0
  16. package/src/cli/commands/bootstrap.ts +710 -0
  17. package/src/cli/commands/catalog.ts +35 -0
  18. package/src/cli/commands/check.ts +348 -0
  19. package/src/cli/commands/checks.ts +756 -0
  20. package/src/cli/commands/ci-init.ts +55 -6
  21. package/src/cli/commands/clean.ts +212 -0
  22. package/src/cli/commands/cleanup.ts +262 -0
  23. package/src/cli/commands/completions.ts +192 -0
  24. package/src/cli/commands/coverage.ts +605 -132
  25. package/src/cli/commands/db.ts +180 -8
  26. package/src/cli/commands/describe.ts +37 -2
  27. package/src/cli/commands/discover.ts +1236 -0
  28. package/src/cli/commands/doctor.ts +607 -0
  29. package/src/cli/commands/fixtures.ts +402 -0
  30. package/src/cli/commands/generate.ts +420 -47
  31. package/src/cli/commands/init/agents-md.ts +61 -0
  32. package/src/cli/commands/init/bootstrap.ts +108 -0
  33. package/src/cli/commands/init/index.ts +244 -0
  34. package/src/cli/commands/init/skills.ts +98 -0
  35. package/src/cli/commands/init/templates/agents.md +77 -0
  36. package/src/cli/commands/init/templates/markdown.d.ts +4 -0
  37. package/src/cli/commands/init/templates/skills/zond-checks.md +397 -0
  38. package/src/cli/commands/init/templates/skills/zond-triage.md +210 -0
  39. package/src/cli/commands/init/templates/skills/zond.md +651 -0
  40. package/src/cli/commands/init/templates/zond-config.yml +14 -0
  41. package/src/cli/commands/prepare-fixtures.ts +135 -0
  42. package/src/cli/commands/probe/mass-assignment.ts +503 -0
  43. package/src/cli/commands/probe/security.ts +454 -0
  44. package/src/cli/commands/probe/static.ts +255 -0
  45. package/src/cli/commands/probe/webhooks.ts +161 -0
  46. package/src/cli/commands/probe.ts +459 -0
  47. package/src/cli/commands/reference.ts +87 -0
  48. package/src/cli/commands/refresh-api.ts +169 -0
  49. package/src/cli/commands/remove-api.ts +150 -0
  50. package/src/cli/commands/report-bundle.ts +318 -0
  51. package/src/cli/commands/report.ts +241 -0
  52. package/src/cli/commands/request.ts +379 -4
  53. package/src/cli/commands/run.ts +911 -33
  54. package/src/cli/commands/session.ts +244 -0
  55. package/src/cli/commands/use.ts +74 -0
  56. package/src/cli/index.ts +36 -607
  57. package/src/cli/json-envelope.ts +112 -3
  58. package/src/cli/json-schemas.ts +263 -0
  59. package/src/cli/program.ts +218 -0
  60. package/src/cli/resolve.ts +105 -0
  61. package/src/cli/status-filter.ts +124 -0
  62. package/src/cli/util/api-context.ts +85 -0
  63. package/src/cli/version.ts +8 -0
  64. package/src/core/anti-fp/bootstrap.ts +34 -0
  65. package/src/core/anti-fp/index.ts +33 -0
  66. package/src/core/anti-fp/registry.ts +44 -0
  67. package/src/core/anti-fp/rules/baseline-echo.ts +74 -0
  68. package/src/core/anti-fp/rules/schemathesis/body_negation_becomes_valid.ts +52 -0
  69. package/src/core/anti-fp/rules/schemathesis/coverage_phase_boundary_positive.ts +38 -0
  70. package/src/core/anti-fp/rules/schemathesis/has_unverifiable_mutations.ts +35 -0
  71. package/src/core/anti-fp/rules/schemathesis/index.ts +24 -0
  72. package/src/core/anti-fp/rules/schemathesis/string_type_mutation_becomes_valid.ts +53 -0
  73. package/src/core/anti-fp/rules/subscription-gated/index.ts +11 -0
  74. package/src/core/anti-fp/rules/subscription-gated/paid-plan-403.ts +75 -0
  75. package/src/core/anti-fp/types.ts +68 -0
  76. package/src/core/checks/checks/_crud-helpers.ts +133 -0
  77. package/src/core/checks/checks/_negative_mutator.ts +133 -0
  78. package/src/core/checks/checks/_readback-helpers.ts +133 -0
  79. package/src/core/checks/checks/content_type_conformance.ts +39 -0
  80. package/src/core/checks/checks/cross_call_references.ts +134 -0
  81. package/src/core/checks/checks/ensure_resource_availability.ts +62 -0
  82. package/src/core/checks/checks/idempotency_replay.ts +246 -0
  83. package/src/core/checks/checks/ignored_auth.ts +211 -0
  84. package/src/core/checks/checks/index.ts +65 -0
  85. package/src/core/checks/checks/lifecycle_transitions.ts +273 -0
  86. package/src/core/checks/checks/missing_required_header.ts +40 -0
  87. package/src/core/checks/checks/negative_data_rejection.ts +45 -0
  88. package/src/core/checks/checks/not_a_server_error.ts +27 -0
  89. package/src/core/checks/checks/open_cors_on_sensitive.ts +131 -0
  90. package/src/core/checks/checks/pagination_invariants.ts +238 -0
  91. package/src/core/checks/checks/positive_data_acceptance.ts +36 -0
  92. package/src/core/checks/checks/rate_limit_headers_absent.ts +77 -0
  93. package/src/core/checks/checks/response_headers_conformance.ts +74 -0
  94. package/src/core/checks/checks/response_schema_conformance.ts +30 -0
  95. package/src/core/checks/checks/status_code_conformance.ts +61 -0
  96. package/src/core/checks/checks/unsupported_method.ts +63 -0
  97. package/src/core/checks/checks/use_after_free.ts +78 -0
  98. package/src/core/checks/index.ts +30 -0
  99. package/src/core/checks/mode.ts +79 -0
  100. package/src/core/checks/recommended-action.ts +64 -0
  101. package/src/core/checks/registry.ts +78 -0
  102. package/src/core/checks/runner.ts +874 -0
  103. package/src/core/checks/sarif.ts +230 -0
  104. package/src/core/checks/stateful.ts +121 -0
  105. package/src/core/checks/types.ts +189 -0
  106. package/src/core/classifier/recommended-action.ts +222 -0
  107. package/src/core/context/current.ts +51 -0
  108. package/src/core/context/session.ts +78 -0
  109. package/src/core/coverage/loader.ts +185 -0
  110. package/src/core/coverage/reasons.ts +300 -0
  111. package/src/core/diagnostics/db-analysis.ts +161 -12
  112. package/src/core/diagnostics/failure-class.ts +120 -0
  113. package/src/core/diagnostics/failure-hints.ts +212 -9
  114. package/src/core/diagnostics/spec-pointer.ts +99 -0
  115. package/src/core/diagnostics/suggested-fixes.ts +156 -0
  116. package/src/core/exporter/case-study/index.ts +270 -0
  117. package/src/core/exporter/curl.ts +40 -0
  118. package/src/core/exporter/exporter.ts +48 -0
  119. package/src/core/exporter/html-report/escape.ts +24 -0
  120. package/src/core/exporter/html-report/index.ts +479 -0
  121. package/src/core/exporter/html-report/script.ts +100 -0
  122. package/src/core/exporter/html-report/styles.ts +408 -0
  123. package/src/core/generator/chunker.ts +53 -15
  124. package/src/core/generator/coverage-phase.ts +0 -0
  125. package/src/core/generator/create-body.ts +89 -0
  126. package/src/core/generator/data-factory.ts +490 -33
  127. package/src/core/generator/describe.ts +1 -1
  128. package/src/core/generator/fixtures-builder.ts +325 -0
  129. package/src/core/generator/index.ts +7 -5
  130. package/src/core/generator/openapi-reader.ts +55 -3
  131. package/src/core/generator/path-param-disambig.ts +114 -0
  132. package/src/core/generator/resources-builder.ts +648 -0
  133. package/src/core/generator/schema-utils.ts +11 -3
  134. package/src/core/generator/serializer.ts +114 -15
  135. package/src/core/generator/suite-generator.ts +484 -77
  136. package/src/core/generator/types.ts +8 -0
  137. package/src/core/identity/identity-file.ts +129 -0
  138. package/src/core/lint/affects.ts +28 -0
  139. package/src/core/lint/config.ts +96 -0
  140. package/src/core/lint/format.ts +42 -0
  141. package/src/core/lint/index.ts +94 -0
  142. package/src/core/lint/reporter.ts +128 -0
  143. package/src/core/lint/rules/consistency.ts +158 -0
  144. package/src/core/lint/rules/heuristics.ts +97 -0
  145. package/src/core/lint/rules/strictness.ts +109 -0
  146. package/src/core/lint/types.ts +96 -0
  147. package/src/core/lint/walker.ts +248 -0
  148. package/src/core/meta/meta-store.ts +6 -73
  149. package/src/core/output/README.md +91 -0
  150. package/src/core/output/index.ts +13 -0
  151. package/src/core/output/run.ts +126 -0
  152. package/src/core/output/types.ts +129 -0
  153. package/src/core/parser/env-interpolation.ts +104 -0
  154. package/src/core/parser/filter.ts +57 -0
  155. package/src/core/parser/schema.ts +132 -5
  156. package/src/core/parser/types.ts +29 -2
  157. package/src/core/parser/variables.ts +0 -0
  158. package/src/core/parser/yaml-parser.ts +108 -13
  159. package/src/core/probe/bootstrap.ts +34 -0
  160. package/src/core/probe/dry-run-envelope.ts +57 -0
  161. package/src/core/probe/mass-assignment-probe-class.ts +198 -0
  162. package/src/core/probe/mass-assignment-probe.ts +1122 -0
  163. package/src/core/probe/mass-assignment-template.ts +212 -0
  164. package/src/core/probe/method-probe.ts +164 -0
  165. package/src/core/probe/method-shared.ts +69 -0
  166. package/src/core/probe/negative-probe.ts +691 -0
  167. package/src/core/probe/orphan-tracker.ts +188 -0
  168. package/src/core/probe/path-discovery.ts +440 -0
  169. package/src/core/probe/probe-harness.ts +120 -0
  170. package/src/core/probe/registry.ts +89 -0
  171. package/src/core/probe/runner.ts +136 -0
  172. package/src/core/probe/security-probe-class.ts +201 -0
  173. package/src/core/probe/security-probe.ts +1453 -0
  174. package/src/core/probe/shared.ts +505 -0
  175. package/src/core/probe/static-probe-class.ts +125 -0
  176. package/src/core/probe/types.ts +165 -0
  177. package/src/core/probe/verdict-aggregator.ts +33 -0
  178. package/src/core/probe/webhooks-probe.ts +284 -0
  179. package/src/core/reporter/console.ts +69 -4
  180. package/src/core/reporter/index.ts +2 -3
  181. package/src/core/reporter/json.ts +15 -2
  182. package/src/core/reporter/junit.ts +27 -12
  183. package/src/core/reporter/ndjson.ts +37 -0
  184. package/src/core/reporter/types.ts +3 -0
  185. package/src/core/runner/assertions.ts +62 -2
  186. package/src/core/runner/async-pool.ts +108 -0
  187. package/src/core/runner/auth-path.ts +8 -0
  188. package/src/core/runner/ci-context.ts +72 -0
  189. package/src/core/runner/executor.ts +391 -52
  190. package/src/core/runner/form-encode.ts +51 -0
  191. package/src/core/runner/http-client.ts +115 -7
  192. package/src/core/runner/learn-drift.ts +293 -0
  193. package/src/core/runner/preflight-vars.ts +149 -0
  194. package/src/core/runner/progress-tracker.ts +73 -0
  195. package/src/core/runner/rate-limiter.ts +203 -0
  196. package/src/core/runner/run-kind.ts +39 -0
  197. package/src/core/runner/schema-validator.ts +312 -0
  198. package/src/core/runner/send-request.ts +153 -20
  199. package/src/core/runner/types.ts +38 -0
  200. package/src/core/secrets/registry.ts +164 -0
  201. package/src/core/secrets/secrets-file.ts +115 -0
  202. package/src/core/selectors/operation-filter.ts +144 -0
  203. package/src/core/setup-api.ts +419 -17
  204. package/src/core/severity/category.ts +94 -0
  205. package/src/core/severity/index.ts +121 -0
  206. package/src/core/spec/layers.ts +154 -0
  207. package/src/core/util/format-eta.ts +21 -0
  208. package/src/core/utils.ts +5 -1
  209. package/src/core/workspace/config.ts +129 -0
  210. package/src/core/workspace/manifest.ts +283 -0
  211. package/src/core/workspace/output-rotation.ts +62 -0
  212. package/src/core/workspace/root.ts +94 -0
  213. package/src/core/workspace/triage-path.ts +87 -0
  214. package/src/db/lint-runs.ts +47 -0
  215. package/src/db/migrate.ts +126 -0
  216. package/src/db/migrations/0001_run_kind.sql +25 -0
  217. package/src/db/migrations/sql.d.ts +4 -0
  218. package/src/db/queries/collections.ts +133 -0
  219. package/src/db/queries/coverage.ts +9 -0
  220. package/src/db/queries/dashboard.ts +59 -0
  221. package/src/db/queries/results.ts +128 -0
  222. package/src/db/queries/runs.ts +235 -0
  223. package/src/db/queries/sessions.ts +42 -0
  224. package/src/db/queries/settings.ts +28 -0
  225. package/src/db/queries/types.ts +172 -0
  226. package/src/db/queries.ts +72 -802
  227. package/src/db/schema.ts +179 -48
  228. package/src/cli/commands/export.ts +0 -144
  229. package/src/cli/commands/guide.ts +0 -127
  230. package/src/cli/commands/init.ts +0 -57
  231. package/src/cli/commands/serve.ts +0 -81
  232. package/src/cli/commands/sync.ts +0 -269
  233. package/src/cli/commands/update.ts +0 -189
  234. package/src/cli/commands/validate.ts +0 -34
  235. package/src/core/exporter/postman.ts +0 -963
  236. package/src/core/generator/guide-builder.ts +0 -253
  237. package/src/core/meta/types.ts +0 -21
  238. package/src/core/parser/index.ts +0 -21
  239. package/src/core/runner/execute-run.ts +0 -132
  240. package/src/core/runner/index.ts +0 -12
  241. package/src/core/sync/spec-differ.ts +0 -38
  242. package/src/web/data/collection-state.ts +0 -362
  243. package/src/web/routes/api.ts +0 -314
  244. package/src/web/routes/dashboard.ts +0 -350
  245. package/src/web/routes/runs.ts +0 -64
  246. package/src/web/schemas.ts +0 -121
  247. package/src/web/server.ts +0 -134
  248. package/src/web/static/htmx.min.cjs +0 -1
  249. package/src/web/static/style.css +0 -1148
  250. package/src/web/views/endpoints-tab.ts +0 -174
  251. package/src/web/views/explorer-tab.ts +0 -402
  252. package/src/web/views/health-strip.ts +0 -92
  253. package/src/web/views/layout.ts +0 -48
  254. package/src/web/views/results.ts +0 -210
  255. package/src/web/views/runs-tab.ts +0 -126
  256. package/src/web/views/suites-tab.ts +0 -181
@@ -3,6 +3,8 @@
3
3
  * Extracted from query-db.ts for reuse in Web UI.
4
4
  */
5
5
 
6
+ import { classify } from "../classifier/recommended-action.ts";
7
+
6
8
  export function statusHint(status: number | null | undefined): string | null {
7
9
  if (!status) return null;
8
10
  if (status >= 500) return "Server-side error — inspect response_body for errorMessage/errorDetail; likely a backend bug";
@@ -41,20 +43,98 @@ export type RecommendedAction =
41
43
  | "report_backend_bug"
42
44
  | "fix_auth_config"
43
45
  | "fix_test_logic"
44
- | "fix_network_config";
46
+ | "fix_network_config"
47
+ | "fix_env"
48
+ /** Fix the OpenAPI spec — emitted by lint-spec.Issue (TASK-294) and
49
+ * by `status_code_conformance` / `*_conformance` checks (ARV-11). */
50
+ | "fix_spec"
51
+ /** Add or correct a fixture in .env.yaml — emitted by discover for
52
+ * miss-* states (TASK-294). */
53
+ | "fix_fixture"
54
+ /** ARV-42 — generator-emitted suite produced a body the API rejected
55
+ * (4xx with validation hint). Editing the YAML is wrong: the next
56
+ * `zond generate` would clobber it. Re-run generate (or refine the
57
+ * spec/.api-resources hints) instead. */
58
+ | "regenerate_suite"
59
+ /** ARV-11 — server accepted an invalid request body. Backend should
60
+ * reject earlier; the test isn't wrong. */
61
+ | "tighten_validation"
62
+ /** ARV-11 — server didn't enforce a header marked `required: true`
63
+ * in the spec. Either enforce it, or drop `required` in the spec. */
64
+ | "add_required_header"
65
+ /** ARV-11 — known limitation that the team has accepted. Agents
66
+ * should not retry, file a bug, or include in dashboards. */
67
+ | "wontfix_known_limitation";
45
68
 
46
69
  export function recommendedAction(
47
70
  failureType: "api_error" | "assertion_failed" | "network_error",
48
71
  responseStatus: number | null,
49
72
  ): RecommendedAction {
50
- if (failureType === "api_error") return "report_backend_bug";
51
- if (failureType === "network_error") {
52
- if (responseStatus === 401 || responseStatus === 403) return "fix_auth_config";
53
- return "fix_network_config";
54
- }
55
- // assertion_failed
56
- if (responseStatus === 401 || responseStatus === 403) return "fix_auth_config";
57
- return "fix_test_logic";
73
+ // ARV-56: delegate to the single classifier.
74
+ const action = classify({
75
+ finding_class: failureType === "api_error" ? "test:api_error" :
76
+ failureType === "network_error" ? "test:network_error" : "test:assertion_failed",
77
+ status: responseStatus,
78
+ });
79
+ // The three failure_type classes are total in the classifier — a missing
80
+ // branch means a future refactor stripped one; surface loudly.
81
+ if (!action) throw new Error(`classifier returned no action for failure_type=${failureType} status=${responseStatus}`);
82
+ return action;
83
+ }
84
+
85
+ /**
86
+ * ARV-42: extended recommender that knows whether the failing test was
87
+ * emitted by `zond generate`. For generated suites, "fix_test_logic" is
88
+ * actively misleading — the generated YAML carries the header
89
+ * "⚠️ Edits will be overwritten on regenerate" and the next `zond audit`
90
+ * really does clobber manual edits. Branch into the actually-actionable
91
+ * remediation instead.
92
+ *
93
+ * - 4xx (400/422) → regenerate_suite: the body the generator emitted
94
+ * didn't pass validation; either re-run generate (so newer heuristics
95
+ * apply, e.g. ARV-38 default-string) or tighten .api-resources hints.
96
+ * - 404 → fix_fixture: a path-param resolved to an empty / stale id
97
+ * in .env.yaml; `prepare-fixtures --seed` is the correct remedy.
98
+ * - everything else → falls back to recommendedAction (auth → 401/403,
99
+ * api_error → 5xx, etc.).
100
+ *
101
+ * `isGenerated` is the heuristic from db-analysis: provenance.type
102
+ * "openapi-generated" OR suite_file under apis/<api>/tests/.
103
+ */
104
+ export function recommendedActionForGenerated(
105
+ failureType: "api_error" | "assertion_failed" | "network_error",
106
+ responseStatus: number | null,
107
+ isGenerated: boolean,
108
+ schemaViolation = false,
109
+ ): RecommendedAction {
110
+ // ARV-56: delegate to classifier. `isGenerated` is encoded via a
111
+ // synthetic suite_path so the same logic flows through classify().
112
+ // ARV-103 (F8): `schemaViolation` propagates the assertion-kind flag —
113
+ // when true, the classifier's "treat schema bugs like 5xx" branch wins
114
+ // over the generator's regenerate_suite default.
115
+ const action = classify({
116
+ finding_class: failureType === "api_error" ? "test:api_error" :
117
+ failureType === "network_error" ? "test:network_error" : "test:assertion_failed",
118
+ status: responseStatus,
119
+ ...(isGenerated ? { suite_path: "apis/_/tests/_.yaml" } : {}),
120
+ ...(schemaViolation ? { schema_violation: true } : {}),
121
+ });
122
+ if (!action) throw new Error(`classifier returned no action for failure_type=${failureType} status=${responseStatus}`);
123
+ return action;
124
+ }
125
+
126
+ /** ARV-42: classify a failing result row as generator-emitted. The two
127
+ * signals are independent — provenance is missing on older runs, while
128
+ * suite_file disambiguates against ad-hoc YAMLs the user dropped into
129
+ * apis/<api>/tests/ themselves (rare but supported). */
130
+ export function isGeneratedTest(
131
+ provenance: { type?: string; generator?: string } | null | undefined,
132
+ suite_file: string | null | undefined,
133
+ ): boolean {
134
+ if (provenance?.type === "openapi-generated") return true;
135
+ if (provenance?.generator && provenance.generator.toLowerCase().includes("zond")) return true;
136
+ if (typeof suite_file === "string" && /(^|\/)apis\/[^/]+\/tests\//.test(suite_file)) return true;
137
+ return false;
58
138
  }
59
139
 
60
140
  export function envCategory(hint: string | undefined): string | null {
@@ -111,3 +191,126 @@ export function computeSharedEnvIssue(
111
191
  // url_malformed
112
192
  return [...failures.map(f => f.hint).filter(Boolean)][0] ?? null;
113
193
  }
194
+
195
+ // ── TASK-98: per-suite env clustering ──────────────────────────────────────
196
+ //
197
+ // Round-3 review showed that the all-or-nothing run-level detector misses
198
+ // real env_issue scenarios: a single suite needs `{{stripe_key}}`, a webhook
199
+ // host is unreachable for one suite only, an auth token expires part-way
200
+ // through. Cluster classification — group by suite, flag a suite when ≥80%
201
+ // of its non-5xx failures share an env-symptom — closes that gap without
202
+ // laundering 5xx (real backend bugs) into env_issue.
203
+ export type EnvSymptom = "missing_var" | "base_url" | "url_malformed" | "auth_expired";
204
+
205
+ function envSymptomOf(failure: {
206
+ hint?: string;
207
+ failure_type: string;
208
+ response_status: number | null;
209
+ }): EnvSymptom | null {
210
+ if (failure.failure_type === "api_error") return null; // 5xx never counted
211
+ const cat = envCategory(failure.hint);
212
+ if (cat === "unresolved_variable") return "missing_var";
213
+ if (cat === "base_url_missing") return "base_url";
214
+ if (cat === "url_malformed") return "url_malformed";
215
+ if (failure.response_status === 401 || failure.response_status === 403) return "auth_expired";
216
+ return null;
217
+ }
218
+
219
+ export interface EnvIssue {
220
+ /** Human-readable summary; used by reporters and shown to the user. */
221
+ message: string;
222
+ /** "run" when the issue spans most/all suites; "suite:<name>" when localized. */
223
+ scope: "run" | `suite:${string}`;
224
+ /** Suites the env_issue covers — one entry for suite scope, ≥2 for run scope. */
225
+ affected_suites: string[];
226
+ /** Histogram of root-cause symptoms across affected failures. */
227
+ symptoms: Partial<Record<EnvSymptom, number>>;
228
+ }
229
+
230
+ /**
231
+ * Cluster non-5xx failures by suite and return per-suite env clusters that
232
+ * meet the env-symptom threshold (default ≥80% AND ≥2 failures). 5xx are
233
+ * excluded so backend bugs cannot be reclassified as env issues.
234
+ */
235
+ export function clusterEnvIssues(
236
+ failures: Array<{
237
+ suite_name: string;
238
+ hint?: string;
239
+ failure_type: string;
240
+ response_status: number | null;
241
+ }>,
242
+ threshold = 0.8,
243
+ ): Array<{ suite: string; symptoms: Partial<Record<EnvSymptom, number>>; total: number }> {
244
+ const bySuite = new Map<string, typeof failures>();
245
+ for (const f of failures) {
246
+ if (f.failure_type === "api_error") continue;
247
+ const list = bySuite.get(f.suite_name) ?? [];
248
+ list.push(f);
249
+ bySuite.set(f.suite_name, list);
250
+ }
251
+ const clusters: Array<{ suite: string; symptoms: Partial<Record<EnvSymptom, number>>; total: number }> = [];
252
+ for (const [suite, items] of bySuite) {
253
+ if (items.length === 0) continue;
254
+ const symptoms: Partial<Record<EnvSymptom, number>> = {};
255
+ let envCount = 0;
256
+ for (const f of items) {
257
+ const s = envSymptomOf(f);
258
+ if (s) {
259
+ symptoms[s] = (symptoms[s] ?? 0) + 1;
260
+ envCount++;
261
+ }
262
+ }
263
+ if (envCount / items.length >= threshold && envCount >= 1) {
264
+ clusters.push({ suite, symptoms, total: items.length });
265
+ }
266
+ }
267
+ return clusters;
268
+ }
269
+
270
+ function formatSymptoms(symptoms: Partial<Record<EnvSymptom, number>>): string {
271
+ const parts: string[] = [];
272
+ for (const k of ["missing_var", "base_url", "url_malformed", "auth_expired"] as EnvSymptom[]) {
273
+ const n = symptoms[k];
274
+ if (n) parts.push(`${k}=${n}`);
275
+ }
276
+ return parts.join(", ");
277
+ }
278
+
279
+ /**
280
+ * Build an EnvIssue envelope from clustered failures. Returns null when no
281
+ * cluster exceeded the threshold. When exactly one suite is affected, scope
282
+ * is `suite:<name>`; ≥2 suites collapse into a `run` scope aggregator.
283
+ */
284
+ export function buildEnvIssue(
285
+ clusters: Array<{ suite: string; symptoms: Partial<Record<EnvSymptom, number>>; total: number }>,
286
+ envFilePath?: string,
287
+ ): EnvIssue | null {
288
+ if (clusters.length === 0) return null;
289
+ const envFile = envFilePath ?? ".env.yaml";
290
+
291
+ const merged: Partial<Record<EnvSymptom, number>> = {};
292
+ for (const c of clusters) {
293
+ for (const [k, v] of Object.entries(c.symptoms)) {
294
+ merged[k as EnvSymptom] = (merged[k as EnvSymptom] ?? 0) + (v ?? 0);
295
+ }
296
+ }
297
+ const affected_suites = clusters.map(c => c.suite).sort();
298
+
299
+ if (clusters.length === 1) {
300
+ const c = clusters[0]!;
301
+ const breakdown = formatSymptoms(c.symptoms);
302
+ return {
303
+ message: `Suite "${c.suite}" looks env-broken (${breakdown}) — check ${envFile}`,
304
+ scope: `suite:${c.suite}`,
305
+ affected_suites,
306
+ symptoms: merged,
307
+ };
308
+ }
309
+ const breakdown = formatSymptoms(merged);
310
+ return {
311
+ message: `${clusters.length} suites look env-broken (${breakdown}) — check ${envFile}`,
312
+ scope: "run",
313
+ affected_suites,
314
+ symptoms: merged,
315
+ };
316
+ }
@@ -0,0 +1,99 @@
1
+ /**
2
+ * TASK-102: build a JSON Pointer (RFC 6901) into the OpenAPI document for the
3
+ * response branch a step exercises, plus a small excerpt of the schema at that
4
+ * pointer. The excerpt is captured at run time and frozen into the DB so that
5
+ * later spec edits can't rewrite history.
6
+ *
7
+ * Inputs come from {@link SourceMetadata} populated by TASK-100:
8
+ * - `endpoint` e.g. "POST /webhooks"
9
+ * - `response_branch` e.g. "422" or "400|422" (first wins for the pointer)
10
+ */
11
+
12
+ import type { SourceMetadata } from "../parser/types.ts";
13
+
14
+ export interface SpecPointer {
15
+ pointer: string;
16
+ excerpt: string;
17
+ }
18
+
19
+ const EXCERPT_MAX_BYTES = 500;
20
+
21
+ function escapeJsonPointerSegment(s: string): string {
22
+ return s.replace(/~/g, "~0").replace(/\//g, "~1");
23
+ }
24
+
25
+ function parseEndpoint(endpoint: string): { method: string; path: string } | null {
26
+ const m = endpoint.match(/^([A-Z]+)\s+(\/.*)$/);
27
+ if (!m) return null;
28
+ return { method: m[1]!.toLowerCase(), path: m[2]! };
29
+ }
30
+
31
+ function pickPrimaryStatus(responseBranch: string | undefined): string | null {
32
+ if (!responseBranch) return null;
33
+ const first = responseBranch.split(/[|,\s]/).find((s) => /^\d{3}$/.test(s));
34
+ return first ?? null;
35
+ }
36
+
37
+ function trimExcerpt(json: string): string {
38
+ if (json.length <= EXCERPT_MAX_BYTES) return json;
39
+ return json.slice(0, EXCERPT_MAX_BYTES) + "\n…[truncated]";
40
+ }
41
+
42
+ /**
43
+ * Resolve the response operation in the OpenAPI document and produce a pointer
44
+ * + frozen excerpt. Returns `null` if any link in the chain is missing — caller
45
+ * persists `null` rather than crashing.
46
+ */
47
+ export function buildSpecPointer(
48
+ source: SourceMetadata | null | undefined,
49
+ openApiDoc: unknown,
50
+ ): SpecPointer | null {
51
+ if (!source || !openApiDoc || typeof openApiDoc !== "object") return null;
52
+ if (typeof source.endpoint !== "string") return null;
53
+
54
+ const parsed = parseEndpoint(source.endpoint);
55
+ if (!parsed) return null;
56
+ const status = pickPrimaryStatus(source.response_branch ?? undefined);
57
+ if (!status) return null;
58
+
59
+ const doc = openApiDoc as Record<string, unknown>;
60
+ const paths = doc.paths as Record<string, unknown> | undefined;
61
+ if (!paths || typeof paths !== "object") return null;
62
+
63
+ const pathItem = paths[parsed.path] as Record<string, unknown> | undefined;
64
+ if (!pathItem || typeof pathItem !== "object") return null;
65
+
66
+ const operation = pathItem[parsed.method] as Record<string, unknown> | undefined;
67
+ if (!operation || typeof operation !== "object") return null;
68
+
69
+ const responses = operation.responses as Record<string, unknown> | undefined;
70
+ if (!responses || typeof responses !== "object") return null;
71
+
72
+ const response = (responses[status] ?? responses.default) as Record<string, unknown> | undefined;
73
+ if (!response || typeof response !== "object") return null;
74
+
75
+ const escapedPath = escapeJsonPointerSegment(parsed.path);
76
+ let pointer = `#/paths/${escapedPath}/${parsed.method}/responses/${status}`;
77
+ let excerptValue: unknown = response;
78
+
79
+ // Drill into application/json schema when available — that's the most useful
80
+ // surface for UI rendering ("backend promised X, returned Y").
81
+ const content = response.content as Record<string, unknown> | undefined;
82
+ if (content && typeof content === "object") {
83
+ const jsonMedia = (content["application/json"] ?? content["application/json; charset=utf-8"]) as
84
+ | Record<string, unknown>
85
+ | undefined;
86
+ if (jsonMedia && typeof jsonMedia === "object" && "schema" in jsonMedia) {
87
+ pointer += "/content/application~1json/schema";
88
+ excerptValue = jsonMedia.schema;
89
+ }
90
+ }
91
+
92
+ let excerpt: string;
93
+ try {
94
+ excerpt = JSON.stringify(excerptValue, null, 2);
95
+ } catch {
96
+ excerpt = String(excerptValue);
97
+ }
98
+ return { pointer, excerpt: trimExcerpt(excerpt) };
99
+ }
@@ -0,0 +1,156 @@
1
+ /**
2
+ * TASK-29: actionable "Suggested fixes" surfaces for `zond db diagnose`.
3
+ *
4
+ * The base diagnose envelope already classifies failures and wires
5
+ * agent_directive / recommended_action / env_issue. This layer adds two
6
+ * concrete, fixable signals that the LLM agent can act on without a second
7
+ * round-trip:
8
+ *
9
+ * 1. Placeholder path-params on 404s — when a 404 hits a URL that still
10
+ * contains literal example/placeholder values (`example`, all-zeros
11
+ * UUID, `your-…-here`, `00000000-…`, …), surface the exact path
12
+ * segment that's broken plus the variable name we'd expect (best-effort
13
+ * from the segment shape).
14
+ *
15
+ * 2. Untilled .env.yaml keys — read the API's .env.yaml and flag values
16
+ * that are empty, `<TODO>`, `example`, `null`, or look like
17
+ * placeholders (`replace-me`, `your-...`). These block CRUD sweeps
18
+ * silently — without them, the agent can't tell apart "value missing"
19
+ * from "value present and wrong".
20
+ */
21
+ import { existsSync, readFileSync } from "node:fs";
22
+ import { parse as parseYaml } from "yaml";
23
+
24
+ export type SuggestedFixKind =
25
+ | "placeholder_path_param"
26
+ | "unfilled_env_key";
27
+
28
+ export interface SuggestedFix {
29
+ kind: SuggestedFixKind;
30
+ /** When `kind=unfilled_env_key`: the key name. When `kind=placeholder_path_param`:
31
+ * the path segment that looks like a placeholder. */
32
+ key: string;
33
+ message: string;
34
+ /** File path the user should edit (usually the API's .env.yaml). */
35
+ source?: string;
36
+ /** Optional one-line example of what to put there. */
37
+ example?: string;
38
+ }
39
+
40
+ const PLACEHOLDER_PATTERNS: Array<{ re: RegExp; reason: string }> = [
41
+ { re: /^example$/i, reason: "literal `example`" },
42
+ { re: /^placeholder$/i, reason: "literal `placeholder`" },
43
+ { re: /^your-[\w-]+-here$/i, reason: "`your-…-here` placeholder" },
44
+ { re: /^00000000-0000-0000-0000-0+(?:beef|dead|cafe|f00d|0+)$/i, reason: "all-zero / sentinel UUID" },
45
+ { re: /^0+$/, reason: "all-zero numeric id" },
46
+ { re: /^replace-me$/i, reason: "`replace-me` placeholder" },
47
+ ];
48
+
49
+ const ENV_PLACEHOLDER_PATTERNS: Array<{ re: RegExp; reason: string }> = [
50
+ { re: /^<.*>$/, reason: "TODO / angle-bracket placeholder" },
51
+ { re: /^example$/i, reason: "literal `example`" },
52
+ { re: /^placeholder$/i, reason: "literal `placeholder`" },
53
+ { re: /^your-[\w-]+-here$/i, reason: "`your-…-here` placeholder" },
54
+ { re: /^replace-?me$/i, reason: "`replace-me` placeholder" },
55
+ { re: /^<TODO>$/i, reason: "explicit TODO" },
56
+ ];
57
+
58
+ /** Identify path segments that look like placeholders. Used on 404 URLs. */
59
+ export function detectPlaceholderSegments(url: string | null): Array<{ segment: string; reason: string }> {
60
+ if (!url) return [];
61
+ let pathname: string;
62
+ try {
63
+ pathname = url.startsWith("http") ? new URL(url).pathname : url.split("?")[0]!;
64
+ } catch {
65
+ pathname = url;
66
+ }
67
+ const out: Array<{ segment: string; reason: string }> = [];
68
+ for (const seg of pathname.split("/").filter(Boolean)) {
69
+ for (const { re, reason } of PLACEHOLDER_PATTERNS) {
70
+ if (re.test(seg)) {
71
+ out.push({ segment: seg, reason });
72
+ break;
73
+ }
74
+ }
75
+ }
76
+ return out;
77
+ }
78
+
79
+ /** Read .env.yaml and return keys whose value is empty/null/placeholder-shaped. */
80
+ export function findUnfilledEnvKeys(envFilePath: string | undefined): SuggestedFix[] {
81
+ if (!envFilePath || !existsSync(envFilePath)) return [];
82
+ let parsed: unknown;
83
+ try {
84
+ parsed = parseYaml(readFileSync(envFilePath, "utf-8"));
85
+ } catch {
86
+ return [];
87
+ }
88
+ if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) return [];
89
+
90
+ const out: SuggestedFix[] = [];
91
+ for (const [key, val] of Object.entries(parsed as Record<string, unknown>)) {
92
+ const reason = classifyEnvValue(val);
93
+ if (reason) {
94
+ out.push({
95
+ kind: "unfilled_env_key",
96
+ key,
97
+ message: `\`${key}\` in ${envFilePath} is unfilled (${reason}). Set it to a real value before re-running.`,
98
+ source: envFilePath,
99
+ });
100
+ }
101
+ }
102
+ return out;
103
+ }
104
+
105
+ function classifyEnvValue(val: unknown): string | null {
106
+ if (val === null || val === undefined) return "null / missing";
107
+ if (typeof val === "string") {
108
+ const trimmed = val.trim();
109
+ if (trimmed === "") return "empty string";
110
+ for (const { re, reason } of ENV_PLACEHOLDER_PATTERNS) {
111
+ if (re.test(trimmed)) return reason;
112
+ }
113
+ }
114
+ return null;
115
+ }
116
+
117
+ export interface BuildSuggestedFixesInput {
118
+ failures: Array<{
119
+ response_status: number | null;
120
+ request_url: string | null;
121
+ suite_name: string;
122
+ test_name: string;
123
+ }>;
124
+ envFilePath?: string;
125
+ }
126
+
127
+ /**
128
+ * Combine placeholder-detection across all 404 failures + env-yaml audit
129
+ * into a single deduplicated list of fixes the agent should apply before
130
+ * re-running.
131
+ */
132
+ export function buildSuggestedFixes(input: BuildSuggestedFixesInput): SuggestedFix[] {
133
+ const fixes: SuggestedFix[] = [];
134
+ const seenSegments = new Set<string>();
135
+
136
+ for (const f of input.failures) {
137
+ if (f.response_status !== 404) continue;
138
+ for (const ph of detectPlaceholderSegments(f.request_url)) {
139
+ const dedupeKey = `seg:${ph.segment}`;
140
+ if (seenSegments.has(dedupeKey)) continue;
141
+ seenSegments.add(dedupeKey);
142
+ fixes.push({
143
+ kind: "placeholder_path_param",
144
+ key: ph.segment,
145
+ message:
146
+ `404 on \`${f.request_url}\` — path segment \`${ph.segment}\` is a ${ph.reason}. ` +
147
+ `Replace with a real id from the live API (e.g. \`zond prepare-fixtures --apply\`) ` +
148
+ `or set the corresponding fixture in ${input.envFilePath ?? ".env.yaml"}.`,
149
+ source: input.envFilePath,
150
+ });
151
+ }
152
+ }
153
+
154
+ fixes.push(...findUnfilledEnvKeys(input.envFilePath));
155
+ return fixes;
156
+ }