@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.
- package/CHANGELOG.md +758 -3
- package/README.md +78 -15
- package/package.json +17 -10
- package/src/cli/argv.ts +122 -0
- package/src/cli/commands/add-api.ts +134 -0
- package/src/cli/commands/api/annotate/idempotency.ts +59 -0
- package/src/cli/commands/api/annotate/index.ts +525 -0
- package/src/cli/commands/api/annotate/lifecycle.ts +74 -0
- package/src/cli/commands/api/annotate/overlay.ts +206 -0
- package/src/cli/commands/api/annotate/pagination.ts +60 -0
- package/src/cli/commands/api/annotate/prompts.ts +183 -0
- package/src/cli/commands/api/annotate/readback.ts +58 -0
- package/src/cli/commands/api/annotate/resources.ts +91 -0
- package/src/cli/commands/api/annotate/seed-bodies.ts +61 -0
- package/src/cli/commands/audit.ts +480 -0
- package/src/cli/commands/bootstrap.ts +710 -0
- package/src/cli/commands/catalog.ts +35 -0
- package/src/cli/commands/check.ts +348 -0
- package/src/cli/commands/checks.ts +756 -0
- package/src/cli/commands/ci-init.ts +55 -6
- package/src/cli/commands/clean.ts +212 -0
- package/src/cli/commands/cleanup.ts +262 -0
- package/src/cli/commands/completions.ts +192 -0
- package/src/cli/commands/coverage.ts +605 -132
- package/src/cli/commands/db.ts +180 -8
- package/src/cli/commands/describe.ts +37 -2
- package/src/cli/commands/discover.ts +1236 -0
- package/src/cli/commands/doctor.ts +607 -0
- package/src/cli/commands/fixtures.ts +402 -0
- package/src/cli/commands/generate.ts +420 -47
- package/src/cli/commands/init/agents-md.ts +61 -0
- package/src/cli/commands/init/bootstrap.ts +108 -0
- package/src/cli/commands/init/index.ts +244 -0
- package/src/cli/commands/init/skills.ts +98 -0
- package/src/cli/commands/init/templates/agents.md +77 -0
- package/src/cli/commands/init/templates/markdown.d.ts +4 -0
- package/src/cli/commands/init/templates/skills/zond-checks.md +397 -0
- package/src/cli/commands/init/templates/skills/zond-triage.md +210 -0
- package/src/cli/commands/init/templates/skills/zond.md +651 -0
- package/src/cli/commands/init/templates/zond-config.yml +14 -0
- package/src/cli/commands/prepare-fixtures.ts +135 -0
- package/src/cli/commands/probe/mass-assignment.ts +503 -0
- package/src/cli/commands/probe/security.ts +454 -0
- package/src/cli/commands/probe/static.ts +255 -0
- package/src/cli/commands/probe/webhooks.ts +161 -0
- package/src/cli/commands/probe.ts +459 -0
- package/src/cli/commands/reference.ts +87 -0
- package/src/cli/commands/refresh-api.ts +169 -0
- package/src/cli/commands/remove-api.ts +150 -0
- package/src/cli/commands/report-bundle.ts +318 -0
- package/src/cli/commands/report.ts +241 -0
- package/src/cli/commands/request.ts +379 -4
- package/src/cli/commands/run.ts +911 -33
- package/src/cli/commands/session.ts +244 -0
- package/src/cli/commands/use.ts +74 -0
- package/src/cli/index.ts +36 -607
- package/src/cli/json-envelope.ts +112 -3
- package/src/cli/json-schemas.ts +263 -0
- package/src/cli/program.ts +218 -0
- package/src/cli/resolve.ts +105 -0
- package/src/cli/status-filter.ts +124 -0
- package/src/cli/util/api-context.ts +85 -0
- package/src/cli/version.ts +8 -0
- package/src/core/anti-fp/bootstrap.ts +34 -0
- package/src/core/anti-fp/index.ts +33 -0
- package/src/core/anti-fp/registry.ts +44 -0
- package/src/core/anti-fp/rules/baseline-echo.ts +74 -0
- package/src/core/anti-fp/rules/schemathesis/body_negation_becomes_valid.ts +52 -0
- package/src/core/anti-fp/rules/schemathesis/coverage_phase_boundary_positive.ts +38 -0
- package/src/core/anti-fp/rules/schemathesis/has_unverifiable_mutations.ts +35 -0
- package/src/core/anti-fp/rules/schemathesis/index.ts +24 -0
- package/src/core/anti-fp/rules/schemathesis/string_type_mutation_becomes_valid.ts +53 -0
- package/src/core/anti-fp/rules/subscription-gated/index.ts +11 -0
- package/src/core/anti-fp/rules/subscription-gated/paid-plan-403.ts +75 -0
- package/src/core/anti-fp/types.ts +68 -0
- package/src/core/checks/checks/_crud-helpers.ts +133 -0
- package/src/core/checks/checks/_negative_mutator.ts +133 -0
- package/src/core/checks/checks/_readback-helpers.ts +133 -0
- package/src/core/checks/checks/content_type_conformance.ts +39 -0
- package/src/core/checks/checks/cross_call_references.ts +134 -0
- package/src/core/checks/checks/ensure_resource_availability.ts +62 -0
- package/src/core/checks/checks/idempotency_replay.ts +246 -0
- package/src/core/checks/checks/ignored_auth.ts +211 -0
- package/src/core/checks/checks/index.ts +65 -0
- package/src/core/checks/checks/lifecycle_transitions.ts +273 -0
- package/src/core/checks/checks/missing_required_header.ts +40 -0
- package/src/core/checks/checks/negative_data_rejection.ts +45 -0
- package/src/core/checks/checks/not_a_server_error.ts +27 -0
- package/src/core/checks/checks/open_cors_on_sensitive.ts +131 -0
- package/src/core/checks/checks/pagination_invariants.ts +238 -0
- package/src/core/checks/checks/positive_data_acceptance.ts +36 -0
- package/src/core/checks/checks/rate_limit_headers_absent.ts +77 -0
- package/src/core/checks/checks/response_headers_conformance.ts +74 -0
- package/src/core/checks/checks/response_schema_conformance.ts +30 -0
- package/src/core/checks/checks/status_code_conformance.ts +61 -0
- package/src/core/checks/checks/unsupported_method.ts +63 -0
- package/src/core/checks/checks/use_after_free.ts +78 -0
- package/src/core/checks/index.ts +30 -0
- package/src/core/checks/mode.ts +79 -0
- package/src/core/checks/recommended-action.ts +64 -0
- package/src/core/checks/registry.ts +78 -0
- package/src/core/checks/runner.ts +874 -0
- package/src/core/checks/sarif.ts +230 -0
- package/src/core/checks/stateful.ts +121 -0
- package/src/core/checks/types.ts +189 -0
- package/src/core/classifier/recommended-action.ts +222 -0
- package/src/core/context/current.ts +51 -0
- package/src/core/context/session.ts +78 -0
- package/src/core/coverage/loader.ts +185 -0
- package/src/core/coverage/reasons.ts +300 -0
- package/src/core/diagnostics/db-analysis.ts +161 -12
- package/src/core/diagnostics/failure-class.ts +120 -0
- package/src/core/diagnostics/failure-hints.ts +212 -9
- package/src/core/diagnostics/spec-pointer.ts +99 -0
- package/src/core/diagnostics/suggested-fixes.ts +156 -0
- package/src/core/exporter/case-study/index.ts +270 -0
- package/src/core/exporter/curl.ts +40 -0
- package/src/core/exporter/exporter.ts +48 -0
- package/src/core/exporter/html-report/escape.ts +24 -0
- package/src/core/exporter/html-report/index.ts +479 -0
- package/src/core/exporter/html-report/script.ts +100 -0
- package/src/core/exporter/html-report/styles.ts +408 -0
- package/src/core/generator/chunker.ts +53 -15
- package/src/core/generator/coverage-phase.ts +0 -0
- package/src/core/generator/create-body.ts +89 -0
- package/src/core/generator/data-factory.ts +490 -33
- package/src/core/generator/describe.ts +1 -1
- package/src/core/generator/fixtures-builder.ts +325 -0
- package/src/core/generator/index.ts +7 -5
- package/src/core/generator/openapi-reader.ts +55 -3
- package/src/core/generator/path-param-disambig.ts +114 -0
- package/src/core/generator/resources-builder.ts +648 -0
- package/src/core/generator/schema-utils.ts +11 -3
- package/src/core/generator/serializer.ts +114 -15
- package/src/core/generator/suite-generator.ts +484 -77
- package/src/core/generator/types.ts +8 -0
- package/src/core/identity/identity-file.ts +129 -0
- package/src/core/lint/affects.ts +28 -0
- package/src/core/lint/config.ts +96 -0
- package/src/core/lint/format.ts +42 -0
- package/src/core/lint/index.ts +94 -0
- package/src/core/lint/reporter.ts +128 -0
- package/src/core/lint/rules/consistency.ts +158 -0
- package/src/core/lint/rules/heuristics.ts +97 -0
- package/src/core/lint/rules/strictness.ts +109 -0
- package/src/core/lint/types.ts +96 -0
- package/src/core/lint/walker.ts +248 -0
- package/src/core/meta/meta-store.ts +6 -73
- package/src/core/output/README.md +91 -0
- package/src/core/output/index.ts +13 -0
- package/src/core/output/run.ts +126 -0
- package/src/core/output/types.ts +129 -0
- package/src/core/parser/env-interpolation.ts +104 -0
- package/src/core/parser/filter.ts +57 -0
- package/src/core/parser/schema.ts +132 -5
- package/src/core/parser/types.ts +29 -2
- package/src/core/parser/variables.ts +0 -0
- package/src/core/parser/yaml-parser.ts +108 -13
- package/src/core/probe/bootstrap.ts +34 -0
- package/src/core/probe/dry-run-envelope.ts +57 -0
- package/src/core/probe/mass-assignment-probe-class.ts +198 -0
- package/src/core/probe/mass-assignment-probe.ts +1122 -0
- package/src/core/probe/mass-assignment-template.ts +212 -0
- package/src/core/probe/method-probe.ts +164 -0
- package/src/core/probe/method-shared.ts +69 -0
- package/src/core/probe/negative-probe.ts +691 -0
- package/src/core/probe/orphan-tracker.ts +188 -0
- package/src/core/probe/path-discovery.ts +440 -0
- package/src/core/probe/probe-harness.ts +120 -0
- package/src/core/probe/registry.ts +89 -0
- package/src/core/probe/runner.ts +136 -0
- package/src/core/probe/security-probe-class.ts +201 -0
- package/src/core/probe/security-probe.ts +1453 -0
- package/src/core/probe/shared.ts +505 -0
- package/src/core/probe/static-probe-class.ts +125 -0
- package/src/core/probe/types.ts +165 -0
- package/src/core/probe/verdict-aggregator.ts +33 -0
- package/src/core/probe/webhooks-probe.ts +284 -0
- package/src/core/reporter/console.ts +69 -4
- package/src/core/reporter/index.ts +2 -3
- package/src/core/reporter/json.ts +15 -2
- package/src/core/reporter/junit.ts +27 -12
- package/src/core/reporter/ndjson.ts +37 -0
- package/src/core/reporter/types.ts +3 -0
- package/src/core/runner/assertions.ts +62 -2
- package/src/core/runner/async-pool.ts +108 -0
- package/src/core/runner/auth-path.ts +8 -0
- package/src/core/runner/ci-context.ts +72 -0
- package/src/core/runner/executor.ts +391 -52
- package/src/core/runner/form-encode.ts +51 -0
- package/src/core/runner/http-client.ts +115 -7
- package/src/core/runner/learn-drift.ts +293 -0
- package/src/core/runner/preflight-vars.ts +149 -0
- package/src/core/runner/progress-tracker.ts +73 -0
- package/src/core/runner/rate-limiter.ts +203 -0
- package/src/core/runner/run-kind.ts +39 -0
- package/src/core/runner/schema-validator.ts +312 -0
- package/src/core/runner/send-request.ts +153 -20
- package/src/core/runner/types.ts +38 -0
- package/src/core/secrets/registry.ts +164 -0
- package/src/core/secrets/secrets-file.ts +115 -0
- package/src/core/selectors/operation-filter.ts +144 -0
- package/src/core/setup-api.ts +419 -17
- package/src/core/severity/category.ts +94 -0
- package/src/core/severity/index.ts +121 -0
- package/src/core/spec/layers.ts +154 -0
- package/src/core/util/format-eta.ts +21 -0
- package/src/core/utils.ts +5 -1
- package/src/core/workspace/config.ts +129 -0
- package/src/core/workspace/manifest.ts +283 -0
- package/src/core/workspace/output-rotation.ts +62 -0
- package/src/core/workspace/root.ts +94 -0
- package/src/core/workspace/triage-path.ts +87 -0
- package/src/db/lint-runs.ts +47 -0
- package/src/db/migrate.ts +126 -0
- package/src/db/migrations/0001_run_kind.sql +25 -0
- package/src/db/migrations/sql.d.ts +4 -0
- package/src/db/queries/collections.ts +133 -0
- package/src/db/queries/coverage.ts +9 -0
- package/src/db/queries/dashboard.ts +59 -0
- package/src/db/queries/results.ts +128 -0
- package/src/db/queries/runs.ts +235 -0
- package/src/db/queries/sessions.ts +42 -0
- package/src/db/queries/settings.ts +28 -0
- package/src/db/queries/types.ts +172 -0
- package/src/db/queries.ts +72 -802
- package/src/db/schema.ts +179 -48
- package/src/cli/commands/export.ts +0 -144
- package/src/cli/commands/guide.ts +0 -127
- package/src/cli/commands/init.ts +0 -57
- package/src/cli/commands/serve.ts +0 -81
- package/src/cli/commands/sync.ts +0 -269
- package/src/cli/commands/update.ts +0 -189
- package/src/cli/commands/validate.ts +0 -34
- package/src/core/exporter/postman.ts +0 -963
- package/src/core/generator/guide-builder.ts +0 -253
- package/src/core/meta/types.ts +0 -21
- package/src/core/parser/index.ts +0 -21
- package/src/core/runner/execute-run.ts +0 -132
- package/src/core/runner/index.ts +0 -12
- package/src/core/sync/spec-differ.ts +0 -38
- package/src/web/data/collection-state.ts +0 -362
- package/src/web/routes/api.ts +0 -314
- package/src/web/routes/dashboard.ts +0 -350
- package/src/web/routes/runs.ts +0 -64
- package/src/web/schemas.ts +0 -121
- package/src/web/server.ts +0 -134
- package/src/web/static/htmx.min.cjs +0 -1
- package/src/web/static/style.css +0 -1148
- package/src/web/views/endpoints-tab.ts +0 -174
- package/src/web/views/explorer-tab.ts +0 -402
- package/src/web/views/health-strip.ts +0 -92
- package/src/web/views/layout.ts +0 -48
- package/src/web/views/results.ts +0 -210
- package/src/web/views/runs-tab.ts +0 -126
- 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
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
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
|
+
}
|