1use serde::{Deserialize, Serialize};
17
18pub const CATALOG_SCHEMA_VERSION: u32 = 1;
21
22#[derive(Debug, Clone, Serialize, Deserialize)]
27pub struct JsonEnvelope<T: Serialize> {
28 #[serde(rename = "schemaVersion")]
29 pub schema_version: u32,
30 pub ok: bool,
31 pub data: Option<T>,
32 pub error: Option<JsonError>,
33 #[serde(default)]
34 pub warnings: Vec<JsonWarning>,
35}
36
37#[derive(Debug, Clone, Serialize, Deserialize)]
38pub struct JsonError {
39 pub code: String,
40 pub message: String,
41 #[serde(default)]
45 pub details: serde_json::Value,
46}
47
48#[derive(Debug, Clone, Serialize, Deserialize)]
49pub struct JsonWarning {
50 pub code: String,
51 pub message: String,
52}
53
54pub trait JsonOutput {
59 const SCHEMA_VERSION: u32;
60 type Data: Serialize;
61 fn into_envelope(self) -> JsonEnvelope<Self::Data>;
62}
63
64impl<T: Serialize> JsonEnvelope<T> {
65 pub fn ok(schema_version: u32, data: T) -> Self {
66 Self {
67 schema_version,
68 ok: true,
69 data: Some(data),
70 error: None,
71 warnings: Vec::new(),
72 }
73 }
74
75 pub fn err(
76 schema_version: u32,
77 code: impl Into<String>,
78 message: impl Into<String>,
79 ) -> JsonEnvelope<T> {
80 Self {
81 schema_version,
82 ok: false,
83 data: None,
84 error: Some(JsonError {
85 code: code.into(),
86 message: message.into(),
87 details: serde_json::Value::Null,
88 }),
89 warnings: Vec::new(),
90 }
91 }
92
93 pub fn with_details(mut self, details: serde_json::Value) -> Self {
94 if let Some(err) = self.error.as_mut() {
95 err.details = details;
96 }
97 self
98 }
99
100 pub fn with_warning(mut self, code: impl Into<String>, message: impl Into<String>) -> Self {
101 self.warnings.push(JsonWarning {
102 code: code.into(),
103 message: message.into(),
104 });
105 self
106 }
107}
108
109#[derive(Debug, Clone, Serialize)]
113pub struct SchemaEntry {
114 pub command: &'static str,
115 #[serde(rename = "schemaVersion")]
116 pub schema_version: u32,
117 pub description: &'static str,
118 #[serde(skip_serializing_if = "Option::is_none", rename = "schemaJson")]
119 pub schema_json: Option<serde_json::Value>,
120}
121
122pub fn catalog() -> Vec<SchemaEntry> {
129 vec![
130 SchemaEntry {
131 command: "doctor",
132 schema_version: crate::commands::doctor::DOCTOR_SCHEMA_VERSION,
133 description: "Capability matrix: host, per-target buildability, per-provider reachability, per-stdlib-effect availability.",
134 schema_json: None,
135 },
136 SchemaEntry {
137 command: "session export",
138 schema_version: 1,
139 description: "Portable Harn session bundle export.",
140 schema_json: None,
141 },
142 SchemaEntry {
143 command: "provider-catalog",
144 schema_version: 1,
145 description: "Resolved provider/model catalog snapshot.",
146 schema_json: None,
147 },
148 SchemaEntry {
149 command: "connect status",
150 schema_version: 1,
151 description: "Outbound-connector readiness report.",
152 schema_json: None,
153 },
154 SchemaEntry {
155 command: "connect setup-plan",
156 schema_version: 1,
157 description: "Step-by-step plan to bring a connector online.",
158 schema_json: None,
159 },
160 SchemaEntry {
161 command: "run",
162 schema_version: crate::commands::run::json_events::RUN_JSON_SCHEMA_VERSION,
163 description: "Pipeline-run NDJSON event stream (stdout, stderr, transcript, tool, hook, persona, result, error).",
164 schema_json: None,
165 },
166 SchemaEntry {
167 command: "parse",
168 schema_version: crate::commands::parse_tokens::PARSE_JSON_SCHEMA_VERSION,
169 description: "Tagged Harn AST tree with byte spans for parser tooling.",
170 schema_json: None,
171 },
172 SchemaEntry {
173 command: "tokens",
174 schema_version: crate::commands::parse_tokens::TOKENS_JSON_SCHEMA_VERSION,
175 description: "Lexer token stream with source lexemes and byte spans.",
176 schema_json: None,
177 },
178 SchemaEntry {
179 command: "check",
180 schema_version: crate::commands::check::CHECK_SCHEMA_VERSION,
181 description: "Per-file static check results with diagnostics and summary counts.",
182 schema_json: None,
183 },
184 SchemaEntry {
185 command: "fmt",
186 schema_version: crate::commands::check::FMT_SCHEMA_VERSION,
187 description: "Per-file formatting result report for write and check modes.",
188 schema_json: None,
189 },
190 SchemaEntry {
191 command: "check provider-matrix",
192 schema_version: crate::commands::check::provider_matrix::PROVIDER_MATRIX_SCHEMA_VERSION,
193 description: "Provider/model capability matrix rows.",
194 schema_json: None,
195 },
196 SchemaEntry {
197 command: "check connector-matrix",
198 schema_version: crate::commands::check::connector_matrix::CONNECTOR_MATRIX_SCHEMA_VERSION,
199 description: "Connector package capability matrix rows.",
200 schema_json: None,
201 },
202 SchemaEntry {
203 command: "test conformance",
204 schema_version: crate::commands::test::CONFORMANCE_TEST_SCHEMA_VERSION,
205 description:
206 "Conformance test results with xfail accounting and a stable fixture snapshot key.",
207 schema_json: None,
208 },
209 SchemaEntry {
210 command: "time run",
211 schema_version: crate::commands::time::TIME_RUN_SCHEMA_VERSION,
212 description:
213 "Per-phase wall-clock + cache hit/miss + per-LLM/tool-call latency for `harn run`.",
214 schema_json: None,
215 },
216 SchemaEntry {
217 command: "fix plan",
218 schema_version: crate::commands::fix::FIX_PLAN_SCHEMA_VERSION,
219 description: "Plan repair-bearing diagnostics without editing files.",
220 schema_json: None,
221 },
222 SchemaEntry {
223 command: "fix apply",
224 schema_version: crate::commands::fix::FIX_APPLY_SCHEMA_VERSION,
225 description: "Apply clean repair edits at or below a declared safety ceiling.",
226 schema_json: None,
227 },
228 SchemaEntry {
229 command: "skills list",
230 schema_version: 1,
231 description: "Canonical Harn skill corpus, frontmatter only.",
232 schema_json: None,
233 },
234 SchemaEntry {
235 command: "skills get",
236 schema_version: 1,
237 description: "One canonical skill's frontmatter (and body with --full).",
238 schema_json: None,
239 },
240 SchemaEntry {
241 command: "pack",
242 schema_version: crate::commands::pack::PACK_SCHEMA_VERSION,
243 description: "Signed-ready .harnpack run-bundle build summary.",
244 schema_json: Some(crate::commands::pack::json_schema()),
245 },
246 SchemaEntry {
247 command: "dev",
248 schema_version: 1,
249 description: "`harn dev --watch` incremental NDJSON event stream (ready / fingerprint_changed / rerun / diagnostics / tests).",
250 schema_json: None,
251 },
252 SchemaEntry {
253 command: "routes",
254 schema_version: 1,
255 description: "Static trigger route, budget, capability, and vendor-lock inventory.",
256 schema_json: None,
257 },
258 SchemaEntry {
259 command: "graph",
260 schema_version: crate::commands::graph::GRAPH_SCHEMA_VERSION,
261 description:
262 "Static module graph with public symbols, imports, capabilities, effects, and host-call surface.",
263 schema_json: None,
264 },
265 SchemaEntry {
266 command: "explain --catalog",
267 schema_version: crate::commands::diagnostics_catalog::SCHEMA_VERSION,
268 description:
269 "Diagnostic-code catalog: per-code summary, repair, safety, related codes.",
270 schema_json: None,
271 },
272 ]
273}
274
275pub fn to_string_pretty<T: Serialize>(envelope: &JsonEnvelope<T>) -> String {
278 serde_json::to_string_pretty(envelope).expect("JsonEnvelope serializes")
279}
280
281#[cfg(test)]
282mod tests {
283 use super::*;
284 use serde_json::json;
285
286 #[derive(Serialize)]
287 struct Payload {
288 value: u32,
289 }
290
291 #[test]
292 fn ok_envelope_round_trips() {
293 let env = JsonEnvelope::ok(7, Payload { value: 42 });
294 let v: serde_json::Value = serde_json::to_value(&env).unwrap();
295 assert_eq!(v["schemaVersion"], 7);
296 assert_eq!(v["ok"], true);
297 assert_eq!(v["data"]["value"], 42);
298 assert!(v["error"].is_null());
301 assert_eq!(v["warnings"], json!([]));
302 }
303
304 #[test]
305 fn err_envelope_carries_details() {
306 let env: JsonEnvelope<()> = JsonEnvelope::err(2, "io", "disk full")
307 .with_details(json!({ "path": "/var/log/harn" }));
308 let v: serde_json::Value = serde_json::to_value(&env).unwrap();
309 assert_eq!(v["schemaVersion"], 2);
310 assert_eq!(v["ok"], false);
311 assert_eq!(v["error"]["code"], "io");
312 assert_eq!(v["error"]["message"], "disk full");
313 assert_eq!(v["error"]["details"]["path"], "/var/log/harn");
314 assert!(v["data"].is_null());
315 }
316
317 #[test]
318 fn warnings_serialize_when_present() {
319 let env = JsonEnvelope::ok(1, Payload { value: 1 })
320 .with_warning("deprecated.flag", "--format=json is deprecated");
321 let v: serde_json::Value = serde_json::to_value(&env).unwrap();
322 assert_eq!(v["warnings"][0]["code"], "deprecated.flag");
323 assert_eq!(v["warnings"][0]["message"], "--format=json is deprecated");
324 }
325
326 #[test]
327 fn catalog_is_nonempty_and_unique() {
328 let entries = catalog();
329 assert!(!entries.is_empty(), "catalog should ship with E2.1 seeds");
330 let mut commands: Vec<_> = entries.iter().map(|e| e.command).collect();
331 commands.sort();
332 let unique_count = {
333 let mut deduped = commands.clone();
334 deduped.dedup();
335 deduped.len()
336 };
337 assert_eq!(commands.len(), unique_count, "command names must be unique");
338 }
339
340 #[test]
341 fn catalog_includes_fix_plan() {
342 let entries = catalog();
343 let entry = entries
344 .iter()
345 .find(|entry| entry.command == "fix plan")
346 .expect("fix plan schema should be registered");
347 assert_eq!(
348 entry.schema_version,
349 crate::commands::fix::FIX_PLAN_SCHEMA_VERSION
350 );
351 let entry = entries
352 .iter()
353 .find(|entry| entry.command == "fix apply")
354 .expect("fix apply schema should be registered");
355 assert_eq!(
356 entry.schema_version,
357 crate::commands::fix::FIX_APPLY_SCHEMA_VERSION
358 );
359 }
360
361 #[test]
362 fn schema_versions_are_positive() {
363 for entry in catalog() {
364 assert!(
365 entry.schema_version >= 1,
366 "{} should have schemaVersion >= 1",
367 entry.command
368 );
369 }
370 }
371}