Skip to main content

harn_vm/
stdlib.rs

1//! Standard library builtins for the Harn VM.
2
3mod agent_sessions;
4pub mod agent_state;
5mod agents;
6mod agents_daemon;
7pub(crate) mod assemble;
8pub mod asset_paths;
9mod bytes;
10mod calendar;
11pub(crate) mod clock;
12mod collections;
13mod command_policy;
14mod compression;
15mod concurrency;
16mod connectors;
17mod cookies;
18mod crypto;
19mod csv;
20mod datetime;
21mod event_log;
22pub(crate) mod files;
23mod flow;
24mod fs;
25mod git;
26pub(crate) mod harn_entry;
27pub(crate) mod hitl;
28mod hitl_read;
29pub mod host;
30pub(crate) mod io;
31mod iter;
32pub(crate) mod json;
33mod json_query;
34mod junit;
35mod logging;
36pub mod long_running;
37mod math;
38mod memory;
39mod monitors;
40mod multipart;
41mod options;
42mod path;
43mod postgres;
44pub mod process;
45mod project;
46mod project_catalog;
47mod project_enrich;
48mod regex;
49pub(crate) mod registration;
50mod review;
51mod runtime_scope;
52pub(crate) mod sandbox;
53pub mod secret_scan;
54mod sets;
55pub(crate) mod shapes;
56mod skills;
57mod strings;
58pub(crate) mod supervisor;
59pub mod template;
60mod testbench;
61mod testing;
62pub(crate) mod tools;
63pub mod tracing;
64mod transcript_compact;
65mod triggers_stdlib;
66mod tui;
67mod types;
68mod url_parse;
69mod vision;
70pub(crate) mod waitpoint;
71mod waitpoints;
72mod web;
73pub mod workflow_messages;
74
75use crate::http::register_http_builtins;
76use crate::llm::register_llm_builtins;
77use crate::mcp::register_mcp_builtins;
78use crate::mcp_server::register_mcp_server_builtins;
79use crate::vm::Vm;
80
81pub(crate) use crate::schema::{json_to_vm_value, schema_result_value};
82pub(crate) fn set_thread_source_dir(dir: &std::path::Path) {
83    process::set_thread_source_dir(dir);
84}
85
86/// Register core builtins: pure/deterministic, no I/O.
87pub fn register_core_stdlib(vm: &mut Vm) {
88    crate::runtime_context::register_runtime_context_builtins(vm);
89    types::register_type_builtins(vm);
90    math::register_math_builtins(vm);
91    strings::register_string_builtins(vm);
92    json::register_json_builtins(vm);
93    datetime::register_datetime_builtins(vm);
94    calendar::register_calendar_builtins(vm);
95    regex::register_regex_builtins(vm);
96    bytes::register_bytes_builtins(vm);
97    compression::register_compression_builtins(vm);
98    command_policy::register_command_policy_builtins(vm);
99    runtime_scope::register_runtime_scope_builtins(vm);
100    crypto::register_crypto_builtins(vm);
101    csv::register_csv_builtins(vm);
102    junit::register_junit_builtins(vm);
103    multipart::register_multipart_builtins(vm);
104    url_parse::register_url_builtins(vm);
105    web::register_web_builtins(vm);
106    cookies::register_cookie_builtins(vm);
107    path::register_path_helper_builtins(vm);
108    sets::register_set_builtins(vm);
109    collections::register_collection_builtins(vm);
110    iter::register_iter_builtins(vm);
111    event_log::register_event_log_builtins(vm);
112    shapes::register_shape_builtins(vm);
113    testing::register_testing_builtins(vm);
114    flow::register_flow_builtins(vm);
115}
116
117/// Register I/O builtins (requires OS access).
118pub fn register_io_stdlib(vm: &mut Vm) {
119    io::register_io_builtins(vm);
120    host::register_host_builtins(vm);
121    fs::register_fs_builtins(vm);
122    files::register_file_builtins(vm);
123    git::register_git_builtins(vm);
124    vision::register_vision_builtins(vm);
125    agent_state::register_agent_state_builtins(vm);
126    memory::register_memory_builtins(vm);
127    process::register_process_builtins(vm);
128    process::register_path_builtins(vm);
129    // Clock builtins overlay process::timestamp/elapsed so they honor
130    // mock_time / advance_time. Register AFTER process to take precedence.
131    clock::register_clock_builtins(vm);
132    testbench::register_testbench_builtins(vm);
133    project::register_project_builtins(vm);
134    tracing::register_tracing_builtins(vm);
135    tui::register_tui_builtins(vm);
136}
137
138/// Register agent builtins (requires network access and async runtime).
139pub fn register_agent_stdlib(vm: &mut Vm) {
140    concurrency::register_concurrency_builtins(vm);
141    connectors::register_connector_builtins(vm);
142    review::register_review_builtins(vm);
143    secret_scan::register_secret_scan_builtins(vm);
144    tools::register_tool_builtins(vm);
145    skills::register_skill_builtins(vm);
146    agents_daemon::register_daemon_builtins(vm);
147    triggers_stdlib::register_trigger_builtins(vm);
148    postgres::register_postgres_builtins(vm);
149    waitpoints::register_waitpoint_builtins(vm);
150    monitors::register_monitor_builtins(vm);
151    hitl::register_hitl_builtins(vm);
152    hitl_read::register_hitl_read_builtins(vm);
153    waitpoint::register_waitpoint_builtins(vm);
154    supervisor::register_supervisor_builtins(vm);
155    agents::register_agent_builtins(vm);
156    agent_sessions::register_agent_session_builtins(vm);
157    workflow_messages::register_workflow_message_builtins(vm);
158    transcript_compact::register_transcript_compaction_builtins(vm);
159    assemble::register_assemble_context_builtin(vm);
160    crate::egress::register_egress_builtins(vm);
161    register_http_builtins(vm);
162    register_llm_builtins(vm);
163    register_mcp_builtins(vm);
164    register_mcp_server_builtins(vm);
165    crate::step_runtime::register_step_builtins(vm);
166}
167
168/// Register all standard builtins on a VM (core + io + agent).
169pub fn register_vm_stdlib(vm: &mut Vm) {
170    register_core_stdlib(vm);
171    register_io_stdlib(vm);
172    register_agent_stdlib(vm);
173}
174
175pub(crate) fn rebind_execution_state_builtins(vm: &mut Vm) {
176    concurrency::register_concurrency_builtins(vm);
177}
178
179fn stdlib_probe_vm() -> Vm {
180    let mut vm = Vm::new();
181    register_vm_stdlib(&mut vm);
182    // Name-only/metadata introspection never accesses this path, but passing
183    // a real per-platform temp dir keeps registration logic honest if a
184    // callee someday validates its parent.
185    let tmp = std::env::temp_dir();
186    crate::store::register_store_builtins(&mut vm, &tmp);
187    crate::checkpoint::register_checkpoint_builtins(&mut vm, &tmp, "default");
188    crate::metadata::register_metadata_builtins(&mut vm, &tmp);
189    crate::metadata::register_scan_builtins(&mut vm);
190    vm
191}
192
193/// Return the canonical list of all stdlib builtin names. Used by
194/// harn-lint and harn-lsp to avoid hardcoded duplicate lists.
195pub fn stdlib_builtin_names() -> Vec<String> {
196    let vm = stdlib_probe_vm();
197    let mut names = vm.builtin_names();
198    // Special opcodes/keywords, not registered builtins, but linter
199    // should recognize them as valid function calls.
200    for extra in [
201        "spawn",
202        "await",
203        "cancel",
204        "cancel_graceful",
205        "__signal_interrupted",
206        "__signal_off_interrupt",
207        "__signal_on_interrupt",
208        "__signal_raise",
209        "is_cancelled",
210    ] {
211        names.push(extra.to_string());
212    }
213    names
214}
215
216/// Return discoverable metadata for registered stdlib builtins.
217pub fn stdlib_builtin_metadata() -> Vec<crate::vm::VmBuiltinMetadata> {
218    stdlib_probe_vm().builtin_metadata()
219}
220
221/// Reset thread-local stdlib state. Call between test runs.
222///
223/// Note: `long_running::reset_state()` is intentionally NOT called here
224/// because that store is process-global, not thread-local. Wiping it
225/// from a per-test reset hook lets one test cancel another test's
226/// in-flight worker thread (and lose its `push_pending_feedback_global`
227/// notification), which surfaces as `walk_dir_long_running` /
228/// `glob_long_running` timing out under parallel test load. The two
229/// call sites that genuinely need a clean handle store —
230/// `stdlib::fs::tests::{walk_dir_long_running,glob_long_running}` — call
231/// `long_running::reset_state()` explicitly while holding
232/// `LONG_RUNNING_TEST_LOCK`.
233pub fn reset_stdlib_state() {
234    logging::reset_logging_state();
235    process::reset_process_state();
236    clock::reset_clock_state();
237    io::reset_io_state();
238    sandbox::reset_sandbox_state();
239    fs::reset_fs_state();
240    json::reset_json_state();
241    host::reset_host_state();
242    crate::egress::reset_egress_policy_for_host();
243    hitl::reset_hitl_state();
244    crate::http::reset_http_state();
245    monitors::reset_monitor_state();
246    waitpoints::reset_waitpoint_state();
247    waitpoint::reset_waitpoint_state();
248    postgres::reset_postgres_state();
249    supervisor::reset_supervisor_state();
250    agents::records::reset_eval_metrics();
251    agents::records::reset_friction_events();
252    tools::clear_current_tool_registry();
253    tools::clear_tool_synthesis_cache();
254    vision::reset_vision_state();
255    crate::skills::clear_current_skill_registry();
256    template::reset_prompt_registry();
257    crate::triggers::clear_webhook_intake_state();
258    crate::llm::cache::reset_in_process_cache_state();
259}