An open API service indexing awesome lists of open source software.

https://github.com/jvoltci/shivya

Consensus-free distributed resource-sharing mesh for edge devices: Hodge curl-projection reconciler + Onsager-coupled active inference (Rust, zero-dep math core).
https://github.com/jvoltci/shivya

active-inference distributed-systems exterior-calculus morphogenesis non-dual rust wasm

Last synced: 7 days ago
JSON representation

Consensus-free distributed resource-sharing mesh for edge devices: Hodge curl-projection reconciler + Onsager-coupled active inference (Rust, zero-dep math core).

Awesome Lists containing this project

README

          

# Shivya: A Consensus-Free Distributed Resource-Sharing Mesh

[![CI Build](https://github.com/jvoltci/shivya/actions/workflows/test.yml/badge.svg)](https://github.com/jvoltci/shivya/actions)
[![crates.io](https://img.shields.io/crates/v/shivya.svg)](https://crates.io/crates/shivya)
[![docs.rs](https://docs.rs/shivya/badge.svg)](https://docs.rs/shivya)
[![License](https://img.shields.io/crates/l/shivya.svg)](https://github.com/jvoltci/shivya/blob/master/LICENSE-MIT)

![Shivya Architecture Hero Banner](docs/assets/hero.svg)

Shivya is a bare-metal, zero-dependency Rust runtime for **load and state balancing across arrays of edge devices**. In one line: a **consensus-free distributed edge-flow runtime** that performs **localized active-inference load balancing** and settles to **eventual curl-free convergence** over a simplicial state complex. It is *not* a replacement for Paxos or Raft; it does not provide linearizable global ordering. What it does provide is:

1. **A curl-projection reconciler** (Hodge **curl projector** over a discrete simplicial state complex — specifically the coexact / `d₁ᵀβ` component is solved for and subtracted; the gradient and harmonic components are not separately extracted) that lets concurrent edge writes settle to a single curl-free state after partitions heal, without a consensus round.
2. **A multi-agent active-inference loop** that diffuses load through symmetric Onsager couplings, so nodes redistribute work toward minima of a collective variational free-energy functional.
3. **A bit-packed VSA memory and cognitive model (Shivya Mind)** that lets individual nodes maintain a power-law decaying episodic history under a fixed 40 KB memory ceiling, segmenting experience boundaries dynamically based on information-theoretic surprise.

The underlying mathematics is cited in [CITATIONS.md](CITATIONS.md). When this regime is appropriate is discussed in [docs/philosophy.md](docs/philosophy.md) (it is *not* a fit for workloads that require linearizable consistency).

---

## The 5-Layer Stack

```mermaid
graph TD
Layer4["Layer 4: Morphogenetic Topology (Reaction-Diffusion Split/Cull)"] --> Layer3["Layer 3: Onsager Ensemble (Reciprocal Flow Diffusion)"]
Layer3 --> Layer2["Layer 2: Register VM + Expression-Tree Hill-Climber"]
Layer2 --> Layer1["Layer 1: Variational Active Inference (Free-Energy Minimisation)"]
Layer1 --> Layer0["Layer 0: Hodge Mesh (Simplicial Boundary Reconciler)"]
```

### Layer 0: Topological Fabric [`shivya-hodge`](https://crates.io/crates/shivya-hodge)
- **What it is:** A simplicial state complex (vertices = nodes, edges = oriented flows, triangles = concurrent contexts) with the Discrete Exterior Calculus boundary operators d0 and d1.
- **What it does:** Projects any 1-chain (edge-flow) delta onto the curl-free subspace. The coexact (curl) component `d₁ᵀβ` is isolated by solving the Laplacian system `L₂ β = d₁ ΔS` with an in-tree Conjugate Gradient solver (CG residual norm `‖r‖ ≤ 1e-8`, max 1000 iterations) and subtracted from the input. The gradient (`d₀α`) and harmonic (`γ`) components are not separately extracted — this is a curl projector, not a full Hodge decomposition. The remaining flow is curl-free and identical across all nodes that observe the same complex.

### Layer 1: Variational Active Inference [`shivya-flux`](https://crates.io/crates/shivya-flux)
- **What it is:** A Gaussian active-inference agent per node, with a generative model over internal beliefs `μ_q`, sensory observations `s`, and active states `a` bounded by a Markov blanket.
- **What it does:** Minimises Variational Free Energy `F = KL(q‖prior) + nll(s | g(μ_q))` via gradient descent on the belief mean. Matrix inversions are ridge-stabilised (1e-6 diagonal regularisation) — degenerate covariances never crash the daemon; see `SubstrateError` in [crates/shivya-hodge/src/error.rs](crates/shivya-hodge/src/error.rs) (re-exported from `shivya-flux`).

### Layer 2: Register-IR Interpreter + Expression-Tree Hill-Climber [`shivya-morphic`](https://crates.io/crates/shivya-morphic)
- **What it is:** A deterministic register-IR interpreter — sandboxed by a hard 500-instruction budget per evaluation — paired with a generative model that can grow its own latent dimension at runtime. The running code is *not* mutated mid-execution; the VM is plain, deterministic interpretation. The instruction stream and register slab are heap-backed `Vec`s, not stack-resident — "fixed-budget", not "stack-allocated".
- **What it does:** Compiles a small expression AST (`Const`, `Var`, `Add`, `Mul`) to register-machine instructions and evaluates them under the 500-cycle ceiling. When the moving-average free energy crosses a novelty threshold, the agent's internal state space expands (e.g., from 2D to 3D), and a *separate, offline* **stochastic 1+1 mutation hill-climber** proposes a replacement symbolic update law (per-node mutation: subtree replacement at p=0.15, otherwise constant jitter / variable index toggle / operator swap; no crossover, no population, greedy accept-if-better against an MSE proxy). The new tree is accepted only if it lowers the free-energy proxy on a held-out window.

### Layer 3: Thermodynamic Collective Ensemble [`shivya-onsager`](https://crates.io/crates/shivya-onsager)
- **What it is:** A coupled set of active-inference agents with a symmetric Onsager flow matrix `L_ij = L_ji` and a Harsanyi-dividend computation over neighbourhood coalitions.
- **What it does:** Migrates belief parameters between adjacent agents at a rate proportional to belief-distance and `L_ij`. The collective free energy `F_collective = Σ F_i − Σ Harsanyi dividends` rewards synergistic neighbourhoods and penalises antagonistic ones, biasing the ensemble toward cooperative configurations. Coalition masks are encoded as `u8`, capping local coalitions at ≤ 8 nodes; the ensemble itself scales beyond that, only the per-node coalition computation does not.

### Layer 4: Morphogenetic Pattern Substrate [`shivya-turing`](https://crates.io/crates/shivya-turing)
- **What it is:** A graph-Laplacian Gierer-Meinhardt reaction-diffusion system over the node mesh, with RK4 integration and an explicit CFL stability bound (`dt ≤ 0.45 / (D_max · degree_max)`).
- **What it does:** Picks up local "stress" hot-spots from the activator field. High-activation nodes can fire a *split* (pre-allocated object-pool mitosis with no runtime resize); low-utility, low-activation nodes are culled (apoptosis), gated on a minimum-cluster-size floor so the mesh can't collapse.

### The Cognitive Core: VSA Episodic Memory [`shivya-mind`](crates/shivya-mind)
- **What it is:** A bit-packed Vector Symbolic Architecture (VSA) memory engine operating in a 10,000-dimensional space `{−1, +1}^D` (MAP-B algebra), running under a fixed 40 KB memory floor.
- **What it does:** Holographically binds role-value sensory experiences (XOR, rotation, and bundling operations) into a single hypervector tally on the wire (~1.25 KB). A predictor monitors the active-inference `F` (Variational Free Energy) surprise rate from `shivya-flux` to segment experiences into discrete episodes (beads). Memory decays naturally following power-law retention dynamics, and retractions use exact algebraic subtraction (`M ← M − λF`).

---

## Decentralized P2P Transport [`shivya-p2p`](https://crates.io/crates/shivya-p2p)

The transport unifies geographically separated daemons into the same statistical field:
- **Structural Kademlia K-bucket table ([routing.rs](crates/shivya-p2p/src/routing.rs)):** 160-bit `NodeId`, 160 buckets indexed by leading-zero bits of XOR distance, `K=4` stack-resident peer slots per bucket (`[Option; K]` — no per-bucket heap allocation), LRU-on-touch ordering, and ping-oldest-first eviction (a full bucket pings its head; if the head doesn't re-touch within 500 ms, the candidate replaces it). Scope: direct neighbourhood tracking. Not a from-scratch RFC-grade Kademlia — see "What's not in scope" below.
- **Kademlia RPC ([transport.rs](crates/shivya-p2p/src/transport.rs)):** `PING` / `PONG` / `FIND_NODE` / `FOUND_NODES` / `STORE` / `FIND_VALUE` / `FOUND_VALUE`, all single-MTU UDP frames with fixed-layout big-endian binary serialisation in [protocol.rs](crates/shivya-p2p/src/protocol.rs) (hand-written, no `serde`/`bincode`). A local DHT key/value store backs `STORE` / `FIND_VALUE`.
- **What's not in scope:** FIND_NODE fans out to every known peer rather than the canonical α-bounded shortlist of closest peers; there is no inflight set, no convergence termination, no formal iterative-lookup state machine. Convergence is bottom-up across ticks. IPv6 socket addresses are not yet supported on the wire. Treat the transport as "good enough for an edge mesh of dozens", not as a drop-in BitTorrent/IPFS replacement.
- **Partition-test hook:** Each transport exposes `block(addr)` / `unblock(addr)` to silently drop frames to a given address. Used by [tests/jepsen_partitions.rs](tests/jepsen_partitions.rs) and [tests/chaos_ensemble.rs](tests/chaos_ensemble.rs) to inject controllable partitions and assert that the Layer-0 reconciler cancels the curl introduced by the split.

---

## Native Edge Daemon (`shivya-cli`)

A Tokio multi-threaded daemon that samples real host telemetry via [`sysinfo`](https://docs.rs/sysinfo) (CPU load, NIC bytes RX/TX, memory pressure), steps the 5-layer stack on a 1 Hz interval, and bridges to the visualiser dashboard. The orchestrator owns a **`WorkloadMeshProxy`** ([crates/shivya-cli/src/bridge.rs](crates/shivya-cli/src/bridge.rs)) that maps idiomatic application signals onto the substrate:

```text
record_queue_len(node, q) → 0-simplex mass at vertex
record_offload(src, dst, rate) → oriented 1-chain entry on the edge
settle() → curl-free routing recommendation per edge
```

The bridge is **live inside the 1 Hz loop**: every tick the orchestrator pulls queue length from the Layer-4 activator field and offload rates from the Onsager coupling × belief differential, runs `settle()`, and writes the reconciled per-edge rates back into `complex.edge_states` and the Onsager `l_matrix`. The current curl norm and the full `Vec` are exposed in the daemon's status JSON.

External callers can drive the bridge directly through the Unix domain socket at `/tmp/shivya_cli_.sock`:

```text
STATUS → SystemStatus JSON (default for empty input)
Q → record_queue_len, return recommendations JSON
O → record_offload, return recommendations JSON
SETTLE → settle + apply, return recommendations JSON
```

Use this if you don't want to read DEC papers — feed it queue lengths and offload rates, and `settle()` returns the substrate's recommendation for how to re-route work.

### CLI commands

- `--port `: UDP listener port (default 8085).
- `--peer `: Bootstrap into an existing peer. Sends a `PING` followed by `FIND_NODE(self)` to seed iterative discovery.
- `--visualize`: Spawns a non-blocking WebSocket server on `127.0.0.1:9002` streaming orchestrator status to the dashboard.
- `--daemon` (unix only): `fork()` and session-detach via the `daemonize` crate *before* the Tokio runtime is constructed. PID file at `/tmp/shivya.pid`; stdout/stderr at `/tmp/shivya.{out,err}`.

```bash
# Seed node with live visualisation
cargo run --release -p shivya-cli -- start --port 8085 --visualize

# Bootstrap a second peer
cargo run --release -p shivya-cli -- start --port 8086 --peer 127.0.0.1:8085

# Query substrate status via local UDS
cargo run --release -p shivya-cli -- status
```

`SIGINT` / `SIGTERM` triggers an orderly shutdown that unbinds the socket file and notifies peers.

---

## Consistency Model

Shivya is **not** a CP system in the CAP sense. It does not give you linearizable reads, single-master semantics, or a transaction log.

What it *does* give you:

| Property | What's guaranteed | What's not |
|---|---|---|
| **Curl-free convergence** | After partitions heal, every node's edge-flow state projects to the same curl-free 1-chain. The Hodge projector is idempotent (the integration test asserts this to 1e-7). | Per-step ordering of writes between nodes. |
| **Continuous belief diffusion** | Adjacent agents exchange belief parameters at rate proportional to `L_ij`, biased by Harsanyi dividends. | Bounded staleness. Diffusion is asymptotic, not bounded-time. |
| **Free-energy minimisation** | Across the chaos→recovery cycle in [tests/chaos_ensemble.rs](tests/chaos_ensemble.rs), the cool-down (post-recovery) F average beats the pre-chaos warm-up average under 15% packet loss + random per-node isolation + a hard programmatic partition. Asserts global stabilisation immediately following partition recovery. | Monotonic descent step-by-step. F may transiently rise *during* the chaos window before stabilising in cool-down. |
| **Survival under telemetry degeneracy** | Singular covariance matrices fall back to ridge-regularised (1e-6) inversion, then to the identity matrix. No `panic!` on the main path. | Numerical "quality" once telemetry collapses — you'll get a finite recommendation, but it carries less information. |
| **Network partition heal** | The Kademlia layer reconverges after split-brain. Chaos test confirms ≥ N/2 peers visible per node after recovery. | An explicit re-election. Recovery is bottom-up: peers find peers, then the math reconverges. |

Use Shivya when "eventually identical edge-flow state" is acceptable and you care about **avoiding the latency and complexity tax of consensus**. Don't use it when you need linearizable per-key semantics, an authoritative transaction log, or strict bounded staleness.

---

## Performance Envelope

Numbers below are from the workspace's own tests on a single laptop (Apple M-class, macOS). Take them as **order-of-magnitude indicators**, not benchmarks.

| Operation | Cost / scale |
|---|---|
| `ensemble.step` (7 nodes, 2-D beliefs, full L0-L4 with chaos) | ~7-8 ms per step |
| `reconcile_state_delta` (6-edge bowtie complex, 2 triangles) | < 0.1 ms (CG solver converges in ≤ 10 iterations) |
| UDP transport `send_to` (per frame) | tens of µs (kernel-bound) |
| Full-mesh discovery, 7 real-UDP nodes on localhost | ~1.5 s to converge to N−1 peers, with retry |
| K-bucket capacity | K=4 per bucket × 160 buckets per node |
| Coalition mask (Harsanyi dividend computation) | u8 ⇒ ≤ 8 nodes per local coalition |
| Morphic VM instruction budget | hard cap, 500 cycles per program |

Convergence behaviour:
- **Curl projector**: idempotent. After one pass the residual `‖d₁ · S_reconciled‖` is below the CG tolerance (1e-8 by default).
- **Belief update (gradient descent on μ_q)**: monotonic on smooth observations; converges in ≤ 10 iterations to a 1e-4 gradient norm in the test cases.
- **Onsager ensemble F**: in the chaos test (7 nodes, 15% drop, hard partition, ~80 steps), the COOLDOWN-phase trailing-10 average (sampled after every block is cleared and a fresh full-mesh discovery has restored N-1 peer visibility) beats the WARMUP-phase leading-10 average — i.e. F stabilises below its pre-chaos baseline immediately after recovery. Step-by-step monotonic descent during the chaos window itself is *not* asserted; F may transiently rise while partitions are active.

---

## Convergence vs Pairwise Gossip Averaging

This section reports head-to-head numbers from `crates/shivya-flux/benches/convergence_vs_gossip.rs` (run with `cargo bench -p shivya-flux --bench convergence_vs_gossip`). The fixture is a 7-node bowtie-plus-tail complex (8 edges, 2 triangles). Gossip is the standard randomised-pairwise-averaging primitive from Boyd et al. 2006, seeded deterministically, sharing the same allocator, same convergence tolerance (`EPS = 1e-3`), same topology. The two algorithms see identical input state per fixture.

There are two fixtures, and **they tell opposite stories on purpose**.

### Fixture A — single-source scalar spike (pure-gradient regime)

`V0 += 50` over a baseline `BASE_LOAD = 100`. The merged disagreement is a pure-gradient field on the node potentials: there is no rotational component for any algorithm to project out.

| Algorithm | Steps | Wall (µs) | Bytes exchanged | Heap allocs | Heap bytes | Final L∞ | Final curl |
|---|---:|---:|---:|---:|---:|---:|---:|
| Shivya (flux + hodge) | 881 | 1067 | 112768 | 20038 | 1081920 | 0.00e0 | 0.00e0 |
| Gossip (pairwise avg) | 471 | 5 | 7536 | 0 | 0 | 7.49e-4 | 0.00e0 |

**Honest reading: pairwise gossip wins this fixture on every cost axis.** ~1.9× fewer steps, ~200× lower wall clock, ~15× less bandwidth, zero allocations vs ~1 MB. Both reach effectively the same final state quality. Shivya is doing work it does not need to do here — the input flow has no curl, and the projector is solving a Laplacian system whose right-hand side is identically zero. If your workload is a scalar quantity diffusing across a graph from a single source, gossip is the right tool. There is no need to retire pairwise averaging just because Shivya exists.

### Fixture B — multi-side concurrent boundary conflict (rotational regime)

Two opposing partition cliques (`L = {V0, V1}`, `R = {V3, V4}`) concurrently write oriented edge flux around the closing edges of each bowtie triangle (`|δ| = 25` per edge). The merged 1-chain is not the gradient of any single scalar field; per the Hodge decomposition, it carries a non-trivial coexact (curl) component on each of the two triangles. This is the regime Shivya was designed for.

| Algorithm | Steps | Wall (µs) | Bytes exchanged | Heap allocs | Heap bytes | Final L∞ | Final curl |
|---|---:|---:|---:|---:|---:|---:|---:|
| Shivya (curl projector) | 1 | 1 | 128 | 22 | 1168 | 0.00e0 | 0.00e0 |
| Gossip (pairwise avg) | 1 | 0 | 144 | 0 | 0 | 0.00e0 | **7.50e+1** |

**Honest reading: on the rotational regime, gossip cannot see the disagreement it has just integrated.** Both algorithms drive their *scalar* node-potential view to the global mean. The decisive difference is the persistent curl on the edge-flow field that gossip leaves behind: `‖d₁ · f‖∞ = 75.0`. Pairwise scalar averaging has no representation of oriented rotational flow; it integrates each clique's edge writes into a node-level summary and then forgets that the edges ever disagreed. Shivya's Hodge projector recognises the curl and removes it in a single pass (no iteration; the projector is idempotent).

### Where each regime occurs in practice

The two fixtures are not equally common in real edge fleets. Single-source scalar diffusion is the everyday case (one hot node, the rest absorb its slack) and is where most published load-balancers operate; gossip handles it well. Multi-side concurrent oriented conflict is the *partition-recovery* case: two halves of a fleet independently route load while disconnected, and on heal the merged edge-flow chain has rotational disagreement around any cycle the partition straddled. Curl-projection turns this from an off-by-some-percent steady-state error into an exact cancellation in one step.

If your workload never spans a partition that closes a cycle, you do not need Shivya — pick gossip. If it does, Shivya gives you a primitive gossip cannot construct.

---

## Rust Integration Example

```rust
use shivya::hodge::complex::SimplicialStateComplex;
use shivya::flux::model::GibbsFluxAgent;
use shivya::morphic::{DynamicGibbsAgent, MorphicHotSwapper, Expr};
use shivya::onsager::OnsagerCollectiveEnsemble;
use std::thread;
use std::time::Duration;

fn main() {
println!("Initializing Shivya substrate...");

// 1. Establish initial 3-node topology
let adjacent_nodes = vec![
vec![1, 2], // Node 0 neighbours
vec![0, 2], // Node 1 neighbours
vec![0, 1], // Node 2 neighbours
];

// 2. Set up active-inference agents
let create_agent = |mu_prior: f64| {
DynamicGibbsAgent::new(
2, 1, 2,
vec![mu_prior, 0.0],
vec![vec![10.0, 0.0], vec![0.0, 10.0]],
vec![vec![1.5, 0.0], vec![0.0, 1.5]],
vec![vec![0.1, 0.0], vec![0.0, 0.1]],
vec![vec![0.0], vec![0.0]],
vec![vec![0.0], vec![0.0]],
vec![0.0, 0.0],
vec![vec![1.0, 0.0], vec![0.0, 1.0]],
5.0, // novelty threshold
)
};

let agents = vec![create_agent(0.1), create_agent(0.2), create_agent(0.15)];

// 3. Construct the Onsager ensemble
let mut ensemble = OnsagerCollectiveEnsemble::new(agents, adjacent_nodes, 0.5);

// 4. Run the loop in a worker thread
let handle = thread::spawn(move || {
let observations = vec![
vec![1.2, 0.8],
vec![2.0, 1.1],
vec![1.5, 0.9],
];
for step in 1..=5 {
let collective_f = ensemble.step(&observations, 0.1, 10, 1e-4, 0.1);
println!("Step {} -> Collective F: {:.6}", step, collective_f);
thread::sleep(Duration::from_millis(50));
}
});

handle.join().unwrap();
println!("Loop terminated stably.");
}
```

For an application-level integration that *doesn't* require knowing what a 1-chain is, use `shivya-cli`'s `WorkloadMeshProxy` ([bridge.rs](crates/shivya-cli/src/bridge.rs)).

---

## Crates

The crates split cleanly into two tiers — *math* (zero-dependency, stack-allocated, WASM-native) and *runtime* (native binaries on top of Tokio):

**Math + simulation tier — Layers 0-4 + Cognitive Core. Zero external dependencies, stack-allocated where possible, build cleanly for `wasm32-unknown-unknown`. This is what powers the in-browser simulation cockpit at [`docs/index.html`](docs/index.html) — including the live "Cognitive Core" panel that streams VSA working-memory signatures from `shivya-mind`:**

- [`crates/shivya-hodge`](https://crates.io/crates/shivya-hodge) — Layer 0: simplicial DEC operators + curl projector
- [`crates/shivya-flux`](https://crates.io/crates/shivya-flux) — Layer 1: variational free-energy agent
- [`crates/shivya-morphic`](https://crates.io/crates/shivya-morphic) — Layer 2: deterministic register-IR interpreter (500-cycle cap) + stochastic 1+1 mutation hill-climber over expression trees
- [`crates/shivya-onsager`](https://crates.io/crates/shivya-onsager) — Layer 3: multi-agent thermodynamic ensemble
- [`crates/shivya-turing`](https://crates.io/crates/shivya-turing) — Layer 4: reaction-diffusion topology adaptation
- [`crates/shivya-mind`](crates/shivya-mind) — Cognitive Core: bit-packed Vector Symbolic Architecture (VSA) episodic memory engine + surprise-driven segmenter
- [`crates/shivya-telemetry`](https://crates.io/crates/shivya-telemetry) — `wasm-bindgen` glue that re-exports the math tier into the browser orchestrator

**Runtime tier — native targets only. These depend on Tokio for asynchronous I/O and the host kernel's socket API, so they are not WASM-portable and are not meant to be:**

- [`crates/shivya-p2p`](https://crates.io/crates/shivya-p2p) — Kademlia XOR routing + UDP transport (Tokio + UDP sockets)
- [`crates/shivya-cli`](https://crates.io/crates/shivya-cli) — native daemon + UDS CLI + `WorkloadMeshProxy` (Tokio runtime, `sysinfo`, optional `daemonize`)

---

## License

Distributed under the terms of both the MIT license and the Apache License (Version 2.0).
See [LICENSE-MIT](LICENSE-MIT) and [LICENSE-APACHE](LICENSE-APACHE) for details.