Skip to main content

kaish_types/
kernel.rs

1//! Kernel-level execution options.
2//!
3//! `ExecuteOptions` is the input to a single kernel `execute` call. It collects
4//! the per-call knobs (variables, timeout, cancellation) so embedders don't need
5//! to manage half a dozen execute-method overloads.
6
7use std::collections::{BTreeMap, HashMap};
8use std::path::PathBuf;
9use std::time::Duration;
10
11use tokio_util::sync::CancellationToken;
12
13use crate::value::Value;
14
15/// Per-call options for `Kernel::execute_with_options`.
16///
17/// Construct with `ExecuteOptions::new()` and the chainable `with_*` builders,
18/// or via `Default`.
19///
20/// # Cancellation vs. timeout — embedder note
21///
22/// If a `cancel_token` is supplied, it is **raced** against the kernel's
23/// internal token. The kernel does NOT cancel the embedder's token on its
24/// own timeouts — it cancels its internal token and returns exit code 124.
25/// So `your_token.is_cancelled()` after the call returns reflects only
26/// whether *you* (or someone sharing your token) cancelled, not whether the
27/// kernel timed out. Distinguish via the returned `ExecResult.code`:
28/// `124` = kernel timeout, `130` = cancellation (Ctrl-C / `Kernel::cancel`).
29#[derive(Default, Clone)]
30pub struct ExecuteOptions {
31    /// Variables exported into this call's environment (per-call overlay).
32    pub vars: HashMap<String, Value>,
33    /// Per-call timeout. Overrides `KernelConfig::request_timeout`.
34    ///
35    /// `None` means no timeout (or whatever the kernel-config default is).
36    /// `Some(Duration::ZERO)` returns exit 124 immediately without spawning
37    /// anything — useful for tests and dry-run paths.
38    /// Any other `Some(d)` lets the kernel run for at most `d` before cancelling
39    /// (which kills external children with the configured grace) and returning 124.
40    pub timeout: Option<Duration>,
41    /// Optional externally-owned cancellation token, *raced* against the kernel's
42    /// internal token. Either firing cancels the request and kills any running
43    /// external children. The kernel does not store this token in its own state —
44    /// it's a per-call read-only input, so embedders are free to drop or reuse
45    /// the original token after the call returns. CancellationToken is internally
46    /// `Arc`-shared, so `clone()` it into the builder if you want to keep your
47    /// original handle.
48    pub cancel_token: Option<CancellationToken>,
49    /// Per-call working directory override.
50    ///
51    /// When `Some(path)`, the kernel runs this call as if `cd path` happened
52    /// first, then restores the prior cwd on return. Useful for embedders that
53    /// run scripts in workspace contexts (notebook cells, per-tool dirs)
54    /// without polluting the long-lived kernel's cwd.
55    pub cwd: Option<PathBuf>,
56    /// W3C `traceparent` of the embedder's active span, e.g.
57    /// `"00-4bf92f3577b34da6a3ce929d0e0e4736-00f067aa0ba902b7-01"`. When set,
58    /// the kernel's execution span parents onto it, so kaish's spans appear as
59    /// children of the embedder's trace rather than as orphan roots.
60    pub traceparent: Option<String>,
61    /// W3C `tracestate` (vendor-specific list) that rides alongside
62    /// `traceparent`. Per the W3C spec, `tracestate` is meaningless without a
63    /// `traceparent`, so the kernel ignores it unless `traceparent` is also set.
64    pub tracestate: Option<String>,
65    /// W3C baggage — cross-cutting identifiers (owner, connection, tenant, …)
66    /// the embedder wants stamped onto the trace. Propagated to every child
67    /// span. Independent of `traceparent`: baggage with no trace context starts
68    /// a fresh root that still carries the identifiers.
69    pub baggage: BTreeMap<String, String>,
70}
71
72impl ExecuteOptions {
73    pub fn new() -> Self {
74        Self::default()
75    }
76
77    /// Replace the entire vars overlay with the given map.
78    pub fn with_vars(mut self, vars: HashMap<String, Value>) -> Self {
79        self.vars = vars;
80        self
81    }
82
83    /// Add a single variable to the overlay (extending; last write wins).
84    pub fn with_var(mut self, name: impl Into<String>, value: Value) -> Self {
85        self.vars.insert(name.into(), value);
86        self
87    }
88
89    pub fn with_timeout(mut self, timeout: Duration) -> Self {
90        self.timeout = Some(timeout);
91        self
92    }
93
94    pub fn with_cancel_token(mut self, token: CancellationToken) -> Self {
95        self.cancel_token = Some(token);
96        self
97    }
98
99    /// Run this call as if `cd path` had happened first; the prior cwd is
100    /// restored on return.
101    pub fn with_cwd(mut self, cwd: PathBuf) -> Self {
102        self.cwd = Some(cwd);
103        self
104    }
105
106    /// Set the W3C `traceparent` the kernel's execution span should parent onto.
107    pub fn with_traceparent(mut self, traceparent: impl Into<String>) -> Self {
108        self.traceparent = Some(traceparent.into());
109        self
110    }
111
112    /// Set the W3C `tracestate` that rides alongside `traceparent`. Ignored by
113    /// the kernel unless a `traceparent` is also present.
114    pub fn with_tracestate(mut self, tracestate: impl Into<String>) -> Self {
115        self.tracestate = Some(tracestate.into());
116        self
117    }
118
119    /// Replace the entire baggage map with the given identifiers.
120    pub fn with_baggage(mut self, baggage: BTreeMap<String, String>) -> Self {
121        self.baggage = baggage;
122        self
123    }
124
125    /// Add a single baggage identifier (extending; last write wins).
126    pub fn with_baggage_entry(mut self, key: impl Into<String>, value: impl Into<String>) -> Self {
127        self.baggage.insert(key.into(), value.into());
128        self
129    }
130}