A zero-dependency SQLite memory layer for AI agents that learns what works: patterns graduate to proven preferences via a built-in experiment loop.
Project description
metabrain
A SQLite memory layer for AI agents that learns what works. Zero dependencies. One file.
Most agent-memory tools store what you tell them and hand it back later. metabrain does that too — but it also closes the loop: a pattern you record enough times graduates into a hypothesis, every outcome you log becomes an experiment for or against it, and once the evidence clears the bar it graduates again into a proven preference. Your agent stops guessing and starts running on rules it earned.
learn(pattern) → recurs → hypothesis (under test)
→ each verdict is an experiment (supports / refutes)
→ evidence clears the bar → preference (a proven rule)
That loop is the whole point. It runs on the Python standard library — no vector database, no server, no API keys.
Install
pip install metabrain
Python 3.10+. No dependencies beyond the standard library. (Import name is metabrain.)
Quick start
from metabrain import MetaBrain
db = MetaBrain("agent.db")
with db.session(task="content") as s:
# A hunch. Record it as you notice it — three times and it's worth testing.
s.learn("pattern", "question hooks lift saves", domain="instagram")
s.learn("pattern", "question hooks lift saves", domain="instagram")
s.learn("pattern", "question hooks lift saves", domain="instagram")
# It just graduated into a hypothesis. Now test it against reality.
h = db.hypotheses(status="testing")[0]
post = s.unit("carousel with a question hook", kind="contract", hypothesis=h.id)
s.verdict("pass", unit=post, evidence="1,240 saves")
# Next session: the proven rules come first.
brief = db.read_start()
for rule in brief.preferences: # things metabrain has *proven*
print("PROVEN:", rule.insight)
for h in brief.open_hypotheses: # things it's still testing
print("testing:", h.statement, f"({h.confidence:.0%})")
You don't have to open a session — the flat API (db.learn(...), db.verdict(...)) works too and attaches to an ambient session automatically, so the telemetry still fills.
Why it's different
| metabrain | typical vector-memory store | |
|---|---|---|
| Remembers what you tell it | ✅ | ✅ |
| Proves which memories actually work | ✅ the learn→experiment→graduate loop | ❌ |
| Working state + telemetry, not just recall | ✅ units, checkpoints, sessions, events | ❌ |
| Infrastructure | a single SQLite file | vector DB / server / API key |
| Dependencies | none (stdlib sqlite3) |
several |
Recall stays deliberately simple — substring + a hit counter — because the moat is the loop, not embedding search. (Semantic recall may arrive later as an opt-in metabrain[embeddings] extra; the core will always be zero-dependency.)
Built for real, stateful products
The loop is general. Three shapes it was designed against:
Self-learning content engine. Each post is a unit; engagement is the verdict. Hooks that keep winning graduate into the brand's proven playbook.
s.learn("pattern", "carousels outperform single images", domain="ig") # ...×3 → hypothesis
for saves, ok in [(1200,"pass"), (90,"fail"), (1500,"pass"), (1100,"pass")]:
post = s.unit(f"carousel ({saves} saves)", kind="contract", hypothesis=h.id)
s.verdict(ok, unit=post, evidence=f"{saves} saves")
# 3/4 supported → graduates into the playbook
Lead capture. Each lead is a unit with its own checkpoint trail; a tactic about what converts graduates once enough leads confirm it.
lead = s.unit({"name": "Acme", "source": "webinar"}, kind="contract")
s.checkpoint({"stage": "demo booked"}, unit=lead)
s.verdict("pass", unit=lead, evidence="closed")
Self-improving job applications. Each application is a unit; "lead with a shipped metric" stays a guess until enough replies prove it, then becomes a rule.
app = s.unit({"company": "Acme"}, kind="contract", hypothesis=h.id)
s.verdict("pass", unit=app, evidence="recruiter replied")
How the tables fill themselves
metabrain has seven tables, and you never write to them directly — correct use of the API fills every one as a side effect. Open a session and each write inherits its id, emits an event, and turns the loop:
| Table | Filled by | When |
|---|---|---|
sessions |
db.session() open/close |
every run |
events |
every write method | always (telemetry is automatic) |
learnings |
learn() — preference rows are graduated |
always |
context |
unit(), checkpoint(), handoff(), verdict() |
always |
hypotheses |
a pattern crossing promote_at (default 3 hits) |
automatic |
experiments |
a verdict() on a unit/hypothesis under test |
automatic |
errors |
capture_error(), and any exception inside a session |
automatic |
The thresholds are tunable and were calibrated on 5,066 real learnings, not guessed: promote_at=3 (where the recurring-pattern tail actually begins), graduate_at=0.8 over a minimum of 3 experiments so a single lucky result can't graduate.
db = MetaBrain("agent.db", promote_at=3, graduate_at=0.8, min_experiments=3)
API
| Method | What it does |
|---|---|
session(*, task, tier, agent, meta) |
Open a session (context manager); records the outcome on close |
learn(type, insight, *, evidence, domain, ...) |
Record/reinforce a lesson; recurring patterns graduate to hypotheses |
recall(query, *, limit) |
Substring-search lessons; bumps hit count (can trigger graduation) |
learnings(*, type, domain, limit) |
Fetch lessons, newest first |
forget(id) |
Delete a lesson |
unit(statement, *, kind, acceptance, hypothesis) |
Open a unit of work; kind="spec" requires acceptance=[...] |
checkpoint(content, *, unit, agent) |
Record progress mid-work |
handoff(content, *, unit, agent) |
Record a brief for the next session |
verdict(result, *, unit, hypothesis, evidence) |
"pass"/"fail"; becomes an experiment when a hypothesis is in play |
hypotheses(*, status, limit) / experiments(*, hypothesis) |
Inspect the loop |
context(*, type, unit, limit) |
Fetch work-state entries |
read_start(*, learnings_limit) |
The "what to know" digest — proven preferences first |
capture_error(tool, error, ...) / errors(*, limit) |
Record / fetch failures |
prune(*, keep) / stats() |
Trim old checkpoints / row counts per table |
Use MetaBrain(":memory:") for an ephemeral in-process store (handy in tests).
Concurrency & safety
Built for multiple agents sharing one file. SQLite runs in WAL mode with a busy timeout so several processes read and write concurrently; within a process a single connection is lock-guarded, and the verdict→graduation path is one critical section so racing verdicts can never double-graduate a hypothesis. Every value is bound as a query parameter — caller strings never reach the SQL text.
It can open and migrate an older metabrain / base-schema database (learnings, context, errors) forward in place. A database created by a different tool whose events/hypotheses/experiments tables have an incompatible shape is detected on open and rejected with a clear IncompatibleDatabaseError, rather than corrupting it.
Development
pip install -e ".[dev]"
pytest
License
MIT © Aria Han
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 metabrain-1.0.0.tar.gz.
File metadata
- Download URL: metabrain-1.0.0.tar.gz
- Upload date:
- Size: 28.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d2b50393191711fb31f0e6da9b038c5a2278872eac55d5839a88d7d37d242bdc
|
|
| MD5 |
3cdd69b3d6c5f3b048c8bcc2888239b4
|
|
| BLAKE2b-256 |
64c32a902925d13852ec408bc0fef1954f7ad1bc235b7c0968e3101fd19318c8
|
File details
Details for the file metabrain-1.0.0-py3-none-any.whl.
File metadata
- Download URL: metabrain-1.0.0-py3-none-any.whl
- Upload date:
- Size: 21.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
4696e1bb5a62bced04fe8b21b301409f61ba0fc206e10e6ab2333a41a133c1ad
|
|
| MD5 |
204a03e127ca3c221245e0865e0744d6
|
|
| BLAKE2b-256 |
2a3db4d5f4de891f85b9261cee8f81961d7a4fe39ba1d22ea35907f79d5a6ec8
|