kanade_shared/wire/command.rs
1use chrono::{DateTime, Utc};
2use serde::{Deserialize, Serialize};
3
4use super::Staleness;
5use crate::manifest::{CheckHint, CollectHint, EmitConfig};
6
7#[derive(Serialize, Deserialize, schemars::JsonSchema, Debug, Clone)]
8pub struct Command {
9 pub id: String,
10 pub version: String,
11 pub request_id: String,
12 /// v0.29 / Issue #19: the deployment / scheduler-fire UUID this
13 /// Command belongs to. Forwarded into `ExecResult.exec_id` by the
14 /// agent so the projector can attribute results back to the
15 /// originating `executions` row. `None` for ad-hoc `kanade run`
16 /// (no deployment row exists). Pre-v0.29 wire used the field name
17 /// `job_id` for this same value — `serde(alias)` keeps old
18 /// publishes in STREAM_EXEC decodable across the upgrade window.
19 #[serde(alias = "job_id")]
20 pub exec_id: Option<String>,
21 pub shell: Shell,
22 /// Inline script body, OR empty when [`script_object`] is set.
23 /// Mutually exclusive with `script_object` at the wire level —
24 /// backend builders fill one or the other (never both) and the
25 /// agent's resolver picks the populated one. Pre-v0.43 wire
26 /// always carries this populated.
27 ///
28 /// [`script_object`]: Self::script_object
29 pub script: String,
30 /// SPEC §2.4.1 / yukimemi/kanade#210: Object Store reference
31 /// (`<name>/<version>` key into `OBJECT_SCRIPTS`). When set,
32 /// the agent fetches the body via `script_cache` and verifies
33 /// its sha256 against [`script_object_sha256`] before launching.
34 /// `None` ⇒ inline `script` carries the body (legacy + the
35 /// majority of jobs).
36 ///
37 /// [`script_object_sha256`]: Self::script_object_sha256
38 #[serde(default, skip_serializing_if = "Option::is_none")]
39 pub script_object: Option<String>,
40 /// Hex-encoded sha256 of the bytes the operator approved at
41 /// Command-build time. Required when [`script_object`] is set;
42 /// the agent treats a mismatch on fetch as "operator
43 /// re-uploaded the script between exec submission and agent
44 /// fire" and aborts the run rather than silently executing the
45 /// new bytes. Pre-v0.43 wire omits this; the resolver path
46 /// requires both fields to be `Some`.
47 ///
48 /// [`script_object`]: Self::script_object
49 #[serde(default, skip_serializing_if = "Option::is_none")]
50 pub script_object_sha256: Option<String>,
51 pub timeout_secs: u64,
52 pub jitter_secs: Option<u64>,
53 /// Which (token, session) combination the agent should launch the
54 /// child process under (v0.21). Defaults to [`RunAs::System`] for
55 /// back-compat with pre-v0.21 backends that don't send this field.
56 #[serde(default)]
57 pub run_as: RunAs,
58 /// Working directory for the spawned child (v0.21.1). `None` ⇒
59 /// inherit the agent's cwd. Pre-v0.21.1 wire payloads omit this
60 /// field and parse fine via `#[serde(default)]`.
61 #[serde(default, skip_serializing_if = "Option::is_none")]
62 pub cwd: Option<String>,
63 /// Absolute time after which the agent should refuse to run
64 /// this Command (v0.22). Set by the scheduler from
65 /// `Schedule.starting_deadline` (humantime) measured against
66 /// the cron tick time. `None` ⇒ no deadline, run whenever
67 /// received (default for ad-hoc `kanade exec` + back-compat
68 /// for pre-v0.22 wire). The agent stamps a synthetic
69 /// `ExecResult { exit_code: 125, stderr: "skipped: deadline
70 /// expired ..." }` when it skips, so the operator sees the
71 /// outcome on the Results / Dashboard pages instead of silence.
72 #[serde(default, skip_serializing_if = "Option::is_none")]
73 pub deadline_at: Option<DateTime<Utc>>,
74 /// v0.26: Manifest-declared Layer 2 staleness policy
75 /// (see SPEC.md §2.6.2). Forwarded from `Manifest.staleness` so
76 /// the agent can evaluate it at fire time without re-fetching the
77 /// Manifest from `BUCKET_JOBS`. Pre-v0.26 wire omits this and
78 /// `#[serde(default)]` falls back to `Staleness::Cached`, matching
79 /// pre-v0.26 behaviour (silently use cached KV values).
80 #[serde(default)]
81 pub staleness: Staleness,
82 /// Issue #246: forwarded from `Manifest.emit` so the agent
83 /// doesn't have to re-fetch the manifest at fire time. When
84 /// `Some` and `EmitKind::Events`, the agent parses script
85 /// stdout as NDJSON `ObsEvent` and publishes each line on
86 /// `obs.<pc_id>`. Pre-#246 wire omits this; the `#[serde(default)]`
87 /// fallback to `None` preserves prior behaviour (stdout flows
88 /// to `ExecResult` unchanged).
89 #[serde(default, skip_serializing_if = "Option::is_none")]
90 pub emit: Option<EmitConfig>,
91 /// #290: forwarded from `Manifest.check` so the agent can build a
92 /// KLP Health-tab [`Check`](crate::ipc::state::Check) from the
93 /// job's stdout without re-fetching the Manifest. When `Some`, the
94 /// agent reads the `status_field` / `detail_field` values out of
95 /// the stdout JSON object after a successful run and caches the
96 /// result into `StateSnapshot.checks`. Pre-#290 wire omits this;
97 /// `#[serde(default)]` → `None` preserves prior behaviour.
98 #[serde(default, skip_serializing_if = "Option::is_none")]
99 pub check: Option<CheckHint>,
100 /// #219: forwarded from `Manifest.collect` so the agent can bundle
101 /// the script's listed files without re-fetching the Manifest. When
102 /// `Some`, the agent — after a successful run — reads the
103 /// `files_field` path array out of the stdout JSON object, zips those
104 /// files (capped at `max_size`), uploads the archive to
105 /// `OBJECT_COLLECTIONS`, and records the key in
106 /// [`ExecResult::collect_object`](super::ExecResult::collect_object).
107 /// Pre-#219 wire omits this; `#[serde(default)]` → `None` preserves
108 /// prior behaviour.
109 #[serde(default, skip_serializing_if = "Option::is_none")]
110 pub collect: Option<CollectHint>,
111 /// #418 Phase 4: lowered from `Schedule.on_failure.retry` by the
112 /// command builders (backend `exec_manifest` + the agent's local
113 /// scheduler). When `Some`, the agent re-runs the script
114 /// in-process on a non-zero exit / timeout, up to `max` extra
115 /// attempts with `backoff_secs` between them, before publishing
116 /// the final outcome. `None` (default) ⇒ no retry, the historical
117 /// behaviour and what ad-hoc `kanade run` / `kanade exec` use.
118 /// Pre-Phase-4 wire omits this; `#[serde(default)]` → `None`.
119 #[serde(default, skip_serializing_if = "Option::is_none")]
120 pub retry: Option<RetrySpec>,
121 /// Job-generic post-step hook lowered from `Manifest.finalize`. When
122 /// `Some` and the main script exits cleanly, the agent runs this hook
123 /// after the collect step (injecting `KANADE_COLLECT_RESULT` for a
124 /// `collect:` job) so the operator can delete / move / notify.
125 /// Best-effort — a finalize failure is logged, never published as the
126 /// run's outcome. Pre-finalize wire omits this; `#[serde(default)]` →
127 /// `None`.
128 #[serde(default, skip_serializing_if = "Option::is_none")]
129 pub finalize: Option<FinalizeCommand>,
130}
131
132/// Lowered, engine-vocabulary form of [`crate::manifest::FinalizeSpec`]
133/// — the post-step hook stamped onto a [`Command`]. The operator-facing
134/// humantime `timeout` is reduced to whole seconds at build time
135/// (mirrors `timeout_secs`), and the manifest `ExecuteShell` to the wire
136/// [`Shell`], so the agent's fire path does no parsing.
137#[derive(Serialize, Deserialize, schemars::JsonSchema, Debug, Clone)]
138pub struct FinalizeCommand {
139 pub shell: Shell,
140 /// Inline script body (inline-only in P1).
141 pub script: String,
142 pub timeout_secs: u64,
143 #[serde(default)]
144 pub run_as: RunAs,
145 #[serde(default, skip_serializing_if = "Option::is_none")]
146 pub cwd: Option<String>,
147}
148
149/// Lowered, engine-vocabulary form of [`crate::manifest::Retry`] — a
150/// fixed-backoff retry policy stamped onto a [`Command`]. The
151/// operator-facing humantime `backoff` is reduced to whole seconds at
152/// build time (mirrors how `jitter_secs` / `timeout_secs` are
153/// pre-lowered) so the agent's fire path does no humantime parsing.
154#[derive(Serialize, Deserialize, schemars::JsonSchema, Debug, Clone, Copy, PartialEq, Eq)]
155pub struct RetrySpec {
156 /// Max additional attempts after the first failure (1..=10,
157 /// enforced by `Schedule::validate`).
158 pub max: u32,
159 /// Seconds slept between attempts.
160 pub backoff_secs: u64,
161}
162
163#[derive(Serialize, Deserialize, schemars::JsonSchema, Debug, Clone, Copy, PartialEq, Eq)]
164#[serde(rename_all = "lowercase")]
165pub enum Shell {
166 Powershell,
167 Cmd,
168}
169
170/// **Token + session combination** the agent uses to spawn a job's
171/// child process. Two orthogonal axes — *whose privileges* and *which
172/// session* — collapse into three meaningful combinations:
173///
174/// | variant | session | privileges | GUI |
175/// |--------------------|------------------------|-------------|-----|
176/// | `System` (default) | Session 0 (services) | LocalSystem | ❌ |
177/// | `User` | active console session | logged-in user (UAC-filtered when admin) | ✅ |
178/// | `SystemGui` | active console session | LocalSystem | ✅ |
179///
180/// `SystemGui` is the "PsExec `-i -s`" pattern: the agent duplicates
181/// its own SYSTEM token and rewrites `TokenSessionId` to the user's
182/// console session, then launches with that hybrid token — useful
183/// when an installer needs admin power *and* needs the user to see
184/// its UI.
185#[derive(
186 Serialize, Deserialize, schemars::JsonSchema, Debug, Clone, Copy, PartialEq, Eq, Default,
187)]
188#[serde(rename_all = "snake_case")]
189pub enum RunAs {
190 /// LocalSystem privileges in Session 0. No GUI. Historical
191 /// default — every pre-v0.21 job ran this way.
192 #[default]
193 System,
194 /// The currently-logged-in console user's identity, in their
195 /// session. Can write HKCU / %APPDATA% / show GUI to the user.
196 /// Privileges are whatever the user has (admin users get the
197 /// UAC-filtered limited token, not the elevated one).
198 User,
199 /// LocalSystem privileges in the user's session — admin power
200 /// with GUI visibility. Niche but real (force-restart dialogs,
201 /// admin installers with progress UI).
202 SystemGui,
203}
204
205#[cfg(test)]
206mod tests {
207 use super::*;
208
209 fn sample_command() -> Command {
210 Command {
211 id: "echo-test".into(),
212 version: "1.0.0".into(),
213 request_id: "req-1".into(),
214 exec_id: Some("dep-1".into()),
215 shell: Shell::Powershell,
216 script: "echo hi".into(),
217 script_object: None,
218 script_object_sha256: None,
219 timeout_secs: 30,
220 jitter_secs: Some(5),
221 run_as: RunAs::System,
222 cwd: None,
223 deadline_at: None,
224 staleness: Staleness::Cached,
225 emit: None,
226 check: None,
227 collect: None,
228 retry: None,
229 finalize: None,
230 }
231 }
232
233 #[test]
234 fn shell_serialises_lowercase() {
235 let json = serde_json::to_string(&Shell::Powershell).unwrap();
236 assert_eq!(json, "\"powershell\"");
237 let json = serde_json::to_string(&Shell::Cmd).unwrap();
238 assert_eq!(json, "\"cmd\"");
239 }
240
241 #[test]
242 fn run_as_serialises_snake_case() {
243 for (mode, expected) in [
244 (RunAs::System, "\"system\""),
245 (RunAs::User, "\"user\""),
246 (RunAs::SystemGui, "\"system_gui\""),
247 ] {
248 let json = serde_json::to_string(&mode).unwrap();
249 assert_eq!(json, expected, "serialise {mode:?}");
250 let back: RunAs = serde_json::from_str(expected).unwrap();
251 assert_eq!(back, mode, "round-trip {expected}");
252 }
253 }
254
255 #[test]
256 fn run_as_defaults_to_system() {
257 assert_eq!(RunAs::default(), RunAs::System);
258 }
259
260 #[test]
261 fn command_round_trips_through_json() {
262 let orig = sample_command();
263 let json = serde_json::to_string(&orig).expect("encode");
264 let decoded: Command = serde_json::from_str(&json).expect("decode");
265 assert_eq!(decoded.id, orig.id);
266 assert_eq!(decoded.version, orig.version);
267 assert_eq!(decoded.request_id, orig.request_id);
268 assert_eq!(decoded.exec_id, orig.exec_id);
269 assert_eq!(decoded.shell, orig.shell);
270 assert_eq!(decoded.script, orig.script);
271 assert_eq!(decoded.timeout_secs, orig.timeout_secs);
272 assert_eq!(decoded.jitter_secs, orig.jitter_secs);
273 assert_eq!(decoded.run_as, orig.run_as);
274 }
275
276 #[test]
277 fn command_round_trips_each_run_as_variant() {
278 for mode in [RunAs::System, RunAs::User, RunAs::SystemGui] {
279 let cmd = Command {
280 run_as: mode,
281 ..sample_command()
282 };
283 let json = serde_json::to_string(&cmd).unwrap();
284 let back: Command = serde_json::from_str(&json).unwrap();
285 assert_eq!(back.run_as, mode);
286 }
287 }
288
289 #[test]
290 fn command_accepts_missing_optional_fields() {
291 let json = r#"{
292 "id": "x",
293 "version": "1.0.0",
294 "request_id": "r",
295 "shell": "cmd",
296 "script": "echo",
297 "timeout_secs": 5
298 }"#;
299 let cmd: Command = serde_json::from_str(json).expect("decode");
300 assert!(cmd.exec_id.is_none());
301 assert!(cmd.jitter_secs.is_none());
302 assert_eq!(cmd.shell, Shell::Cmd);
303 // Pre-v0.21 wire payloads omit run_as → falls back to System.
304 assert_eq!(cmd.run_as, RunAs::System);
305 // Pre-v0.21.1 omit cwd → None (= inherit agent cwd).
306 assert!(cmd.cwd.is_none());
307 // Pre-v0.22 omit deadline_at → None (= no deadline).
308 assert!(cmd.deadline_at.is_none());
309 // Pre-v0.43 wire omits both script_object fields — agent
310 // falls back to the inline `script` body.
311 assert!(cmd.script_object.is_none());
312 assert!(cmd.script_object_sha256.is_none());
313 }
314
315 #[test]
316 fn command_round_trips_script_object_fields() {
317 // yukimemi/kanade#210: backend builds Commands carrying an
318 // OBJECT_SCRIPTS reference + the operator-approved digest;
319 // agent resolves on fetch. Both fields must survive a JSON
320 // round-trip with the same shape.
321 let cmd = Command {
322 script: String::new(),
323 script_object: Some("cleanup-disk-temp/1.0.1".into()),
324 script_object_sha256: Some(
325 "deadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef".into(),
326 ),
327 ..sample_command()
328 };
329 let json = serde_json::to_string(&cmd).expect("encode");
330 let back: Command = serde_json::from_str(&json).expect("decode");
331 assert_eq!(back.script, "");
332 assert_eq!(
333 back.script_object.as_deref(),
334 Some("cleanup-disk-temp/1.0.1")
335 );
336 assert_eq!(
337 back.script_object_sha256.as_deref(),
338 Some("deadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef")
339 );
340 }
341
342 #[test]
343 fn command_decodes_legacy_job_id_field_as_exec_id() {
344 // v0.29 / Issue #19: Commands sitting in STREAM_EXEC published
345 // by a pre-v0.29 backend still carry the field named `job_id`.
346 // The `#[serde(alias = "job_id")]` on `exec_id` keeps them
347 // decodable through the upgrade window so the agent doesn't
348 // start dropping replays on first boot of a new binary.
349 let json = r#"{
350 "id": "x",
351 "version": "1.0.0",
352 "request_id": "r",
353 "job_id": "legacy-exec-uuid",
354 "shell": "powershell",
355 "script": "echo",
356 "timeout_secs": 5
357 }"#;
358 let cmd: Command = serde_json::from_str(json).expect("decode legacy");
359 assert_eq!(cmd.exec_id.as_deref(), Some("legacy-exec-uuid"));
360 }
361
362 #[test]
363 fn command_deadline_at_round_trips() {
364 use chrono::TimeZone;
365 let deadline = Utc.with_ymd_and_hms(2026, 5, 18, 9, 30, 0).unwrap();
366 let cmd = Command {
367 deadline_at: Some(deadline),
368 ..sample_command()
369 };
370 let json = serde_json::to_string(&cmd).unwrap();
371 let back: Command = serde_json::from_str(&json).unwrap();
372 assert_eq!(back.deadline_at, Some(deadline));
373 }
374
375 #[test]
376 fn command_retry_round_trips() {
377 // #418 Phase 4: a stamped retry policy must survive the wire
378 // so the agent can apply it on a live publish or a STREAM_EXEC
379 // replay.
380 let cmd = Command {
381 retry: Some(RetrySpec {
382 max: 3,
383 backoff_secs: 600,
384 }),
385 ..sample_command()
386 };
387 let json = serde_json::to_string(&cmd).unwrap();
388 let back: Command = serde_json::from_str(&json).unwrap();
389 assert_eq!(
390 back.retry,
391 Some(RetrySpec {
392 max: 3,
393 backoff_secs: 600
394 })
395 );
396 }
397
398 #[test]
399 fn command_omits_retry_when_absent() {
400 // skip_serializing_if keeps the field off the wire for the
401 // common (no-retry) case, and pre-Phase-4 payloads that never
402 // had it still decode (serde default → None).
403 let json = serde_json::to_string(&sample_command()).unwrap();
404 assert!(
405 !json.contains("retry"),
406 "retry must not appear when None: {json}"
407 );
408 }
409
410 #[test]
411 fn command_collect_round_trips_and_omits_when_absent() {
412 // #219: `collect` is off the wire when None (skip_serializing_if),
413 // so pre-#219 readers don't trip over it...
414 let json = serde_json::to_string(&sample_command()).unwrap();
415 assert!(
416 !json.contains("collect"),
417 "collect must be absent when None: {json}"
418 );
419 // ...and a forwarded CollectHint survives the round-trip.
420 let cmd = Command {
421 collect: Some(CollectHint {
422 name: "diag".into(),
423 description: Some("logs".into()),
424 max_size: Some("50MB".into()),
425 files_field: "files".into(),
426 }),
427 ..sample_command()
428 };
429 let back: Command = serde_json::from_str(&serde_json::to_string(&cmd).unwrap()).unwrap();
430 let c = back.collect.expect("collect survived round-trip");
431 assert_eq!(c.name, "diag");
432 assert_eq!(c.max_size.as_deref(), Some("50MB"));
433 assert_eq!(c.files_field, "files");
434 }
435
436 #[test]
437 fn command_finalize_round_trips_and_omits_when_absent() {
438 // Off the wire when None (skip_serializing_if), so pre-finalize
439 // readers don't trip over it...
440 let json = serde_json::to_string(&sample_command()).unwrap();
441 assert!(
442 !json.contains("finalize"),
443 "finalize must be absent when None: {json}"
444 );
445 // ...and a forwarded FinalizeCommand survives the round-trip.
446 let cmd = Command {
447 finalize: Some(FinalizeCommand {
448 shell: Shell::Powershell,
449 script: "Remove-Item $env:FILE".into(),
450 timeout_secs: 30,
451 run_as: RunAs::System,
452 cwd: None,
453 }),
454 ..sample_command()
455 };
456 let back: Command = serde_json::from_str(&serde_json::to_string(&cmd).unwrap()).unwrap();
457 let f = back.finalize.expect("finalize survived round-trip");
458 assert_eq!(f.shell, Shell::Powershell);
459 assert_eq!(f.script, "Remove-Item $env:FILE");
460 assert_eq!(f.timeout_secs, 30);
461 assert_eq!(f.run_as, RunAs::System);
462 }
463}