harn_vm/orchestration/
mod.rs1use std::path::PathBuf;
2use std::{cell::RefCell, thread_local};
3
4use serde::{Deserialize, Serialize};
5
6use crate::llm::vm_value_to_json;
7use crate::value::{VmError, VmValue};
8
9pub(crate) fn now_rfc3339() -> String {
10 use std::time::{SystemTime, UNIX_EPOCH};
11 let ts = SystemTime::now()
12 .duration_since(UNIX_EPOCH)
13 .unwrap_or_default()
14 .as_secs();
15 format!("{ts}")
16}
17
18pub(crate) fn new_id(prefix: &str) -> String {
19 format!("{prefix}_{}", uuid::Uuid::now_v7())
20}
21
22pub(crate) fn default_run_dir() -> PathBuf {
23 let base = std::env::current_dir().unwrap_or_else(|_| PathBuf::from("."));
24 crate::runtime_paths::run_root(&base)
25}
26
27mod hooks;
28pub use hooks::*;
29
30mod pipeline_lifecycle;
31pub use pipeline_lifecycle::*;
32
33mod settlement_agent;
34pub use settlement_agent::*;
35
36mod lifecycle_receipts;
37pub use lifecycle_receipts::*;
38
39mod command_policy;
40pub use command_policy::*;
41
42mod compaction;
43pub use compaction::*;
44
45mod compact_lifecycle;
46pub use compact_lifecycle::*;
47
48mod compaction_policy_registry;
49pub use compaction_policy_registry::*;
50
51pub mod agent_inbox;
52
53mod artifacts;
54pub use artifacts::*;
55
56mod assemble;
57pub use assemble::*;
58
59mod handoffs;
60pub use handoffs::*;
61
62mod friction;
63pub use friction::*;
64
65mod crystallize;
66pub use crystallize::*;
67
68mod release_fixture;
69pub use release_fixture::*;
70
71mod replay_oracle;
72pub use replay_oracle::*;
73
74mod replay_bench;
75pub use replay_bench::*;
76
77mod policy;
78pub use policy::*;
79
80mod stage_options;
81pub use stage_options::*;
82
83mod workflow;
84pub use workflow::*;
85
86mod workflow_bundle;
87pub use workflow_bundle::*;
88
89mod workflow_patch;
90pub use workflow_patch::*;
91
92mod safe_function_tools;
93pub use safe_function_tools::*;
94
95mod nested_invocation;
96pub use nested_invocation::*;
97
98#[cfg(test)]
99mod workflow_test_fixtures;
100
101mod records;
102pub use records::*;
103
104mod context_eval;
105pub use context_eval::*;
106
107mod skill_gate;
108pub use skill_gate::*;
109
110mod merge_captain_audit;
111pub use merge_captain_audit::*;
112
113mod merge_captain_driver;
114pub use merge_captain_driver::*;
115
116mod merge_captain_ladder;
117pub use merge_captain_ladder::*;
118
119mod merge_captain_iteration;
120pub use merge_captain_iteration::*;
121
122pub mod playground;
123
124thread_local! {
125 static CURRENT_MUTATION_SESSION: RefCell<Option<MutationSessionRecord>> = const { RefCell::new(None) };
126 static CURRENT_WORKFLOW_SKILL_CONTEXT: RefCell<Option<WorkflowSkillContext>> = const { RefCell::new(None) };
132}
133
134#[derive(Clone, Default)]
139pub struct WorkflowSkillContext {
140 pub registry: Option<VmValue>,
141 pub match_config: Option<VmValue>,
142}
143
144pub fn install_workflow_skill_context(context: Option<WorkflowSkillContext>) {
145 CURRENT_WORKFLOW_SKILL_CONTEXT.with(|slot| {
146 *slot.borrow_mut() = context;
147 });
148}
149
150pub fn current_workflow_skill_context() -> Option<WorkflowSkillContext> {
151 CURRENT_WORKFLOW_SKILL_CONTEXT.with(|slot| slot.borrow().clone())
152}
153
154pub struct WorkflowSkillContextGuard;
158
159impl Drop for WorkflowSkillContextGuard {
160 fn drop(&mut self) {
161 install_workflow_skill_context(None);
162 }
163}
164
165#[derive(Clone, Debug, Default, Serialize, Deserialize, PartialEq, Eq)]
166#[serde(default)]
167pub struct MutationSessionRecord {
168 pub session_id: String,
169 pub parent_session_id: Option<String>,
170 pub run_id: Option<String>,
171 pub worker_id: Option<String>,
172 pub execution_kind: Option<String>,
173 pub mutation_scope: String,
174 pub approval_policy: Option<ToolApprovalPolicy>,
178}
179
180impl MutationSessionRecord {
181 pub fn normalize(mut self) -> Self {
182 if self.session_id.is_empty() {
183 self.session_id = new_id("session");
184 }
185 if self.mutation_scope.is_empty() {
186 self.mutation_scope = "read_only".to_string();
187 }
188 self
189 }
190}
191
192pub fn install_current_mutation_session(session: Option<MutationSessionRecord>) {
193 CURRENT_MUTATION_SESSION.with(|slot| {
194 *slot.borrow_mut() = session.map(MutationSessionRecord::normalize);
195 });
196}
197
198pub fn current_mutation_session() -> Option<MutationSessionRecord> {
199 CURRENT_MUTATION_SESSION.with(|slot| slot.borrow().clone())
200}
201pub(crate) fn parse_json_payload<T: for<'de> Deserialize<'de>>(
202 json: serde_json::Value,
203 label: &str,
204) -> Result<T, VmError> {
205 let payload = json.to_string();
206 let mut deserializer = serde_json::Deserializer::from_str(&payload);
207 let mut tracker = serde_path_to_error::Track::new();
208 let path_deserializer = serde_path_to_error::Deserializer::new(&mut deserializer, &mut tracker);
209 T::deserialize(path_deserializer).map_err(|error| {
210 let snippet = if payload.len() > 600 {
211 format!("{}...", &payload[..600])
212 } else {
213 payload.clone()
214 };
215 VmError::Runtime(format!(
216 "{label} parse error at {}: {} | payload={}",
217 tracker.path(),
218 error,
219 snippet
220 ))
221 })
222}
223
224pub(crate) fn parse_json_value<T: for<'de> Deserialize<'de>>(
225 value: &VmValue,
226) -> Result<T, VmError> {
227 parse_json_payload(vm_value_to_json(value), "orchestration")
228}
229
230#[cfg(test)]
231mod tests;