A Python-based deterministic execution engine for DAGs. Treats every operation as a pure function with aggressive caching, deduplication, and bit-for-bit reproducibility.
Project description
Invariant
A Python-based deterministic execution engine for directed acyclic graphs (DAGs). Invariant treats every operation as a pure function, providing aggressive caching, deduplication, and bit-for-bit reproducibility.
Invariant was motivated by the need for deterministic graphics pipelines: icons, badges, dynamic UI components, and data visualizations where aggressive caching and bit-for-bit reproducibility matter. Critically, layouts can be described without a final size—size is injected at execution time and everything else is derived via the expression language. The engine itself is domain-agnostic; domain implementations like Invariant GFX provide graphics ops and plug in via the op registry.
Features
- Aggressive Caching: Artifacts are reused across runs if inputs match
- Deduplication: Identical operations execute only once
- Reproducibility: Bit-for-bit identical outputs across runs
- Ephemeral nodes: Set
cache=Falseto skip caching for frequently-changing outputs (e.g., time-dependent values) - Immutability: Artifacts are frozen once created
- Determinism: Operations rely only on explicit inputs
- Serializable graphs: Versioned JSON wire format for storage, transmission, and interoperability
Installation
# From PyPI
pip install invariant-core
# From source
git clone https://github.com/kws/invariant
cd invariant
uv sync
Quick Start
from invariant import Executor, Node, OpRegistry, cel, ref
from invariant.ops import stdlib
from invariant.store.memory import MemoryStore
# Create registry and register operations
registry = OpRegistry()
registry.register_package("stdlib", stdlib)
# Pipeline: compute (3 * 4) + (5 * 6), then scale the total
#
# ab ──┐
# ├── total ── scaled
# cd ──┘
#
# Literal values flow directly into params — no wrapper nodes needed.
# ref() passes a computed artifact to a downstream op.
# cel() evaluates a CEL (Common Expression Language) expression against
# upstream artifacts, useful for extracting or transforming values
# without a dedicated op.
graph = {
"ab": Node(
op_name="stdlib:multiply",
params={"a": 3, "b": 4}, # literal inputs
deps=[]
),
"cd": Node(
op_name="stdlib:multiply",
params={"a": 5, "b": 6}, # literal inputs
deps=[]
),
"total": Node(
op_name="stdlib:add",
params={"a": ref("ab"), "b": ref("cd")}, # ref() passes artifacts directly
deps=["ab", "cd"]
),
"scaled": Node(
op_name="stdlib:multiply",
params={"a": ref("total"), "b": cel("ab + cd")}, # cel() computes from upstreams
deps=["ab", "cd", "total"]
),
}
# Execute the graph
store = MemoryStore()
executor = Executor(registry=registry, store=store)
results = executor.execute(graph)
print(results["ab"]) # 12
print(results["cd"]) # 30
print(results["total"]) # 42
print(results["scaled"]) # 42 * (12 + 30) = 1764
Architecture
Invariant separates graph definition from execution in two phases:
- Phase 1: Context Resolution - Builds input manifests for each node
- Phase 2: Action Execution - Executes operations or retrieves from cache
Documentation
| Document | Description |
|---|---|
| docs/architecture.md | System overview, design philosophy, and reference test pipeline |
| docs/expressions.md | Normative reference for ref(), cel(), ${...} parameter markers and the CEL expression language |
| docs/executor.md | Normative reference for the two-phase execution model, caching, and artifact storage |
| docs/serialization.md | Normative reference for graph JSON wire format (Node, SubGraphNode, ref, cel) |
| examples/README.md | Runnable examples with walkthroughs, DAG diagrams, and run instructions |
| AGENTS.md | Quick-start guide for AI agents working with this codebase |
Development
# Run tests
uv run pytest
# Run tests with coverage
uv run pytest --cov=src --cov-report=html
# Run linting
uv run ruff check src/ tests/
# Format code
uv run ruff format src/ tests/
License
MIT License - see LICENSE for details.
Project details
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 invariant_core-0.4.0.tar.gz.
File metadata
- Download URL: invariant_core-0.4.0.tar.gz
- Upload date:
- Size: 240.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.10.2 {"installer":{"name":"uv","version":"0.10.2","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
908bac76de6f6ac4d45344af926cbb29500aa82455df1048df9660d150a0d052
|
|
| MD5 |
ef7fc44df6e2e386ecc9f63228d2ac28
|
|
| BLAKE2b-256 |
09660e2fade263fcedbd688eb6d70a6546dedf08b811790a393ff243a36d93d7
|
File details
Details for the file invariant_core-0.4.0-py3-none-any.whl.
File metadata
- Download URL: invariant_core-0.4.0-py3-none-any.whl
- Upload date:
- Size: 38.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.10.2 {"installer":{"name":"uv","version":"0.10.2","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d899b1f030bb80119cebbf37cd081b0f3f1a84c641a6f095101aeb93352a20bd
|
|
| MD5 |
7af5b4940728bcf24115c87c95086250
|
|
| BLAKE2b-256 |
064b2b4476aedab212fcd25c224a077db679a6cab8bd7a7054c47483cb4b950c
|