Reference implementation (Python) of the harel statechart engine.
The normative SPEC.md, the JSON Schema for machine YAML, and the cross-language
conformance suite live in the spec repo. This repository implements that spec in
Python and is correct iff it passes the conformance suite.
Implements the harel spec v0.0.1 (early alpha; all fruwehq harel repos share one synchronized version).
Status: passing the full conformance suite — all 22 engine cases
(conformance/01–22) plus conformance/cli/01–02. Implements YAML 1.2 loading
- validation, the full statechart semantics (RTC dispatch, hierarchy, orthogonal
regions +
done, shallow/deep history, esvs, CEL guards, structured actions, active objects + bus, defer, timers, faults), static contracts, snapshot round-trip + safe-point migration, Mermaidexport, and the §13 CLI. Built up the build order in issue #3.
The cross-language conformance suite is the single source of truth for correctness;
this repository is correct iff it passes it. The suite lives in
fruwehq/harel-conformance; the test
harness fetches it at the matching release tag (v0.0.1) into a gitignored
.cache/ — no git submodule. The normative SPEC.md and JSON Schema live in
fruwehq/harel; the schema-drift test fetches the
schema at the same tag.
For offline work, point the harness at a local checkout:
export HAREL_CONFORMANCE_DIR=/path/to/harel-conformance # the suite
export HAREL_SPEC_DIR=/path/to/harel # the schema (optional)
- Load and validate machine YAML against
schema/machine.schema.json, parsed under the YAML 1.2 core schema (onlytrue/falseare booleans). - Execute statecharts per
SPEC.md: run-to-completion; hierarchy; orthogonal regions (+done); shallow/deep history;initialtransitions;esvs(extended-state variables declared in states, hierarchical) includingexternalesvs + theenvevent andrefresh;defer(deferred-set, edge-triggered); timers via an injected clock; active-object spawning;publish(directed / by subscription / scoped); and faults (theerrorevent). - Guards in CEL (e.g.
cel-python); structured actions (assign/publish/refresh/spawn/stop) with CEL values. - Adapters — bus / queue / clock / store / observer (SPEC §8), each with a simple in-memory default for tests.
- An
exportcommand that renders a machine (and an instance's currentstate_config) to MermaidstateDiagram-v2(SPEC §12), behind a pluggable exporter interface so more formats (PlantUML, SCXML, …) can be added later. - A test harness that runs the upstream conformance cases against this engine.
The CLI (harel …) is a thin wrapper over a programmatic API; an engine can be
embedded in a host program without the CLI or the file-backed store (SPEC §2):
import harel
defs = harel.load_definitions(open("gate.yaml").read())
harel.validate(defs[0].raw) # raises ValidationError if invalid
host = harel.Host()
host.register_all(defs)
inst = host.create_root(host.machines["gate"], "g1", external={"fare": 50})
host.run_to_quiescence()
host.deliver("g1", "coin", {"amount": 100}) # typed event; False if rejected
host.run_to_quiescence()
assert inst.active_leaf_names() == ["unlocked"]
assert inst.resolved_esvs()["fare"] == 50
assert inst.status is harel.Status.ACTIVE
host.advance("30s") # virtual clock
snaps = host.snapshot_all() # persist / round-trip (§8)
host.restore_all(snaps)The public surface is everything exported from the top-level harel package
(harel.__all__): Host, Instance, Definition, Machine, Status, Event,
load_definitions / load_definition, validate / collect_errors, and the
error types. See tests/test_library_api.py.
src/harel/— the package.tests/— unit tests and the conformance harness.
python -m venv .venv && . .venv/bin/activate
pip install -e '.[dev]'
ruff check . && mypy src/harel && pytest # the suite is fetched into .cache/ on first run
MIT — see LICENSE.