Skip to main content

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}