@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
@@ -0,0 +1,235 @@
1
+ import { getDb, withDbRetry } from "../schema.ts";
2
+ import type { TestRunResult } from "../../core/runner/types.ts";
3
+ import type { CreateRunOpts, RunRecord, RunSummary, RunFilters } from "./types.ts";
4
+
5
+ function buildRunFilterSQL(filters: RunFilters): { where: string; params: unknown[] } {
6
+ const clauses: string[] = [];
7
+ const params: unknown[] = [];
8
+
9
+ if (filters.status === "has_failures") {
10
+ clauses.push("r.failed > 0");
11
+ } else if (filters.status === "all_passed") {
12
+ clauses.push("r.failed = 0 AND r.total > 0");
13
+ }
14
+
15
+ if (filters.environment) {
16
+ clauses.push("r.environment = ?");
17
+ params.push(filters.environment);
18
+ }
19
+
20
+ if (filters.date_from) {
21
+ clauses.push("r.started_at >= ?");
22
+ params.push(filters.date_from);
23
+ }
24
+
25
+ if (filters.date_to) {
26
+ clauses.push("r.started_at <= ?");
27
+ params.push(filters.date_to + "T23:59:59");
28
+ }
29
+
30
+ if (filters.test_name) {
31
+ clauses.push("r.id IN (SELECT DISTINCT run_id FROM results WHERE test_name LIKE ?)");
32
+ params.push(`%${filters.test_name}%`);
33
+ }
34
+
35
+ if (filters.trigger) {
36
+ clauses.push("r.trigger = ?");
37
+ params.push(filters.trigger);
38
+ }
39
+
40
+ const where = clauses.length > 0 ? "WHERE " + clauses.join(" AND ") : "";
41
+ return { where, params };
42
+ }
43
+
44
+ export function createRun(opts: CreateRunOpts): number {
45
+ const db = getDb();
46
+ const stmt = db.prepare(`
47
+ INSERT INTO runs (started_at, environment, trigger, commit_sha, branch, collection_id, session_id, tags, run_kind)
48
+ VALUES ($started_at, $environment, $trigger, $commit_sha, $branch, $collection_id, $session_id, $tags, $run_kind)
49
+ `);
50
+ const result = withDbRetry("createRun", () => stmt.run({
51
+ $started_at: opts.started_at,
52
+ $environment: opts.environment ?? null,
53
+ $trigger: opts.trigger ?? "manual",
54
+ $commit_sha: opts.commit_sha ?? null,
55
+ $branch: opts.branch ?? null,
56
+ $collection_id: opts.collection_id ?? null,
57
+ $session_id: opts.session_id ?? null,
58
+ $tags: opts.tags && opts.tags.length > 0 ? JSON.stringify(opts.tags) : null,
59
+ // ARV-55: default 'regular' here too — DB default would also catch it,
60
+ // but spelling it out keeps INSERTs idempotent and matches the type.
61
+ $run_kind: opts.run_kind ?? "regular",
62
+ }));
63
+ return Number(result.lastInsertRowid);
64
+ }
65
+
66
+ /** Decode the JSON-encoded `tags` column into a string array. Returns null
67
+ * if the column is null or unparseable (legacy rows / corruption). */
68
+ function decodeTags(raw: unknown): string[] | null {
69
+ if (raw == null) return null;
70
+ if (typeof raw !== "string") return null;
71
+ try {
72
+ const v = JSON.parse(raw);
73
+ if (Array.isArray(v) && v.every((x) => typeof x === "string")) return v;
74
+ return null;
75
+ } catch {
76
+ return null;
77
+ }
78
+ }
79
+
80
+ function decodeRunKind(raw: unknown): "regular" | "probe" | "check" {
81
+ // Migration v10 backfills legacy rows; this is a belt-and-suspenders
82
+ // normaliser for any value SQLite returns from `run_kind`.
83
+ if (raw === "probe" || raw === "check") return raw;
84
+ return "regular";
85
+ }
86
+
87
+ function decodeRunRow(row: unknown): RunRecord | null {
88
+ if (!row || typeof row !== "object") return null;
89
+ const r = row as Record<string, unknown> & { tags?: unknown; run_kind?: unknown };
90
+ return {
91
+ ...(r as unknown as RunRecord),
92
+ tags: decodeTags(r.tags),
93
+ run_kind: decodeRunKind(r.run_kind),
94
+ };
95
+ }
96
+
97
+ export function finalizeRun(runId: number, results: TestRunResult[]): void {
98
+ const db = getDb();
99
+
100
+ const total = results.reduce((s, r) => s + r.total, 0);
101
+ const passed = results.reduce((s, r) => s + r.passed, 0);
102
+ const failed = results.reduce((s, r) => s + r.failed, 0);
103
+ const skipped = results.reduce((s, r) => s + r.skipped, 0);
104
+
105
+ const started = results[0]?.started_at ?? new Date().toISOString();
106
+ const finished = results[results.length - 1]?.finished_at ?? new Date().toISOString();
107
+ const durationMs = new Date(finished).getTime() - new Date(started).getTime();
108
+
109
+ const stmt = db.prepare(`
110
+ UPDATE runs
111
+ SET finished_at = $finished_at,
112
+ total = $total,
113
+ passed = $passed,
114
+ failed = $failed,
115
+ skipped = $skipped,
116
+ duration_ms = $duration_ms
117
+ WHERE id = $id
118
+ `);
119
+ withDbRetry("finalizeRun", () => stmt.run({
120
+ $finished_at: finished,
121
+ $total: total,
122
+ $passed: passed,
123
+ $failed: failed,
124
+ $skipped: skipped,
125
+ $duration_ms: durationMs,
126
+ $id: runId,
127
+ }));
128
+ }
129
+
130
+ export function getRunById(runId: number): RunRecord | null {
131
+ const db = getDb();
132
+ const row = db.query("SELECT * FROM runs WHERE id = ?").get(runId);
133
+ return decodeRunRow(row);
134
+ }
135
+
136
+ /** TASK-274: list runs of a collection with optional time-window or
137
+ * tag-membership filters, ordered by started_at ASC (matches the
138
+ * session-based loader so coverage union order is stable). NULL collection
139
+ * is intentionally excluded — for tag/since selectors the user has
140
+ * pinpointed an API, ad-hoc/probe runs should be tagged or use --union
141
+ * session to be picked up. */
142
+ export function listRunsByCollectionFiltered(
143
+ collectionId: number,
144
+ filters: { since?: string; tag?: string; limit?: number },
145
+ ): RunRecord[] {
146
+ const db = getDb();
147
+ const clauses: string[] = ["collection_id = ?", "finished_at IS NOT NULL"];
148
+ const params: unknown[] = [collectionId];
149
+ if (filters.since) {
150
+ clauses.push("started_at >= ?");
151
+ params.push(filters.since);
152
+ }
153
+ if (filters.tag) {
154
+ // tags is a JSON array of strings — match exact element via LIKE on the
155
+ // serialised form. Cheap and correct for our small N (one row per run);
156
+ // a JSON1-table-function approach would be overkill here.
157
+ clauses.push("tags LIKE ?");
158
+ params.push(`%"${filters.tag.replace(/[\\%_]/g, "\\$&")}"%`);
159
+ }
160
+ const limitClause = filters.limit && filters.limit > 0 ? ` LIMIT ${filters.limit}` : "";
161
+ const rows = db.query(
162
+ `SELECT * FROM runs WHERE ${clauses.join(" AND ")} ORDER BY started_at ASC${limitClause}`,
163
+ ).all(...(params as (string | number)[]));
164
+ const out: RunRecord[] = [];
165
+ for (const r of rows) {
166
+ const decoded = decodeRunRow(r);
167
+ if (decoded) out.push(decoded);
168
+ }
169
+ return out;
170
+ }
171
+
172
+ export function listRuns(limit = 20, offset = 0, filters?: RunFilters): RunSummary[] {
173
+ const db = getDb();
174
+ if (filters && Object.values(filters).some(Boolean)) {
175
+ const { where, params } = buildRunFilterSQL(filters);
176
+ return db.query(`
177
+ SELECT r.id, r.started_at, r.finished_at, r.total, r.passed, r.failed, r.skipped, r.environment, r.duration_ms, r.collection_id, r.session_id
178
+ FROM runs r
179
+ ${where}
180
+ ORDER BY r.started_at DESC
181
+ LIMIT ? OFFSET ?
182
+ `).all(...(params as (string | number)[]), limit, offset) as RunSummary[];
183
+ }
184
+ return db.query(`
185
+ SELECT id, started_at, finished_at, total, passed, failed, skipped, environment, duration_ms, collection_id, session_id
186
+ FROM runs
187
+ ORDER BY started_at DESC
188
+ LIMIT ? OFFSET ?
189
+ `).all(limit, offset) as RunSummary[];
190
+ }
191
+
192
+ /** TASK-266: latest run with at least one failure (for `zond db diagnose`
193
+ * default and `zond-triage` skill). Returns null when no failing run exists. */
194
+ export function getLatestFailingRunId(): number | null {
195
+ const db = getDb();
196
+ const row = db.query(`
197
+ SELECT id FROM runs
198
+ WHERE failed > 0
199
+ ORDER BY started_at DESC
200
+ LIMIT 1
201
+ `).get() as { id: number } | undefined;
202
+ return row?.id ?? null;
203
+ }
204
+
205
+ /** TASK-266: latest run regardless of status (for `--latest`). */
206
+ export function getLatestRunId(): number | null {
207
+ const db = getDb();
208
+ const row = db.query(`
209
+ SELECT id FROM runs
210
+ ORDER BY started_at DESC
211
+ LIMIT 1
212
+ `).get() as { id: number } | undefined;
213
+ return row?.id ?? null;
214
+ }
215
+
216
+ export function deleteRun(runId: number): boolean {
217
+ const db = getDb();
218
+ // results are cascade-deleted via FK; but SQLite FK delete cascade requires explicit config
219
+ return withDbRetry("deleteRun", () => {
220
+ db.prepare("DELETE FROM results WHERE run_id = ?").run(runId);
221
+ const result = db.prepare("DELETE FROM runs WHERE id = ?").run(runId);
222
+ return result.changes > 0;
223
+ });
224
+ }
225
+
226
+ export function countRuns(filters?: RunFilters): number {
227
+ const db = getDb();
228
+ if (filters && Object.values(filters).some(Boolean)) {
229
+ const { where, params } = buildRunFilterSQL(filters);
230
+ const row = db.query(`SELECT COUNT(*) AS cnt FROM runs r ${where}`).get(...(params as (string | number)[])) as { cnt: number };
231
+ return row.cnt;
232
+ }
233
+ const row = db.query("SELECT COUNT(*) AS cnt FROM runs").get() as { cnt: number };
234
+ return row.cnt;
235
+ }
@@ -0,0 +1,42 @@
1
+ import { getDb } from "../schema.ts";
2
+ import type { RunSummary, SessionSummary } from "./types.ts";
3
+
4
+ export function listSessions(limit = 20, offset = 0): SessionSummary[] {
5
+ const db = getDb();
6
+ return db.query(`
7
+ SELECT
8
+ session_id,
9
+ MIN(started_at) AS started_at,
10
+ MAX(finished_at) AS finished_at,
11
+ COUNT(*) AS run_count,
12
+ COALESCE(SUM(total), 0) AS total,
13
+ COALESCE(SUM(passed), 0) AS passed,
14
+ COALESCE(SUM(failed), 0) AS failed,
15
+ COALESCE(SUM(skipped), 0) AS skipped,
16
+ SUM(duration_ms) AS duration_ms,
17
+ MAX(environment) AS environment
18
+ FROM runs
19
+ WHERE session_id IS NOT NULL
20
+ GROUP BY session_id
21
+ ORDER BY started_at DESC
22
+ LIMIT ? OFFSET ?
23
+ `).all(limit, offset) as SessionSummary[];
24
+ }
25
+
26
+ export function countSessions(): number {
27
+ const db = getDb();
28
+ const row = db.query(
29
+ "SELECT COUNT(DISTINCT session_id) AS cnt FROM runs WHERE session_id IS NOT NULL",
30
+ ).get() as { cnt: number };
31
+ return row.cnt;
32
+ }
33
+
34
+ export function listRunsBySession(sessionId: string): RunSummary[] {
35
+ const db = getDb();
36
+ return db.query(`
37
+ SELECT id, started_at, finished_at, total, passed, failed, skipped, environment, duration_ms, collection_id, session_id
38
+ FROM runs
39
+ WHERE session_id = ?
40
+ ORDER BY started_at ASC
41
+ `).all(sessionId) as RunSummary[];
42
+ }
@@ -0,0 +1,28 @@
1
+ /**
2
+ * Generic key-value settings table. The two helpers are currently
3
+ * unused by any caller — kept as the canonical access point so future
4
+ * features (UI prefs, ephemeral run state) can use them without
5
+ * re-rolling SQL. See TASK-179 / TASK-187.
6
+ */
7
+ import { getDb } from "../schema.ts";
8
+
9
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars -- intentional: see file docstring
10
+ function getSetting(key: string): string | null {
11
+ const db = getDb();
12
+ const row = db.query("SELECT value FROM settings WHERE key = ?").get(key) as { value: string } | null;
13
+ return row?.value ?? null;
14
+ }
15
+
16
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars -- intentional: see file docstring
17
+ function setSetting(key: string, value: string): void {
18
+ const db = getDb();
19
+ db.prepare(`
20
+ INSERT INTO settings (key, value) VALUES ($key, $value)
21
+ ON CONFLICT(key) DO UPDATE SET value = excluded.value
22
+ `).run({ $key: key, $value: value });
23
+ }
24
+
25
+ // Reference the helpers so module-private code keeps tsc/knip happy
26
+ // while we leave the slot reserved for future settings consumers.
27
+ void getSetting;
28
+ void setSetting;
@@ -0,0 +1,172 @@
1
+ /**
2
+ * Shared types and helpers for the per-domain query modules
3
+ * (`runs.ts`, `sessions.ts`, `results.ts`, `collections.ts`,
4
+ * `dashboard.ts`, `settings.ts`). Centralised here so domains can
5
+ * import each other's record shapes without circular deps.
6
+ */
7
+ import { resolve } from "path";
8
+
9
+ export function normalizePath(p: string): string {
10
+ return resolve(p).replace(/\\/g, "/");
11
+ }
12
+
13
+ export interface CreateRunOpts {
14
+ started_at: string;
15
+ environment?: string;
16
+ trigger?: string;
17
+ commit_sha?: string;
18
+ branch?: string;
19
+ collection_id?: number;
20
+ session_id?: string;
21
+ /** TASK-274: union of suite-level tags actually executed in the run, plus
22
+ * any explicit `--tag <x>` filters from the CLI. Persisted as a JSON
23
+ * array string so `coverage --union tag:<name>` can filter run-rows. */
24
+ tags?: string[];
25
+ /** ARV-55: 'regular' (default) | 'probe' | 'check'. Determines whether
26
+ * coverage's default query picks the run up. Callers compute the kind
27
+ * via `detectRunKind(suite_files)` before insert; INSERT defaults to
28
+ * 'regular' for legacy code paths that don't know about it yet. */
29
+ run_kind?: import("../../core/runner/run-kind.ts").RunKind;
30
+ }
31
+
32
+ export interface RunRecord {
33
+ id: number;
34
+ started_at: string;
35
+ finished_at: string | null;
36
+ total: number;
37
+ passed: number;
38
+ failed: number;
39
+ skipped: number;
40
+ trigger: string;
41
+ commit_sha: string | null;
42
+ branch: string | null;
43
+ environment: string | null;
44
+ duration_ms: number | null;
45
+ collection_id: number | null;
46
+ session_id: string | null;
47
+ /** TASK-274: tag list captured at run time (JSON-encoded in DB). null
48
+ * on legacy rows persisted before migration v9. */
49
+ tags: string[] | null;
50
+ /** ARV-55: 'regular' | 'probe' | 'check'. Persisted as TEXT NOT NULL
51
+ * (default 'regular'); old rows are backfilled by migration v10. */
52
+ run_kind: import("../../core/runner/run-kind.ts").RunKind;
53
+ }
54
+
55
+ export interface RunSummary {
56
+ id: number;
57
+ started_at: string;
58
+ finished_at: string | null;
59
+ total: number;
60
+ passed: number;
61
+ failed: number;
62
+ skipped: number;
63
+ environment: string | null;
64
+ duration_ms: number | null;
65
+ collection_id: number | null;
66
+ session_id: string | null;
67
+ }
68
+
69
+ export interface SessionSummary {
70
+ session_id: string;
71
+ started_at: string;
72
+ finished_at: string | null;
73
+ run_count: number;
74
+ total: number;
75
+ passed: number;
76
+ failed: number;
77
+ skipped: number;
78
+ duration_ms: number | null;
79
+ environment: string | null;
80
+ }
81
+
82
+ export interface CollectionRecord {
83
+ id: number;
84
+ name: string;
85
+ base_dir: string | null;
86
+ test_path: string;
87
+ openapi_spec: string | null;
88
+ created_at: string;
89
+ }
90
+
91
+ export interface CollectionSummary {
92
+ id: number;
93
+ name: string;
94
+ base_dir: string | null;
95
+ test_path: string;
96
+ openapi_spec: string | null;
97
+ created_at: string;
98
+ total_runs: number;
99
+ pass_rate: number;
100
+ last_run_at: string | null;
101
+ last_run_passed: number;
102
+ last_run_failed: number;
103
+ last_run_total: number;
104
+ }
105
+
106
+ export interface CreateCollectionOpts {
107
+ name: string;
108
+ base_dir?: string;
109
+ test_path: string;
110
+ openapi_spec?: string;
111
+ }
112
+
113
+ export interface StoredStepResult {
114
+ id: number;
115
+ run_id: number;
116
+ suite_name: string;
117
+ test_name: string;
118
+ status: string;
119
+ duration_ms: number;
120
+ request_method: string | null;
121
+ request_url: string | null;
122
+ request_body: string | null;
123
+ response_status: number | null;
124
+ response_body: string | null;
125
+ response_headers: string | null;
126
+ error_message: string | null;
127
+ assertions: import("../../core/runner/types.ts").AssertionResult[];
128
+ captures: Record<string, unknown>;
129
+ suite_file: string | null;
130
+ provenance: import("../../core/parser/types.ts").SourceMetadata | null;
131
+ failure_class: import("../../core/diagnostics/failure-class.ts").FailureClass | null;
132
+ failure_class_reason: string | null;
133
+ spec_pointer: string | null;
134
+ spec_excerpt: string | null;
135
+ }
136
+
137
+ export interface RunFilters {
138
+ status?: string;
139
+ environment?: string;
140
+ date_from?: string;
141
+ date_to?: string;
142
+ test_name?: string;
143
+ /** TASK-116: filter by run origin — "ci" (CI runs only) or "manual"
144
+ * (interactive / probe / ad-hoc). Also accepts arbitrary trigger
145
+ * strings if the workspace introduces custom ones. */
146
+ trigger?: string;
147
+ }
148
+
149
+ export interface DashboardStats {
150
+ totalRuns: number;
151
+ totalTests: number;
152
+ overallPassRate: number;
153
+ avgDuration: number;
154
+ }
155
+
156
+ export interface PassRateTrendPoint {
157
+ run_id: number;
158
+ started_at: string;
159
+ pass_rate: number;
160
+ }
161
+
162
+ export interface SlowestTest {
163
+ suite_name: string;
164
+ test_name: string;
165
+ avg_duration: number;
166
+ }
167
+
168
+ export interface FlakyTest {
169
+ suite_name: string;
170
+ test_name: string;
171
+ distinct_statuses: number;
172
+ }