Deterministic VM for LLM program execution
Project description
Governed Agent Execution runtime for LLM workflows.
Deterministic. Replayable. Enforcement-first.
LLM support is optional.
LLMs are signal generators. Execution authority belongs to the runtime.
What nano-vm Is
Most AI frameworks answer: how do you coordinate agents?
Almost nobody answers: who guarantees execution correctness?
nano-vm is the answer to the second question.
It is a deterministic FSM execution kernel for LLM workflows and stateful business processes. The runtime — not the model, not the tool, not the prompt — controls state transitions.
Core invariant:
δ(S, E) → S'
Where S is current execution state, E is a validated event, S' is the next deterministic state.
Why not Temporal? Temporal solves durable execution for distributed systems. nano-vm solves governed execution for LLM workflows — embedded, no infrastructure, Python-native, with a governance layer that understands LLM-specific failure modes (output enum violations, retry storms, evaluator awareness).
Architecture
events / webhooks / tools / LLMs
↓
ExecutionVM ← FSM, step lifecycle, budget guards
↓
deterministic FSM ← ASTEngine (no eval()), sandboxed conditions
↓
replayable trace ← sha256 snapshot per step, Merkle chain
Formally:
nondeterminism ∈ signal generation
determinism ∈ runtime execution
| Layer | Role | Deterministic |
|---|---|---|
| Signal | LLM / webhook / API / user input | ❌ |
| Validator | schema + policy validation | ✅ |
| FSM | transition authority | ✅ |
| Policy | transition selection | ✅ |
| Tool executor | side effects | enforced |
Install
pip install llm-nano-vm
pip install llm-nano-vm[litellm] # for LLM provider support
Using nano-vm Without LLMs
LLMs are not required. nano-vm runs as a pure deterministic workflow engine.
from nano_vm import ExecutionVM, Program
program = Program.from_dict({
"name": "payment_flow",
"steps": [
{"id": "reserve", "type": "tool", "tool": "reserve_funds"},
{"id": "capture", "type": "tool", "tool": "capture_payment"},
{"id": "receipt", "type": "tool", "tool": "send_receipt"},
]
})
vm = ExecutionVM(tools={
"reserve_funds": reserve_funds,
"capture_payment": capture_payment,
"send_receipt": send_receipt,
})
trace = await vm.run(program)
print(trace.status) # SUCCESS
No LLM. The runtime still guarantees: deterministic ordering, replayable execution, trace visibility, transition enforcement, idempotent re-execution across restarts.
Quick Start — LLM Pipeline (guardrail that never skips)
from nano_vm import ExecutionVM, Program
from nano_vm.adapters import LiteLLMAdapter
program = Program.from_dict({
"name": "customer_refund",
"steps": [
{
"id": "analyze",
"type": "llm",
"prompt": "Is this a valid refund request? Reply 'yes' or 'no'.\nRequest: $user_input",
"output_key": "decision",
"allowed_outputs": ["yes", "no"], # v0.8.0 — runtime enum gate
},
{
"id": "guardrail",
"type": "condition",
"condition": "'yes' in '$decision'",
"then": "process_refund",
"otherwise": "reject",
},
{"id": "process_refund", "type": "tool", "tool": "issue_refund", "is_terminal": True},
{"id": "reject", "type": "tool", "tool": "send_rejection", "is_terminal": True},
],
})
vm = ExecutionVM(
llm=LiteLLMAdapter("openai/gpt-4o-mini"),
tools={"issue_refund": ..., "send_rejection": ...},
)
trace = await vm.run(program, context={"user_input": "I was charged twice"})
print(trace.status) # SUCCESS
print(trace.total_cost_usd()) # e.g. 0.000034
The guardrail step cannot be skipped, reordered, or overridden by the model.
Suspend / Resume — Async Business Processes
Return the sentinel "PENDING" from any tool to suspend execution:
async def initiate_payment(**kwargs) -> str:
await register_webhook(kwargs["order_id"])
return "PENDING" # FSM → SUSPENDED, cursor persisted
FSM transition: RUNNING → SUSPENDED → RUNNING → SUCCESS
This enables: payment settlement, courier confirmation, approval systems, human-in-the-loop, webhook orchestration. The process can restart. The cursor survives.
trace = await vm.run(program, context={"order_id": "123"})
assert trace.status.name == "SUSPENDED"
trace = await vm.resume_with_program(
program=program,
trace_id=trace.trace_id,
webhook_event={"type": "payment.confirmed", "order_id": "123"},
)
assert trace.status.name == "SUCCESS"
Note:
"PENDING"is a reserved FSM sentinel. Do not return it as a domain status. Use"REQUIRES_ACTION","AWAITING_3DS", or any other string for domain-specific states.
LLM Output Enforcement — allowed_outputs (v0.8.0)
Validates the model's raw output against an explicit enum before it enters the FSM context. This isn't a prompt hint. It's a runtime gate.
{
"id": "classify",
"type": "llm",
"prompt": "Classify the request. Reply ONLY with: refund / query / other",
"output_key": "category",
"allowed_outputs": ["refund", "query", "other"],
"on_error": "skip", # output → "refund" (first element) on mismatch
}
on_error |
On mismatch |
|---|---|
fail (default) |
VMError → trace.FAILED |
skip |
output replaced with allowed_outputs[0] |
retry |
retry up to max_retries; VMError if exhausted |
Evaluator Blindness by Design
Benchmark research across 10 frontier models found that every model changes behavior when it detects it's being monitored. The best model scored 84% on "acts the same whether watched or not."
The ProjectionLayer gives the LLM only a target=LLM projection of state. Governance metadata — GovernanceEnvelope, canonical_hash, policy_hash, retry counters — never reaches the prompt. The model cannot observe its own audit trail.
Evaluator blindness is structural, not configured.
FSM Transition Model
| Current state | Event | Next state |
|---|---|---|
RUNNING |
tool success | RUNNING |
RUNNING |
tool returns "PENDING" |
SUSPENDED |
RUNNING |
tool error (on_error=fail) |
FAILED |
RUNNING |
tool error (on_error=skip) |
RUNNING (output=None) |
RUNNING |
condition branch taken | RUNNING (jump to then/otherwise) |
RUNNING |
max_steps / max_tokens exceeded |
BUDGET_EXCEEDED |
RUNNING |
max_stalled_steps exceeded |
STALLED |
RUNNING |
no more steps | SUCCESS |
SUSPENDED |
resume_with_program() called |
RUNNING (from cursor) |
| terminal | — | absorbing (no further transitions) |
Terminal states: SUCCESS, FAILED, BUDGET_EXCEEDED, STALLED.
Program DSL
Four step types:
| Type | Purpose |
|---|---|
llm |
call the model; result stored in output_key |
tool |
call a Python function; return "PENDING" to suspend |
condition |
branch on an expression; then / otherwise |
parallel |
run independent sub-steps concurrently via asyncio.gather |
Step fields (v0.8.0):
| Field | Default | Description |
|---|---|---|
on_error |
fail |
fail · skip · retry |
max_retries |
3 |
total attempts; backoff: 1s, 2s, 4s… cap 30s |
max_concurrency |
None |
parallel blocks only |
is_terminal |
False |
return SUCCESS after this step (leaf nodes) |
next_step |
None |
jump to named step instead of returning SUCCESS |
allowed_outputs |
None |
LLM-only — accepted output enum; ValidationError if empty |
timeout_seconds |
None |
LLM-only — asyncio.wait_for timeout in seconds |
on_timeout |
'fail' |
'fail' · 'fallback' (→ allowed_outputs[0] or '') |
Program budget options (v0.4.0+):
| Option | Default | Description |
|---|---|---|
max_steps |
None |
BUDGET_EXCEEDED if exceeded |
max_stalled_steps |
None |
STALLED after N consecutive no-op fingerprints |
max_tokens |
None |
BUDGET_EXCEEDED when total tokens exceed limit |
Variable interpolation
| Syntax | Resolves to |
|---|---|
$key |
value from initial context (typed — int/dict/list preserved) |
$step_id.output |
output of a previous step |
$step_id.output.field |
field within a step's dict output |
Condition expressions — Security
⚠ ASTEngine replaces
eval(). Conditions are parsed into a validated JSON AST and evaluated by a pure, sandboxed interpreter. No Python builtins are accessible.
Supported: ==, !=, >, <, in, not in, and, or, not, contains, dotted-path $var.field.
Not supported: method calls (.lower(), .strip()), arithmetic, parentheses grouping. Using an unsupported form raises ASTEvalError at parse time (v0.7.5+).
# ❌ WRONG — method call raises ASTEvalError
{"condition": "'yes' in '$decision'.lower()"}
# ✅ CORRECT
{"condition": "'yes' in '$decision'"}
MCP Integration
nano-vm pairs with nano-vm-mcp — an MCP gateway
that exposes run_program, get_trace, list_programs, get_program, delete_program
over stdio or SSE transport with bearer auth, SQLite WAL persistence, and GovernanceEnvelope audit trail.
Claude Code / MCP Client
↓
nano-vm-mcp ← decides how execution is allowed to proceed
↓
deterministic FSM ← guarantees correctness
↓
GovernanceEnvelope ← proves it happened
GovernanceEnvelope
Each successful execution step produces a GovernanceEnvelope stored in SQLite WAL:
| Field | Description |
|---|---|
execution_id |
Session / trace identifier |
step_id |
Step index within the execution |
policy_hash |
SHA-256 of the active PolicySnapshot |
canonical_snapshot_hash |
Merkle/delta hash of CanonicalState |
payload |
Projected (sanitized) step output |
CapabilityRef and GDPR Tombstoning
Sensitive values are stored as CapabilityRef tokens (vault://secret/<id>).
On a GDPR erasure event, the ref is tombstoned. All subsequent projections return
[REDACTED_TOMBSTONE], preserving the hash chain. Forensic auditability survives erasure.
Observability
trace.trace_id # UUID4 — stable for OTel propagation
trace.status # TraceStatus.SUCCESS | FAILED | SUSPENDED | BUDGET_EXCEEDED | STALLED
trace.final_output
trace.total_tokens() # O(1) incremental accumulator
trace.total_cost_usd() # requires LiteLLMAdapter
trace.state_snapshots # list[(step_index, sha256_hex)]
for step in trace.steps:
print(step.step_id, step.status, step.duration_ms, step.usage)
Testing — Deterministic by Design
from nano_vm import ExecutionVM, Program, TraceStatus
from nano_vm.adapters import MockLLMAdapter
vm = ExecutionVM(llm=MockLLMAdapter("yes")) # always returns "yes"
# Per-call sequence
vm = ExecutionVM(llm=MockLLMAdapter(["SAFE", "yes"]))
# Per-prompt substring mapping
vm = ExecutionVM(llm=MockLLMAdapter({
"Classify": "SAFE",
"eligible": "yes",
"__default__": "ok",
}))
trace = await vm.run(program, context={"user_input": "refund"})
assert trace.status == TraceStatus.SUCCESS
Same input → same step sequence. No API key required.
State Determinism vs Semantic Determinism
nano-vm guarantees State Determinism — step execution order, no skipping, reproducible
trace structure — regardless of LLM output. It does not guarantee Semantic Determinism
(LLM text may differ across runs even at temperature=0.0). Use MockLLMAdapter for both.
Planner (Optional)
from nano_vm import Planner
planner = Planner(llm=adapter, max_retries=2, temperature=0.0)
program = await planner.generate(
"Fetch latest AI news, summarize, classify by topic",
available_tools=["fetch_rss", "summarize", "classify"],
context_keys=["user_id"],
)
trace = await vm.run(program)
Exactly 1 LLM call → validated Program. Planner output is probabilistic; execution
remains deterministic. Review Planner-generated programs before deploying to production.
Performance
The VM introduces near-zero overhead. The bottleneck is the LLM API or external I/O.
v0.8.2 test suite (435/435 · 0 violations)
| Suite | Result |
|---|---|
| MoMo PoC v4 | 9/9 PASS |
| Stripe PoC v1 | 9/9 PASS |
| FSM invariant suite (v0.6.0) | 13/13 · 1,020,000 ops · 0 violations |
| Integration suite (v0.7.3) | 10/10 · 1,096,500 ops · 0 violations |
| 10k stress (v0.7.0) | 14,327 graphs/sec · 0.70 s/run |
Invariants verified: no step skipping, no out-of-order execution, no duplicate step_id in trace, all terminal states absorbing.
Integration benchmark detail (v0.7.3)
Environment: QEMU/KVM · Intel Xeon E5-2697A v4 · 2 cores · Python 3.12 · Mock adapter.
| ID | Scenario | Mean TPS | p95 avg |
|---|---|---|---|
| BM-INT-01 | Refund pipeline | 2,300/s | 0.66 ms |
| BM-INT-02 | Double-execution guard | 2,400/s | 0.67 ms |
| BM-INT-03 | Budget enforcement | 1,100/s | 331 ms |
| BM-INT-04 | Parallel throughput | 436/s | 542 ms |
| BM-INT-05 | MCP store round-trip | 3,000/s | 0.42 ms |
| BM-INT-06 | GovernanceEnvelope | 1,300/s | 171 ms |
| BM-INT-07 | Crash consistency | 7/s | 233 ms |
| BM-INT-08 | Replay equivalence | 1,300/s | 1.30 ms |
| BM-INT-09 | Adversarial retries | 2,400/s | 0.64 ms |
| BM-INT-10 | Long-horizon | 30/s | 3,606 ms |
Comparison
| LangChain | CrewAI | Temporal | nano-vm | |
|---|---|---|---|---|
| LLM-native | ✅ | ✅ | ❌ | ✅ |
| Deterministic FSM | ❌ | ❌ | ✅ | ✅ |
| Replayable traces | partial | minimal | ✅ | ✅ |
| Suspend/resume | partial | partial | ✅ | ✅ |
| Runtime guardrails | ❌ | ❌ | partial | ✅ |
| LLM output enforcement | ❌ | ❌ | ❌ | ✅ |
| Evaluator blindness | ❌ | ❌ | ❌ | ✅ |
| Lightweight / embedded | ❌ | ❌ | ❌ | ✅ |
| Business workflows | partial | ❌ | ✅ | ✅ |
| AI workflows | ✅ | ✅ | partial | ✅ |
vs Marvin / DSPy: those optimize what the LLM produces (structured outputs, prompt tuning). nano-vm controls when and whether steps run — orthogonal concerns, composable.
vs Temporal / Cadence: Temporal solves durable execution for distributed systems. nano-vm solves governed execution for LLM workflows — embedded, no infrastructure, Python-native.
When to Use
Use nano-vm when:
- workflow structure is known in advance
- correctness and auditability matter (fintech, compliance, enterprise)
- you need a reproducible trace for debugging or logging
- guardrails must be enforced at the system level, not in the prompt
- async orchestration with suspend/resume is required
- you need LLM output validated at the runtime level before it affects state
Do NOT use when:
- workflow must be discovered fully at runtime
- the task is open-ended creative reasoning
- fully autonomous multi-agent coordination is required
Roadmap
Done:
- Deterministic FSM runtime (v0.1)
-
parallelsteps —asyncio.gather(v0.2.0) -
retrypolicy +max_concurrency(v0.3.0) - Budget guards:
max_steps,max_stalled_steps,max_tokens(v0.4.0) -
state_snapshots— sha256 fingerprint per step (v0.4.0) -
Planner— intent → Program in 1 LLM call (v0.5.0) - FSM invariant stress suite — 13/13 · 1,020,000 ops (v0.6.0)
-
suspend / resume—"PENDING"sentinel +CursorRepository(v0.7.0) -
BudgetInterrupt+_emit_interrupt()hook (v0.7.0) -
Trace.trace_id— UUID4, OTel-ready (v0.7.0) -
erase()— GDPR tombstoning with hash-chain preservation (v0.7.0) -
ASTEngine—eval()removed; sandboxed condition evaluator (v0.7.0) - Integration benchmark suite — 10/10 · 1,096,500 ops (v0.7.3)
-
Step.is_terminal,Step.next_step— branch semantics (v0.7.4) - ASTEngine METHOD_CALL guard —
ASTEvalErrorat parse time (v0.7.5) -
py.typedmarker — PEP 561 (v0.7.4) - MCP server —
nano-vm-mcpwith GovernanceEnvelope, CapabilityRef, SSE + stdio -
Step.allowed_outputs— LLM output validation against enum (v0.8.0) -
Step.timeout_seconds+on_timeout— per-step LLM timeout (v0.8.0) -
inspect.iscoroutinefunction— Python 3.14 deprecation fix (v0.8.2)
Upcoming — observability (v0.8.x):
-
TraceAnalyzer— rollback density, tool churn rate, path variance, invariant violation rate -
ProgramValidator— static analysis: unreachable steps, missing targets, cycle detection - OpenTelemetry span per FSM step
- Incremental counters in
Trace:llm_calls,tool_calls,retries_total
Upcoming — execution graph (v0.8.x):
-
depends_on+TopologicalSorter— declarative dependency graph overparallel
Upcoming — gateway (v0.9.x):
-
nano-vm-mcp:GovernedToolExecutor+ circuit breaker -
replan_on_interrupt— Planner-driven continuation on budget interrupts
Contact & Support
Author: @ale007xd on Telegram · @ale007xd on X
USDT (TON): UQCakyytrEGBikOi3eYMpveGHXDB1-fd6lcuQC9VvKqMrI-9
License
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file llm_nano_vm-0.8.3.tar.gz.
File metadata
- Download URL: llm_nano_vm-0.8.3.tar.gz
- Upload date:
- Size: 1.2 MB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
709fc7b57c44c3cb68c918e3f4d481b8b3315288d8ce0621bedc885a3952ee70
|
|
| MD5 |
fb741d903b306f6941398e5bb7c7715c
|
|
| BLAKE2b-256 |
b2c87dd770ddf5fdfa6eb4a0aae98b428a4d94b85a75cd7b7d117ccad01a76bc
|
File details
Details for the file llm_nano_vm-0.8.3-py3-none-any.whl.
File metadata
- Download URL: llm_nano_vm-0.8.3-py3-none-any.whl
- Upload date:
- Size: 42.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6e0159933ec5c8a51a6b756a8227e1fb372304df6e006d69e93b6c7944869a6d
|
|
| MD5 |
f601362427f40ac21cd5fcd43f231b74
|
|
| BLAKE2b-256 |
7015c3381707a9af51103b6f241eb371cc992ff0eb28acce6ec8d532d83e70c0
|