@kirrosh/zond 0.22.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 +648 -0
- package/README.md +58 -6
- package/package.json +9 -6
- 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 +43 -0
- package/src/cli/commands/clean.ts +212 -0
- package/src/cli/commands/cleanup.ts +262 -0
- package/src/cli/commands/completions.ts +16 -0
- package/src/cli/commands/coverage.ts +605 -132
- package/src/cli/commands/db.ts +178 -7
- 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 -46
- package/src/cli/commands/init/bootstrap.ts +30 -1
- package/src/cli/commands/{init.ts → init/index.ts} +99 -5
- package/src/cli/commands/init/skills.ts +56 -3
- package/src/cli/commands/init/templates/agents.md +65 -61
- 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 +592 -125
- package/src/cli/commands/init/templates/zond-config.yml +8 -9
- 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 +842 -53
- package/src/cli/commands/session.ts +244 -0
- package/src/cli/commands/use.ts +18 -1
- package/src/cli/index.ts +20 -3
- package/src/cli/json-envelope.ts +112 -3
- package/src/cli/json-schemas.ts +263 -0
- package/src/cli/program.ts +198 -635
- 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 +5 -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 +22 -6
- 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 +151 -11
- 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 +42 -16
- 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 +445 -19
- 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 +37 -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 +103 -13
- package/src/core/generator/suite-generator.ts +419 -111
- 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 +129 -4
- package/src/core/parser/types.ts +19 -1
- package/src/core/parser/variables.ts +0 -0
- package/src/core/parser/yaml-parser.ts +58 -12
- 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 +43 -76
- package/src/core/probe/method-shared.ts +69 -0
- package/src/core/probe/negative-probe.ts +183 -149
- 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 +41 -2
- package/src/core/reporter/index.ts +2 -3
- package/src/core/reporter/json.ts +11 -1
- 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 +58 -1
- 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 +264 -20
- package/src/core/runner/form-encode.ts +51 -0
- package/src/core/runner/http-client.ts +75 -2
- 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 +89 -17
- 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 +415 -16
- 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/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 +178 -50
- package/src/cli/commands/export.ts +0 -144
- package/src/cli/commands/guide.ts +0 -127
- package/src/cli/commands/init/templates/skills/scenarios.md +0 -97
- package/src/cli/commands/probe-methods.ts +0 -108
- package/src/cli/commands/probe-validation.ts +0 -124
- package/src/cli/commands/serve.ts +0 -114
- package/src/cli/commands/sync.ts +0 -268
- package/src/cli/commands/update.ts +0 -189
- package/src/cli/commands/validate.ts +0 -34
- package/src/core/diagnostics/render-md.ts +0 -112
- package/src/core/exporter/postman.ts +0 -963
- package/src/core/generator/guide-builder.ts +0 -253
- package/src/core/meta/types.ts +0 -19
- 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
|
@@ -0,0 +1,244 @@
|
|
|
1
|
+
import { randomUUID } from "node:crypto";
|
|
2
|
+
import {
|
|
3
|
+
clearCurrentSession,
|
|
4
|
+
readCurrentSession,
|
|
5
|
+
sessionFilePath,
|
|
6
|
+
writeCurrentSession,
|
|
7
|
+
type SessionRecord,
|
|
8
|
+
} from "../../core/context/session.ts";
|
|
9
|
+
import { jsonError, jsonOk, printJson } from "../json-envelope.ts";
|
|
10
|
+
import { printError, printSuccess } from "../output.ts";
|
|
11
|
+
import { listSessions, countSessions } from "../../db/queries.ts";
|
|
12
|
+
import { getDb } from "../../db/schema.ts";
|
|
13
|
+
import { parsePositiveInt } from "../argv.ts";
|
|
14
|
+
|
|
15
|
+
export interface SessionStartOptions {
|
|
16
|
+
label?: string;
|
|
17
|
+
id?: string;
|
|
18
|
+
json?: boolean;
|
|
19
|
+
/** ARV-155: replace the existing active session instead of erroring out
|
|
20
|
+
* ("Session already active …"). Useful in ralph-loop iterations where a
|
|
21
|
+
* previous turn left a stale `.zond/current-session` behind. */
|
|
22
|
+
force?: boolean;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
export interface SessionEndOptions {
|
|
26
|
+
json?: boolean;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export interface SessionStatusOptions {
|
|
30
|
+
json?: boolean;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
function isValidUuid(s: string): boolean {
|
|
34
|
+
// Permissive — accept any RFC4122-shaped UUID (v1-v8) and zero-UUID for tests.
|
|
35
|
+
return /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i.test(s);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
export async function sessionStartCommand(opts: SessionStartOptions): Promise<number> {
|
|
39
|
+
const existing = readCurrentSession();
|
|
40
|
+
if (existing && !opts.force) {
|
|
41
|
+
const message =
|
|
42
|
+
`Session already active (${existing.id}). Run 'zond session end' first, or pass --force to replace it.`;
|
|
43
|
+
if (opts.json) printJson(jsonError("session", [message]));
|
|
44
|
+
else printError(message);
|
|
45
|
+
return 1;
|
|
46
|
+
}
|
|
47
|
+
if (existing && opts.force) {
|
|
48
|
+
clearCurrentSession();
|
|
49
|
+
if (!opts.json) {
|
|
50
|
+
process.stdout.write(` Replaced active session ${existing.id}${existing.label ? ` (${existing.label})` : ""}.\n`);
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
let id = opts.id?.trim();
|
|
55
|
+
if (id && !isValidUuid(id)) {
|
|
56
|
+
const message = `Invalid --id: ${id} is not a UUID`;
|
|
57
|
+
if (opts.json) printJson(jsonError("session", [message]));
|
|
58
|
+
else printError(message);
|
|
59
|
+
return 1;
|
|
60
|
+
}
|
|
61
|
+
if (!id) id = randomUUID();
|
|
62
|
+
|
|
63
|
+
const record: SessionRecord = {
|
|
64
|
+
id,
|
|
65
|
+
label: opts.label?.trim() || undefined,
|
|
66
|
+
started_at: new Date().toISOString(),
|
|
67
|
+
};
|
|
68
|
+
|
|
69
|
+
let path: string;
|
|
70
|
+
try {
|
|
71
|
+
path = writeCurrentSession(record);
|
|
72
|
+
} catch (err) {
|
|
73
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
74
|
+
if (opts.json) printJson(jsonError("session", [message]));
|
|
75
|
+
else printError(message);
|
|
76
|
+
return 1;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
if (opts.json) {
|
|
80
|
+
printJson(jsonOk("session", { action: "started", ...record, path }));
|
|
81
|
+
} else {
|
|
82
|
+
printSuccess(`Session ${record.id} started${record.label ? ` (${record.label})` : ""}`);
|
|
83
|
+
process.stdout.write(` All subsequent 'zond run' calls in this workspace inherit this session_id.\n`);
|
|
84
|
+
process.stdout.write(` Stored at ${path}. Run 'zond session end' to clear.\n`);
|
|
85
|
+
}
|
|
86
|
+
return 0;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
export async function sessionEndCommand(opts: SessionEndOptions): Promise<number> {
|
|
90
|
+
const existing = readCurrentSession();
|
|
91
|
+
const path = sessionFilePath();
|
|
92
|
+
const removed = clearCurrentSession();
|
|
93
|
+
|
|
94
|
+
if (opts.json) {
|
|
95
|
+
printJson(jsonOk("session", {
|
|
96
|
+
action: "ended",
|
|
97
|
+
removed,
|
|
98
|
+
previous_id: existing?.id ?? null,
|
|
99
|
+
path,
|
|
100
|
+
}));
|
|
101
|
+
return 0;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
if (!removed) {
|
|
105
|
+
process.stdout.write(`No active session (${path} not present).\n`);
|
|
106
|
+
return 0;
|
|
107
|
+
}
|
|
108
|
+
printSuccess(`Session ${existing?.id ?? "(unknown)"} ended`);
|
|
109
|
+
return 0;
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
export async function sessionStatusCommand(opts: SessionStatusOptions): Promise<number> {
|
|
113
|
+
const record = readCurrentSession();
|
|
114
|
+
const path = sessionFilePath();
|
|
115
|
+
const env = process.env.ZOND_SESSION_ID?.trim() || null;
|
|
116
|
+
|
|
117
|
+
if (opts.json) {
|
|
118
|
+
printJson(jsonOk("session", {
|
|
119
|
+
action: "status",
|
|
120
|
+
active: !!record,
|
|
121
|
+
session: record,
|
|
122
|
+
env_session_id: env,
|
|
123
|
+
path,
|
|
124
|
+
}));
|
|
125
|
+
return 0;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
if (record) {
|
|
129
|
+
process.stdout.write(`session_id: ${record.id}\n`);
|
|
130
|
+
if (record.label) process.stdout.write(`label: ${record.label}\n`);
|
|
131
|
+
process.stdout.write(`started_at: ${record.started_at}\n`);
|
|
132
|
+
process.stdout.write(`path: ${path}\n`);
|
|
133
|
+
if (env && env !== record.id) {
|
|
134
|
+
process.stdout.write(`\nNote: ZOND_SESSION_ID is set to '${env}' and overrides the file.\n`);
|
|
135
|
+
}
|
|
136
|
+
return 0;
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
if (env) {
|
|
140
|
+
process.stdout.write(`No active session file. ZOND_SESSION_ID is set to '${env}' (env wins).\n`);
|
|
141
|
+
return 0;
|
|
142
|
+
}
|
|
143
|
+
process.stdout.write(
|
|
144
|
+
`No active session. Run 'zond session start' to group subsequent 'zond run' calls.\n`,
|
|
145
|
+
);
|
|
146
|
+
return 0;
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
// ARV-43: list past sessions surfaced from the runs table so users can
|
|
150
|
+
// discover session_ids for `zond coverage --union session --session-id <id>`
|
|
151
|
+
// without dropping into sqlite. Labels live only in .zond/current-session
|
|
152
|
+
// (not persisted per run), so we only show what's stored in the DB.
|
|
153
|
+
export interface SessionListOptions {
|
|
154
|
+
limit?: number;
|
|
155
|
+
json?: boolean;
|
|
156
|
+
dbPath?: string;
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
export async function sessionListCommand(opts: SessionListOptions): Promise<number> {
|
|
160
|
+
const limit = opts.limit ?? 20;
|
|
161
|
+
let sessions: ReturnType<typeof listSessions>;
|
|
162
|
+
let total: number;
|
|
163
|
+
try {
|
|
164
|
+
getDb(opts.dbPath);
|
|
165
|
+
sessions = listSessions(limit);
|
|
166
|
+
total = countSessions();
|
|
167
|
+
} catch (err) {
|
|
168
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
169
|
+
if (opts.json) printJson(jsonError("session", [message]));
|
|
170
|
+
else printError(message);
|
|
171
|
+
return 1;
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
if (opts.json) {
|
|
175
|
+
printJson(jsonOk("session", { action: "list", limit, total, sessions }));
|
|
176
|
+
return 0;
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
if (sessions.length === 0) {
|
|
180
|
+
process.stdout.write("No sessions recorded yet. Run 'zond session start' before 'zond run' to group runs.\n");
|
|
181
|
+
return 0;
|
|
182
|
+
}
|
|
183
|
+
process.stdout.write(`Showing ${sessions.length} of ${total} session(s):\n\n`);
|
|
184
|
+
process.stdout.write("session_id started_at finished_at runs pass/fail/skip\n");
|
|
185
|
+
for (const s of sessions) {
|
|
186
|
+
const started = s.started_at ? s.started_at.replace("T", " ").slice(0, 19) : "—".padEnd(19);
|
|
187
|
+
const finished = s.finished_at ? s.finished_at.replace("T", " ").slice(0, 19) : "(open)".padEnd(19);
|
|
188
|
+
const runs = String(s.run_count).padStart(4);
|
|
189
|
+
const counts = `${s.passed}/${s.failed}/${s.skipped}`;
|
|
190
|
+
process.stdout.write(`${s.session_id} ${started} ${finished} ${runs} ${counts}\n`);
|
|
191
|
+
}
|
|
192
|
+
return 0;
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
import type { Command } from "commander";
|
|
196
|
+
import { globalJson } from "../resolve.ts";
|
|
197
|
+
|
|
198
|
+
export function registerSession(program: Command): void {
|
|
199
|
+
// Group multiple `zond run` calls under one session_id without juggling env
|
|
200
|
+
// vars. `start` writes a UUID to .zond/current-session; subsequent `run`
|
|
201
|
+
// calls auto-pick it up (priority: --session-id flag > ZOND_SESSION_ID env
|
|
202
|
+
// > current-session file).
|
|
203
|
+
const session = program.command("session").description("Manage run grouping (campaigns)");
|
|
204
|
+
session
|
|
205
|
+
.command("start")
|
|
206
|
+
.description("Begin a session — group all subsequent 'zond run' calls under one session_id (.zond/current-session)")
|
|
207
|
+
.option("--label <text>", "Optional human-readable label shown alongside the session in the UI")
|
|
208
|
+
.option("--id <uuid>", "Reuse a specific UUID instead of generating one (useful for CI)")
|
|
209
|
+
.option("--force", "Replace any already-active session instead of erroring out (ARV-155)")
|
|
210
|
+
.action(async (opts, cmd: Command) => {
|
|
211
|
+
process.exitCode = await sessionStartCommand({
|
|
212
|
+
label: opts.label,
|
|
213
|
+
id: opts.id,
|
|
214
|
+
force: opts.force === true,
|
|
215
|
+
json: globalJson(cmd),
|
|
216
|
+
});
|
|
217
|
+
});
|
|
218
|
+
session
|
|
219
|
+
.command("end")
|
|
220
|
+
.description("End the current session — remove .zond/current-session")
|
|
221
|
+
.action(async (_opts, cmd: Command) => {
|
|
222
|
+
process.exitCode = await sessionEndCommand({ json: globalJson(cmd) });
|
|
223
|
+
});
|
|
224
|
+
session
|
|
225
|
+
.command("status")
|
|
226
|
+
.description("Show the active session (if any)")
|
|
227
|
+
.action(async (_opts, cmd: Command) => {
|
|
228
|
+
process.exitCode = await sessionStatusCommand({ json: globalJson(cmd) });
|
|
229
|
+
});
|
|
230
|
+
// ARV-43: complete the start/end/status/list quartet so coverage --union
|
|
231
|
+
// session --session-id <id> is discoverable without sqlite spelunking.
|
|
232
|
+
session
|
|
233
|
+
.command("list")
|
|
234
|
+
.description("List recent sessions (id, started_at, finished_at, run counts) so coverage --session-id is discoverable")
|
|
235
|
+
.option("--limit <n>", "Max sessions to print (default 20)", parsePositiveInt("--limit"))
|
|
236
|
+
.option("--db <path>", "Path to SQLite database file")
|
|
237
|
+
.action(async (opts, cmd: Command) => {
|
|
238
|
+
process.exitCode = await sessionListCommand({
|
|
239
|
+
limit: opts.limit,
|
|
240
|
+
dbPath: opts.db,
|
|
241
|
+
json: globalJson(cmd),
|
|
242
|
+
});
|
|
243
|
+
});
|
|
244
|
+
}
|
package/src/cli/commands/use.ts
CHANGED
|
@@ -23,7 +23,7 @@ export async function useCommand(opts: UseOptions): Promise<number> {
|
|
|
23
23
|
} else if (removed) {
|
|
24
24
|
printSuccess(`Cleared ${path}`);
|
|
25
25
|
} else {
|
|
26
|
-
process.stdout.write(`No .zond-
|
|
26
|
+
process.stdout.write(`No .zond/current-api file in ${process.cwd()}\n`);
|
|
27
27
|
}
|
|
28
28
|
return 0;
|
|
29
29
|
}
|
|
@@ -55,3 +55,20 @@ export async function useCommand(opts: UseOptions): Promise<number> {
|
|
|
55
55
|
}
|
|
56
56
|
return 0;
|
|
57
57
|
}
|
|
58
|
+
|
|
59
|
+
import type { Command } from "commander";
|
|
60
|
+
import { globalJson } from "../resolve.ts";
|
|
61
|
+
|
|
62
|
+
export function registerUse(program: Command): void {
|
|
63
|
+
program
|
|
64
|
+
.command("use [api]")
|
|
65
|
+
.description("Set or show the current API for this workspace (.zond/current-api). TASK-290: resolution chain is per-cmd --api > global --api > ZOND_API env > this file.")
|
|
66
|
+
.option("--clear", "Remove .zond/current-api from the workspace")
|
|
67
|
+
.action(async (api: string | undefined, opts, cmd: Command) => {
|
|
68
|
+
process.exitCode = await useCommand({
|
|
69
|
+
api,
|
|
70
|
+
clear: opts.clear === true,
|
|
71
|
+
json: globalJson(cmd),
|
|
72
|
+
});
|
|
73
|
+
});
|
|
74
|
+
}
|
package/src/cli/index.ts
CHANGED
|
@@ -1,12 +1,29 @@
|
|
|
1
1
|
#!/usr/bin/env bun
|
|
2
2
|
|
|
3
|
+
import { createHash } from "node:crypto";
|
|
3
4
|
import { CommanderError } from "commander";
|
|
4
5
|
import { buildProgram, preprocessArgv } from "./program.ts";
|
|
5
|
-
import { printError } from "./output.ts";
|
|
6
6
|
import { VERSION } from "./version.ts";
|
|
7
7
|
|
|
8
8
|
export { VERSION };
|
|
9
9
|
|
|
10
|
+
/**
|
|
11
|
+
* Anything that reaches this handler is an *unexpected* throw — command
|
|
12
|
+
* implementations are expected to catch their own usage/config errors and
|
|
13
|
+
* return exit 2 themselves. Tag these with `[zond:internal]` so operators
|
|
14
|
+
* can tell them apart from sandbox/SIGKILL/OOM (137/143). See ZOND.md →
|
|
15
|
+
* "Exit codes" for the full taxonomy.
|
|
16
|
+
*/
|
|
17
|
+
function reportInternalError(err: unknown): void {
|
|
18
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
19
|
+
const stack = err instanceof Error && err.stack ? err.stack : message;
|
|
20
|
+
const stackHash = createHash("sha1").update(stack).digest("hex").slice(0, 8);
|
|
21
|
+
process.stderr.write(`[zond:internal] zond ${VERSION} — uncaught ${message} (stack ${stackHash})\n`);
|
|
22
|
+
if (err instanceof Error && err.stack) {
|
|
23
|
+
process.stderr.write(err.stack + "\n");
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
|
|
10
27
|
async function main(): Promise<void> {
|
|
11
28
|
const program = buildProgram();
|
|
12
29
|
try {
|
|
@@ -22,8 +39,8 @@ async function main(): Promise<void> {
|
|
|
22
39
|
process.exitCode = 2;
|
|
23
40
|
return;
|
|
24
41
|
}
|
|
25
|
-
|
|
26
|
-
process.exitCode =
|
|
42
|
+
reportInternalError(err);
|
|
43
|
+
process.exitCode = 3;
|
|
27
44
|
}
|
|
28
45
|
}
|
|
29
46
|
|
package/src/cli/json-envelope.ts
CHANGED
|
@@ -1,19 +1,128 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Single source of truth for `--json` output across all CLI commands.
|
|
3
|
+
*
|
|
4
|
+
* Every `--json` response carries the same envelope:
|
|
5
|
+
*
|
|
6
|
+
* { ok, command, data, warnings, errors, exit_code? }
|
|
7
|
+
*
|
|
8
|
+
* Commands construct the payload (`data`) and ask one of the helpers
|
|
9
|
+
* below to render it. Don't `console.log(JSON.stringify(...))` ad-hoc
|
|
10
|
+
* for `--json` paths — go through `printJson` / `writeEnvelope` so the
|
|
11
|
+
* shape stays uniform (TASK-73, TASK-74, closed by TASK-184).
|
|
12
|
+
*
|
|
13
|
+
* TASK-296: errors[] is a list of `{code, message, details?}` so an
|
|
14
|
+
* agent can route on `code` without parsing the human message. Helpers
|
|
15
|
+
* accept either a bare `string` (auto-wrapped with code `unknown_error`)
|
|
16
|
+
* or a structured `ZondError` to keep call sites short.
|
|
17
|
+
*/
|
|
18
|
+
|
|
19
|
+
// TASK-295: types are derived from the zod schemas in `./json-schemas.ts`
|
|
20
|
+
// so the published JSON Schema (docs/json-schema/) and the runtime types
|
|
21
|
+
// can never drift. Edit the enum/object there, run `bun run schemas`,
|
|
22
|
+
// commit the regenerated docs.
|
|
23
|
+
import type { z } from "zod";
|
|
24
|
+
import {
|
|
25
|
+
ZondErrorCodeSchema,
|
|
26
|
+
ZondErrorSchema,
|
|
27
|
+
} from "./json-schemas.ts";
|
|
28
|
+
|
|
29
|
+
export type ZondErrorCode = z.infer<typeof ZondErrorCodeSchema>;
|
|
30
|
+
export type ZondError = z.infer<typeof ZondErrorSchema>;
|
|
31
|
+
|
|
1
32
|
export interface JsonEnvelope<T = unknown> {
|
|
2
33
|
ok: boolean;
|
|
3
34
|
command: string;
|
|
4
35
|
data: T;
|
|
5
36
|
warnings: string[];
|
|
6
|
-
errors:
|
|
37
|
+
errors: ZondError[];
|
|
38
|
+
/** Exit code the process will return. Present on error envelopes so a
|
|
39
|
+
* caller can read the taxonomy class without re-parsing $? from a shell
|
|
40
|
+
* (see ZOND.md → "Exit codes"). */
|
|
41
|
+
exit_code?: number;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/** Accept either a bare string (auto-coded `unknown_error`) or a fully-
|
|
45
|
+
* structured `ZondError`. Lets us migrate ~100 call sites incrementally
|
|
46
|
+
* without breaking the schema. */
|
|
47
|
+
export type ErrorInput = string | ZondError;
|
|
48
|
+
|
|
49
|
+
function normalizeErrors(errs: readonly ErrorInput[]): ZondError[] {
|
|
50
|
+
return errs.map(e =>
|
|
51
|
+
typeof e === "string" ? { code: "unknown_error" as const, message: e } : e,
|
|
52
|
+
);
|
|
7
53
|
}
|
|
8
54
|
|
|
9
55
|
export function jsonOk<T>(command: string, data: T, warnings?: string[]): JsonEnvelope<T> {
|
|
10
56
|
return { ok: true, command, data, warnings: warnings ?? [], errors: [] };
|
|
11
57
|
}
|
|
12
58
|
|
|
13
|
-
export function jsonError(
|
|
14
|
-
|
|
59
|
+
export function jsonError(
|
|
60
|
+
command: string,
|
|
61
|
+
errors: readonly ErrorInput[],
|
|
62
|
+
warnings?: string[],
|
|
63
|
+
exitCode = 2,
|
|
64
|
+
): JsonEnvelope<null> {
|
|
65
|
+
return {
|
|
66
|
+
ok: false,
|
|
67
|
+
command,
|
|
68
|
+
data: null,
|
|
69
|
+
warnings: warnings ?? [],
|
|
70
|
+
errors: normalizeErrors(errors),
|
|
71
|
+
exit_code: exitCode,
|
|
72
|
+
};
|
|
15
73
|
}
|
|
16
74
|
|
|
17
75
|
export function printJson(envelope: JsonEnvelope): void {
|
|
18
76
|
process.stdout.write(JSON.stringify(envelope, null, 2) + "\n");
|
|
19
77
|
}
|
|
78
|
+
|
|
79
|
+
/** Convenience constructor: `zerr("env_missing", "base_url not set", { var: "base_url" })`. */
|
|
80
|
+
export function zerr(code: ZondErrorCode, message: string, details?: Record<string, unknown>): ZondError {
|
|
81
|
+
return details ? { code, message, details } : { code, message };
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
/** Discriminated-union result an action can hand back to {@link writeEnvelope}. */
|
|
85
|
+
export type EnvelopeResult<T> =
|
|
86
|
+
| { ok: true; data: T; warnings?: string[] }
|
|
87
|
+
| { ok: false; errors: readonly ErrorInput[]; warnings?: string[]; exitCode?: number };
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* Render a typed `EnvelopeResult` to stdout as a JSON envelope and return
|
|
91
|
+
* the process exit code (0 on ok, `result.exitCode ?? 2` on error). This
|
|
92
|
+
* is the recommended wrapper for new commands — it lets the handler
|
|
93
|
+
* focus on producing a payload and surfaces the right exit code in one
|
|
94
|
+
* call:
|
|
95
|
+
*
|
|
96
|
+
* const result = await doWork();
|
|
97
|
+
* if (options.json) return writeEnvelope("my-cmd", result);
|
|
98
|
+
* // …human path…
|
|
99
|
+
*/
|
|
100
|
+
export function writeEnvelope<T>(command: string, result: EnvelopeResult<T>): number {
|
|
101
|
+
if (result.ok) {
|
|
102
|
+
printJson(jsonOk(command, result.data, result.warnings));
|
|
103
|
+
return 0;
|
|
104
|
+
}
|
|
105
|
+
const exit = result.exitCode ?? 2;
|
|
106
|
+
printJson(jsonError(command, result.errors, result.warnings, exit));
|
|
107
|
+
return exit;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
/**
|
|
111
|
+
* High-order wrapper for `--json` action handlers. Accepts an async
|
|
112
|
+
* producer and renders its return value (or thrown error) as an
|
|
113
|
+
* envelope. Errors thrown synchronously or asynchronously become
|
|
114
|
+
* `{ ok: false, errors: [{code: "unknown_error", message}] }` with
|
|
115
|
+
* `exitCode = 2`.
|
|
116
|
+
*/
|
|
117
|
+
export async function withEnvelope<T>(
|
|
118
|
+
command: string,
|
|
119
|
+
produce: () => Promise<{ data: T; warnings?: string[] }>,
|
|
120
|
+
): Promise<number> {
|
|
121
|
+
try {
|
|
122
|
+
const { data, warnings } = await produce();
|
|
123
|
+
return writeEnvelope(command, { ok: true, data, warnings });
|
|
124
|
+
} catch (err) {
|
|
125
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
126
|
+
return writeEnvelope(command, { ok: false, errors: [message] });
|
|
127
|
+
}
|
|
128
|
+
}
|