kanade_shared/wire/command.rs
1use chrono::{DateTime, Utc};
2use serde::{Deserialize, Serialize};
3
4use super::Staleness;
5use crate::manifest::{CheckHint, 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 /// #418 Phase 4: lowered from `Schedule.on_failure.retry` by the
101 /// command builders (backend `exec_manifest` + the agent's local
102 /// scheduler). When `Some`, the agent re-runs the script
103 /// in-process on a non-zero exit / timeout, up to `max` extra
104 /// attempts with `backoff_secs` between them, before publishing
105 /// the final outcome. `None` (default) ⇒ no retry, the historical
106 /// behaviour and what ad-hoc `kanade run` / `kanade exec` use.
107 /// Pre-Phase-4 wire omits this; `#[serde(default)]` → `None`.
108 #[serde(default, skip_serializing_if = "Option::is_none")]
109 pub retry: Option<RetrySpec>,
110}
111
112/// Lowered, engine-vocabulary form of [`crate::manifest::Retry`] — a
113/// fixed-backoff retry policy stamped onto a [`Command`]. The
114/// operator-facing humantime `backoff` is reduced to whole seconds at
115/// build time (mirrors how `jitter_secs` / `timeout_secs` are
116/// pre-lowered) so the agent's fire path does no humantime parsing.
117#[derive(Serialize, Deserialize, schemars::JsonSchema, Debug, Clone, Copy, PartialEq, Eq)]
118pub struct RetrySpec {
119 /// Max additional attempts after the first failure (1..=10,
120 /// enforced by `Schedule::validate`).
121 pub max: u32,
122 /// Seconds slept between attempts.
123 pub backoff_secs: u64,
124}
125
126#[derive(Serialize, Deserialize, schemars::JsonSchema, Debug, Clone, Copy, PartialEq, Eq)]
127#[serde(rename_all = "lowercase")]
128pub enum Shell {
129 Powershell,
130 Cmd,
131}
132
133/// **Token + session combination** the agent uses to spawn a job's
134/// child process. Two orthogonal axes — *whose privileges* and *which
135/// session* — collapse into three meaningful combinations:
136///
137/// | variant | session | privileges | GUI |
138/// |--------------------|------------------------|-------------|-----|
139/// | `System` (default) | Session 0 (services) | LocalSystem | ❌ |
140/// | `User` | active console session | logged-in user (UAC-filtered when admin) | ✅ |
141/// | `SystemGui` | active console session | LocalSystem | ✅ |
142///
143/// `SystemGui` is the "PsExec `-i -s`" pattern: the agent duplicates
144/// its own SYSTEM token and rewrites `TokenSessionId` to the user's
145/// console session, then launches with that hybrid token — useful
146/// when an installer needs admin power *and* needs the user to see
147/// its UI.
148#[derive(
149 Serialize, Deserialize, schemars::JsonSchema, Debug, Clone, Copy, PartialEq, Eq, Default,
150)]
151#[serde(rename_all = "snake_case")]
152pub enum RunAs {
153 /// LocalSystem privileges in Session 0. No GUI. Historical
154 /// default — every pre-v0.21 job ran this way.
155 #[default]
156 System,
157 /// The currently-logged-in console user's identity, in their
158 /// session. Can write HKCU / %APPDATA% / show GUI to the user.
159 /// Privileges are whatever the user has (admin users get the
160 /// UAC-filtered limited token, not the elevated one).
161 User,
162 /// LocalSystem privileges in the user's session — admin power
163 /// with GUI visibility. Niche but real (force-restart dialogs,
164 /// admin installers with progress UI).
165 SystemGui,
166}
167
168#[cfg(test)]
169mod tests {
170 use super::*;
171
172 fn sample_command() -> Command {
173 Command {
174 id: "echo-test".into(),
175 version: "1.0.0".into(),
176 request_id: "req-1".into(),
177 exec_id: Some("dep-1".into()),
178 shell: Shell::Powershell,
179 script: "echo hi".into(),
180 script_object: None,
181 script_object_sha256: None,
182 timeout_secs: 30,
183 jitter_secs: Some(5),
184 run_as: RunAs::System,
185 cwd: None,
186 deadline_at: None,
187 staleness: Staleness::Cached,
188 emit: None,
189 check: None,
190 retry: None,
191 }
192 }
193
194 #[test]
195 fn shell_serialises_lowercase() {
196 let json = serde_json::to_string(&Shell::Powershell).unwrap();
197 assert_eq!(json, "\"powershell\"");
198 let json = serde_json::to_string(&Shell::Cmd).unwrap();
199 assert_eq!(json, "\"cmd\"");
200 }
201
202 #[test]
203 fn run_as_serialises_snake_case() {
204 for (mode, expected) in [
205 (RunAs::System, "\"system\""),
206 (RunAs::User, "\"user\""),
207 (RunAs::SystemGui, "\"system_gui\""),
208 ] {
209 let json = serde_json::to_string(&mode).unwrap();
210 assert_eq!(json, expected, "serialise {mode:?}");
211 let back: RunAs = serde_json::from_str(expected).unwrap();
212 assert_eq!(back, mode, "round-trip {expected}");
213 }
214 }
215
216 #[test]
217 fn run_as_defaults_to_system() {
218 assert_eq!(RunAs::default(), RunAs::System);
219 }
220
221 #[test]
222 fn command_round_trips_through_json() {
223 let orig = sample_command();
224 let json = serde_json::to_string(&orig).expect("encode");
225 let decoded: Command = serde_json::from_str(&json).expect("decode");
226 assert_eq!(decoded.id, orig.id);
227 assert_eq!(decoded.version, orig.version);
228 assert_eq!(decoded.request_id, orig.request_id);
229 assert_eq!(decoded.exec_id, orig.exec_id);
230 assert_eq!(decoded.shell, orig.shell);
231 assert_eq!(decoded.script, orig.script);
232 assert_eq!(decoded.timeout_secs, orig.timeout_secs);
233 assert_eq!(decoded.jitter_secs, orig.jitter_secs);
234 assert_eq!(decoded.run_as, orig.run_as);
235 }
236
237 #[test]
238 fn command_round_trips_each_run_as_variant() {
239 for mode in [RunAs::System, RunAs::User, RunAs::SystemGui] {
240 let cmd = Command {
241 run_as: mode,
242 ..sample_command()
243 };
244 let json = serde_json::to_string(&cmd).unwrap();
245 let back: Command = serde_json::from_str(&json).unwrap();
246 assert_eq!(back.run_as, mode);
247 }
248 }
249
250 #[test]
251 fn command_accepts_missing_optional_fields() {
252 let json = r#"{
253 "id": "x",
254 "version": "1.0.0",
255 "request_id": "r",
256 "shell": "cmd",
257 "script": "echo",
258 "timeout_secs": 5
259 }"#;
260 let cmd: Command = serde_json::from_str(json).expect("decode");
261 assert!(cmd.exec_id.is_none());
262 assert!(cmd.jitter_secs.is_none());
263 assert_eq!(cmd.shell, Shell::Cmd);
264 // Pre-v0.21 wire payloads omit run_as → falls back to System.
265 assert_eq!(cmd.run_as, RunAs::System);
266 // Pre-v0.21.1 omit cwd → None (= inherit agent cwd).
267 assert!(cmd.cwd.is_none());
268 // Pre-v0.22 omit deadline_at → None (= no deadline).
269 assert!(cmd.deadline_at.is_none());
270 // Pre-v0.43 wire omits both script_object fields — agent
271 // falls back to the inline `script` body.
272 assert!(cmd.script_object.is_none());
273 assert!(cmd.script_object_sha256.is_none());
274 }
275
276 #[test]
277 fn command_round_trips_script_object_fields() {
278 // yukimemi/kanade#210: backend builds Commands carrying an
279 // OBJECT_SCRIPTS reference + the operator-approved digest;
280 // agent resolves on fetch. Both fields must survive a JSON
281 // round-trip with the same shape.
282 let cmd = Command {
283 script: String::new(),
284 script_object: Some("cleanup-disk-temp/1.0.1".into()),
285 script_object_sha256: Some(
286 "deadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef".into(),
287 ),
288 ..sample_command()
289 };
290 let json = serde_json::to_string(&cmd).expect("encode");
291 let back: Command = serde_json::from_str(&json).expect("decode");
292 assert_eq!(back.script, "");
293 assert_eq!(
294 back.script_object.as_deref(),
295 Some("cleanup-disk-temp/1.0.1")
296 );
297 assert_eq!(
298 back.script_object_sha256.as_deref(),
299 Some("deadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef")
300 );
301 }
302
303 #[test]
304 fn command_decodes_legacy_job_id_field_as_exec_id() {
305 // v0.29 / Issue #19: Commands sitting in STREAM_EXEC published
306 // by a pre-v0.29 backend still carry the field named `job_id`.
307 // The `#[serde(alias = "job_id")]` on `exec_id` keeps them
308 // decodable through the upgrade window so the agent doesn't
309 // start dropping replays on first boot of a new binary.
310 let json = r#"{
311 "id": "x",
312 "version": "1.0.0",
313 "request_id": "r",
314 "job_id": "legacy-exec-uuid",
315 "shell": "powershell",
316 "script": "echo",
317 "timeout_secs": 5
318 }"#;
319 let cmd: Command = serde_json::from_str(json).expect("decode legacy");
320 assert_eq!(cmd.exec_id.as_deref(), Some("legacy-exec-uuid"));
321 }
322
323 #[test]
324 fn command_deadline_at_round_trips() {
325 use chrono::TimeZone;
326 let deadline = Utc.with_ymd_and_hms(2026, 5, 18, 9, 30, 0).unwrap();
327 let cmd = Command {
328 deadline_at: Some(deadline),
329 ..sample_command()
330 };
331 let json = serde_json::to_string(&cmd).unwrap();
332 let back: Command = serde_json::from_str(&json).unwrap();
333 assert_eq!(back.deadline_at, Some(deadline));
334 }
335
336 #[test]
337 fn command_retry_round_trips() {
338 // #418 Phase 4: a stamped retry policy must survive the wire
339 // so the agent can apply it on a live publish or a STREAM_EXEC
340 // replay.
341 let cmd = Command {
342 retry: Some(RetrySpec {
343 max: 3,
344 backoff_secs: 600,
345 }),
346 ..sample_command()
347 };
348 let json = serde_json::to_string(&cmd).unwrap();
349 let back: Command = serde_json::from_str(&json).unwrap();
350 assert_eq!(
351 back.retry,
352 Some(RetrySpec {
353 max: 3,
354 backoff_secs: 600
355 })
356 );
357 }
358
359 #[test]
360 fn command_omits_retry_when_absent() {
361 // skip_serializing_if keeps the field off the wire for the
362 // common (no-retry) case, and pre-Phase-4 payloads that never
363 // had it still decode (serde default → None).
364 let json = serde_json::to_string(&sample_command()).unwrap();
365 assert!(
366 !json.contains("retry"),
367 "retry must not appear when None: {json}"
368 );
369 }
370}