A probabilistic agent state layer. Bayesian belief tracking for long-running personal AI agents.
Project description
nous-state
A probabilistic agent state layer for long-running personal AI agents.
"Knowledge is prediction, not storage."
The Problem
Every long-running AI agent eventually hits the same wall:
User (Month 1): "I work at Sarvam AI using Mistral for NyayaSahayak."
User (Month 4): "I switched to GPT-4 and joined Google DeepMind."
Agent (Month 5): *confidently tells someone Pranav uses Mistral at Sarvam AI*
Vector databases store both facts. Knowledge graphs require manual conflict resolution. Both need an expensive LLM call to decide which fact wins. Neither gives you a mathematically principled answer.
nous-state solves this with Bayesian probability distributions — the same math used in GPS navigation, spam filters, and medical diagnostics.
How It Works
Instead of storing facts, Nous maintains belief distributions over entity attributes:
Pranav.employer = { "Sarvam AI": 0.82, "unknown": 0.18 }
When new evidence arrives, it performs a Bayesian update:
"Pranav joined Google DeepMind" →
Pranav.employer = { "Google DeepMind": 0.86, "Sarvam AI": 0.12, "unknown": 0.02 }
Every update is recorded as an immutable Delta — a change in understanding, not just a fact. This means:
- Contradictions are resolved mathematically, not heuristically
- History is queryable — "What did the agent believe about Pranav in March?"
- Forgetting is principled — unused beliefs decay toward uncertainty via entropy
- Identity resolution is automatic — two entity names with high mutual information are flagged as likely the same person
Install
# From PyPI (coming soon)
pip install nous-state
# From GitHub (now)
pip install git+https://github.com/Pranavsingh431/nous-state.git
Zero runtime dependencies. Pure Python stdlib only (math, sqlite3, json, urllib).
Quickstart
Rule-based extraction (no LLM needed)
from nous import Nous
memory = Nous("agent_memory.db")
# Session 1
memory.observe("Pranav works at Sarvam AI as an ML engineer.")
memory.observe("He is building NyayaSahayak using Mistral.")
# Session 4 — things changed
memory.observe("Pranav left Sarvam AI and joined Google DeepMind.")
memory.observe("NyayaSahayak now uses GPT-4 for better legal reasoning.")
# Query current beliefs
print(memory.predict("Pranav", "employer"))
# → {"Google DeepMind": 0.86, "Sarvam AI": 0.12}
print(memory.predict("NyayaSahayak", "model"))
# → {"GPT-4": 0.86, "Mistral": 0.12}
With LLM extraction (natural language → structured beliefs)
from nous import Nous
from nous.llm_extractor import LLMExtractor
extractor = LLMExtractor(
api_key="sk-or-...", # Any OpenRouter key
user_context={"name": "Pranav"} # Resolves "I/me/my" → "Pranav"
)
memory = Nous("agent_memory.db", extractor=extractor)
# Now feed raw conversation turns directly
memory.observe("I switched from Mistral to GPT-4 because legal reasoning improved.")
memory.observe("Actually wait, someone said Pranav is still at Sarvam AI?")
memory.observe("No confirmed, he's definitely at Google DeepMind on Gemini.")
print(memory.predict("Pranav", "employer"))
# → {"Google DeepMind": 0.97, "Sarvam AI": 0.03}
Explainability — why does the agent believe this?
# Full auditable history
for delta in memory.history("Pranav", "employer"):
print(f"Surprise: {delta.surprise:.1f} bits | {delta.evidence[:60]}")
# → Surprise: 4.3 bits | Actually, I left Sarvam AI and joined Google...
# → Surprise: 0.2 bits | Pranav is definitely at Google DeepMind, I saw...
# → Surprise: 0.1 bits | Yes confirmed, Pranav is at Google DeepMind...
# Time-travel: what did the agent believe 30 days ago?
past_belief = memory.query_at("Pranav", "employer", at_time=timestamp_30_days_ago)
Surprise scoring — filter noise before it enters memory
# How surprising is this claim, given what we already know?
bits = memory.surprise("Pranav is still at Sarvam AI.")
# → 5.1 bits (high — contradicts current belief)
bits = memory.surprise("Pranav works at Google DeepMind.")
# → 0.1 bits (low — we already know this)
API Reference
Nous(db_path, extractor=None)
| Method | Description |
|---|---|
observe(text, source, reliability) |
Ingest text, update beliefs |
predict(entity, attribute) |
Get current probability distribution |
query_at(entity, attribute, timestamp) |
Time-travel query |
history(entity, attribute) |
Full delta log for an attribute |
explain(entity, attribute, value) |
Why does the agent believe X? |
surprise(text) |
Information content in bits before observing |
get_coupling(entity_a, entity_b) |
Identity similarity score (0–1) |
get_entity_profile(entity) |
All attributes for an entity |
apply_decay(current_time) |
Apply forgetting to stale dimensions |
LLMExtractor(api_key, model, user_context)
Works with any OpenAI-compatible API endpoint (OpenRouter, OpenAI, local via LM Studio).
| Parameter | Default | Description |
|---|---|---|
api_key |
required | Your API key |
model |
google/gemini-2.5-flash |
Any model on OpenRouter |
user_context |
{} |
Dict with name key to resolve "I/me/my" |
Architecture
Natural Language
↓
LLMExtractor (or rule-based Extractor)
↓
(entity, attribute, value) tuples
↓
BayesianUpdater → surprise score → posterior distribution
↓
Dimension (probability distribution) + Delta (immutable history)
↓ ↓
WorldModel (in-memory cache) DeltaLog (SQLite)
↓
PersistenceLayer (SQLite — survives restarts)
Key properties:
O(1)reads — dictionary lookup, no vector searchO(k)writes — multiply k floats, normalize (k = number of known values, typically < 10)- Append-only delta log — the most scalable write pattern in systems
- Zero external dependencies
Why Not Vector DB / Knowledge Graph?
| Problem | Vector DB | Knowledge Graph | nous-state |
|---|---|---|---|
| Contradictory facts | Stores both, LLM decides | Manual conflict rules | Bayesian update (automatic) |
| Stale high-confidence facts | Still retrieved | Still in graph | Probability mass shifts |
| "Why does agent believe X?" | Not possible | Requires audit log | Native (delta history) |
| Identity resolution | Cosine similarity | Entity linking heuristics | Mutual information coupling |
| Compute cost (read) | O(n) ANN search | O(edges) traversal | O(1) dict lookup |
License
MIT — see LICENSE.
Contributing
Issues and PRs welcome. This is early-stage — if you hit a real-world edge case, opening an issue is genuinely valuable.
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 nous_state-0.1.0.tar.gz.
File metadata
- Download URL: nous_state-0.1.0.tar.gz
- Upload date:
- Size: 28.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.10.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ae1bd0c7258d656cb72f6e527017c32fe2728cd1e3f3ac7cc116b3065bd29f4b
|
|
| MD5 |
e1b2934d5d1c752db40310e26cb744f3
|
|
| BLAKE2b-256 |
71eab03c97fc6cf70668c38f64ad8bb48153480dffe9c09f8e8b354bc3ec0531
|
File details
Details for the file nous_state-0.1.0-py3-none-any.whl.
File metadata
- Download URL: nous_state-0.1.0-py3-none-any.whl
- Upload date:
- Size: 24.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.10.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e63776f1f071dfb183b109f96c8ed32749ffd28d50aae7d29d9b7b5424c605ec
|
|
| MD5 |
5bb7e971be67036f4d25b4a223030aa4
|
|
| BLAKE2b-256 |
369df4bafc39af5c7d8c74de6f09f22c448df900fa7d518fcbc87c73bd7588f8
|