Perfect memory. Exact reasoning. Complete history. Bi-temporal graph memory for AI coding agents.
Project description
Temporal Reasoning
Perfect memory. Exact reasoning. Complete history.
Temporal Reasoning gives AI coding agents bi-temporal graph memory: query any past state, traverse live dependency graphs, and correlate architectural decisions with structural change — all with deterministic Datalog, no fuzzy retrieval.
Questions Only Temporal Reasoning Can Answer
These queries are impossible with git log, vector search, or key-value memory:
; What did the dependency graph look like before the auth refactor?
[:find ?caller ?callee
:as-of 30
:where [?caller :calls ?callee]]
; When did this coupling first appear — and what decision caused it?
[:find ?reason
:where [:project/service-a :depends-on :project/service-b]
[?d :motivated-by ?c]
[?c :description ?reason]]
; Which modules were coupled to the payment service when we made the DB decision?
[:find ?module
:as-of 15
:where [?module :depends-on :service/payment]]
This is the only tool where both the decision and the structural change live as datoms in the same graph and can be joined in a single query. See Phase 5 for code structure evolution from git history.
Why Temporal Reasoning?
Most memory tools for agents are key-value stores or vector databases. They answer "what do you know now?" Temporal Reasoning answers a harder question: "what did you know then, and what changed?"
Time travel. Every write is stamped with a transaction number. You can query the graph as it existed at any past transaction:
# Decision made in session 1, transaction 3
transact('[[:project/db :name "PostgreSQL"]]', reason="Initial choice")
# Changed in session 4, transaction 11
retract('[[:project/db :name "PostgreSQL"]]', reason="Switching to CockroachDB for geo-distribution")
transact('[[:project/db :name "CockroachDB"]]', reason="Switching to CockroachDB for geo-distribution")
# Later: what did we think the database was before session 4?
query("[:find ?name :as-of 10 :where [:project/db :name ?name]]")
# → "PostgreSQL"
# What do we think now?
query("[:find ?name :where [:project/db :name ?name]]")
# → "CockroachDB"
Retraction with preserved history. Changing your mind doesn't erase the record. Retracted facts stay in the bi-temporal log and remain queryable at their original transaction time. This means the agent can always reconstruct why a decision changed, not just what the current state is.
Exact Datalog queries, not fuzzy search. Results are deterministic and reproducible — no embedding model, no similarity threshold, no hallucinated retrievals. A query either matches or it doesn't.
Graph traversal. Entities are first-class nodes — not isolated key-value blobs. Store service-calls-service as a real graph edge (:calls :project/auth-service) and traverse it with Datalog joins. Fixed-depth transitive queries (2-hop, 3-hop) are expressed as multi-hop joins. Rules unify multiple edge types under a single named relation.
Local and offline. A single binary and a file. No API key, no network dependency, no cloud service to go down.
Architecture
┌──────────────────────────────────────────────────────────────────┐
│ AI Coding Agent │
│ (Claude Code, OpenCode, Codex) │
└──────────┬───────────────────────────────────────┬───────────────┘
│ MCP tool calls │ per-turn hooks
│ (minigraf_query, minigraf_transact, …) │ (UserPromptSubmit / Stop)
▼ ▼
┌──────────────────────────┐ ┌─────────────────────────────┐
│ MCP Server │ │ Hook scripts │
│ mcp_server.py │◄────────│ prepare_hook.py │
│ (persistent stdio) │ │ finalize_hook.py │
└──────────┬───────────────┘ └─────────────────────────────┘
│
▼
┌──────────────────────────────────────────────────────────────────┐
│ MiniGrafDb Python binding (minigraf package) │
│ https://github.com/project-minigraf/minigraf │
│ - Bi-temporal Datalog engine │
│ - Transaction time + Valid time │
└──────────┬───────────────────────────────────────────────────────┘
│
▼
┌──────────────────────────────────────────────────────────────────┐
│ Graph File │
│ memory.graph (current working directory) │
└──────────────────────────────────────────────────────────────────┘
Install
git clone https://github.com/project-minigraf/temporal_reasoning
cd /your/project
python /path/to/temporal_reasoning/install.py
Run install.py from your project root. It creates a virtualenv, installs dependencies, and writes .mcp.json and .claude/settings*.json into your project directory. That's it.
Optional — LLM extraction strategy: install.py defaults to heuristic (regex) extraction, which requires no API key. To use LLM-based extraction, set MINIGRAF_EXTRACTION_STRATEGY=llm and ANTHROPIC_API_KEY=<your key> in .claude/settings.local.json after running the script.
OpenCode
python /path/to/temporal_reasoning/install.py
This also syncs the skill into .opencode/skills/temporal-reasoning.
Quick Start
from minigraf import query, transact
# Store a decision
transact("[[:decision/cache-strategy :decision/description \"use Redis\"]]",
reason="Architecture decision for low-latency caching")
# Query decisions
result = query("[:find ?d :where [?e :decision/description ?d]]")
Storage Location
Default: memory.graph in the current working directory.
Override: MINIGRAF_GRAPH_PATH=/custom/path python ...
Per-Turn Auto-Memory
When running under Claude Code with the hook configuration in hooks/claude-code.json, the system automatically injects relevant memory context before each turn and extracts durable facts after each turn — without the agent explicitly calling any tool.
Prepare phase (before the turn)
prepare_hook.py fires on the UserPromptSubmit event. It:
- Extracts candidate entity tokens from the user's message (stop-word filtered, minimum 4 characters).
- Queries the graph for facts whose values contain those tokens, using
:valid-atset to the current UTC timestamp so only currently-valid facts are returned. - Falls back to a broad scan (capped by
MINIGRAF_PREPARE_SCAN_LIMIT, default 50 rows) when no entity-specific results are found. - Returns the results as
additionalContextprepended to the agent's working context for that turn.
For messages containing temporal signals (e.g. "before", "last week", "as of") with an explicit ISO date, :valid-at is set to that date instead (midnight UTC), enabling point-in-time recall.
Finalize phase (after the turn)
finalize_hook.py fires on the Stop event. It reads the last user+assistant exchange from the transcript, then runs the configured extraction strategy:
| Strategy | Behaviour |
|---|---|
heuristic (default) |
Regex patterns detect decision-signal phrases ("we'll use X", "decided to use X", "always use X", "depends on X", …) and transact the matched tokens as :decision/, :preference/, :constraint/, or :dependency/ entities. |
llm |
Sends the exchange to a lightweight Claude model (claude-haiku-4-5-20251001 by default) with a structured prompt. The model returns a Datalog transact expression; an optional ; valid-at: YYYY-MM-DD comment sets the fact's valid time. Falls back to the agent strategy on error. |
agent |
Uses MCP sampling to ask the connected agent itself for a memory block in the same Datalog format. |
Configuration
| Environment variable | Default | Effect |
|---|---|---|
MINIGRAF_EXTRACTION_STRATEGY |
heuristic |
Finalize strategy: heuristic, llm, or agent |
MINIGRAF_PREPARE_SCAN_LIMIT |
50 |
Max rows returned by the broad fallback scan in the prepare phase |
MINIGRAF_LLM_MODEL |
claude-haiku-4-5-20251001 |
Model used when MINIGRAF_EXTRACTION_STRATEGY=llm |
ANTHROPIC_API_KEY |
— | Required when MINIGRAF_EXTRACTION_STRATEGY=llm and using a Claude model |
OPENAI_API_KEY |
— | Required when MINIGRAF_EXTRACTION_STRATEGY=llm and MINIGRAF_LLM_MODEL is an OpenAI model (e.g. gpt-4o-mini) |
MINIGRAF_GRAPH_PATH |
memory.graph |
Override the graph file location |
Files
| File | Purpose |
|---|---|
mcp_server.py |
Persistent stdio MCP server — primary interface to the graph |
minigraf.py |
Python CLI wrapper (direct use outside MCP) |
hooks/prepare_hook.py |
Claude Code UserPromptSubmit hook — injects memory context |
hooks/ingest_hook.py |
Claude Code UserPromptSubmit hook — triggers background git ingestion |
hooks/finalize_hook.py |
Claude Code Stop hook — extracts and stores facts |
hooks/claude-code.json |
Hook + MCP configuration for Claude Code |
report_issue.py |
GitHub issue reporter |
install.py |
Setup script |
pyproject.toml |
Python packaging |
tools/*.json |
Tool schemas |
Tools
- minigraf_query — Query memory with Datalog
- minigraf_transact — Store facts (reason required)
- minigraf_retract — Retract facts (original stays in history)
- minigraf_report_issue — File GitHub issues
- memory_prepare_turn — Retrieve relevant context for the current user message
- memory_finalize_turn — Extract and store memorable facts after a turn
- minigraf_audit — Audit all entities against the schema; retracts violators (history preserved)
- minigraf_ingest_git — Ingest code structure from git history into the bi-temporal graph (background task)
- minigraf_ingest_status — Poll progress of a running git ingestion; reports wall-clock time and final commit hash of the last completed run (including hook-driven ingestion)
Query Examples
# Basic query
query("[:find ?x :where [?e :attr ?x]]")
# Temporal query (state at transaction N)
query("[:find ?x :as-of 5 :where [?e :attr ?x]]")
# Aggregation
query("[:find (count ?e) :where [?e :decision/description ?d]]")
# Single-hop graph traversal — what does api-gateway call?
query("[:find ?name :where [:project/api-gateway :calls ?svc] [?svc :name ?name]]")
# Two-hop join — transitive impact: what depends on key-store (directly or via one intermediate)?
query("""[:find ?svc
:where [?mid :depends-on :project/key-store]
[?svc :depends-on ?mid]]""")
# Decision traceability — why did we choose asyncio?
query("[:find ?reason :where [:decision/asyncio-choice :motivated-by ?c] [?c :description ?reason]]")
# Typed entity query — list all stored components
query("[:find ?name :where [?e :entity-type :type/component] [?e :name ?name]]")
Skill Benchmarks
Twelve evals run in isolated sandboxes measure how the skill changes behavior versus a no-skill baseline. Each eval uses a fresh graph with pre-seeded state where relevant.
| Eval | What it tests | With Skill | Without Skill |
|---|---|---|---|
| 1 — Decision storage | Persists architectural decisions with correct naming + reasons | 5/5 | 0/5 |
| 2 — Memory retrieval | Queries memory and cites stored facts by name | 4/5 | 3/5 |
| 3 — Cross-session preference | Discovers and applies a constraint never stated in the current conversation | 4/4 | 0/4 |
| 4 — Conflict detection | Surfaces architectural conflicts before silently overriding decisions | 4/4 | 0/4 |
| 5 — Entity reference storage | Stores relationships as traversable graph edges, not dead-end strings | 5/5 | 0/5 |
| 6 — Transitive impact analysis | Traverses a multi-hop dependency chain to find all affected services | 5/5 | 4/5 |
| 7 — Decision traceability | Follows a :motivated-by edge to surface the constraint behind a decision |
5/5 | 1/5 |
| 8 — Git ingestion | Checks status before starting ingestion; moves on without polling | 6/6 | 0/6 |
| 9 — Ingest status | Reports idle/running/complete accurately; surfaces errors | 5/5 | 0/5 |
| 10 — Memory prepare-turn | Injects relevant context before the agent responds | 5/5 | 0/5 |
| 11 — Audit | Detects and retracts schema violations | 4/5 | 0/5 |
| 12 — Already running | Does not re-trigger ingestion when already in progress | 4/5 | 2/5 |
| Total | 56/59 (95%) | 10/59 (17%) |
The cross-session preference eval is the most discriminating for memory recall: the prompt says "make sure it fits with how we do things" with no hint that a relevant constraint exists. The skill queries memory, finds a stored no-mocks preference, and writes a test using real database connections.
The transitive impact eval is the most discriminating for graph traversal: given "key-store is being replaced — what breaks?" the skill executes a 2-hop Datalog join and returns a full impact chain; without it, the agent correctly admits it cannot name the affected services.
See evals/benchmark.md for full results and per-eval breakdowns.
Phases
- Phase 1 — Python skill layer ✓
- Phase 2 — Write policy, report_issue, install, skill benchmarks ✓
- Phase 3 — MCP server, per-turn auto-memory hooks ✓
- Phase 4 — Entity normalization, schema-aware extraction, minigraf_audit ✓
- Phase 5 — Code structure ingestion from git history, minigraf_ingest_git ✓
- Phase 6 — Observability and trust for automatic memory (planned)
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 temporal_reasoning-0.4.0.tar.gz.
File metadata
- Download URL: temporal_reasoning-0.4.0.tar.gz
- Upload date:
- Size: 74.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
3b3951abc6cba585e79c335307a9c23d65f539df35990829953557565dd267db
|
|
| MD5 |
1e690e488d0473faa2bfb469a3836c67
|
|
| BLAKE2b-256 |
21f698361d5bfdcc4c7913fab317287e2e56d5708fc82b7715294d969563a93a
|
Provenance
The following attestation bundles were made for temporal_reasoning-0.4.0.tar.gz:
Publisher:
release.yml on project-minigraf/temporal_reasoning
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
temporal_reasoning-0.4.0.tar.gz -
Subject digest:
3b3951abc6cba585e79c335307a9c23d65f539df35990829953557565dd267db - Sigstore transparency entry: 2065772348
- Sigstore integration time:
-
Permalink:
project-minigraf/temporal_reasoning@87cbcc1a3f9c57a44c7ba9c21533398c81f5bc78 -
Branch / Tag:
refs/tags/v0.4.0 - Owner: https://github.com/project-minigraf
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@87cbcc1a3f9c57a44c7ba9c21533398c81f5bc78 -
Trigger Event:
push
-
Statement type:
File details
Details for the file temporal_reasoning-0.4.0-py3-none-any.whl.
File metadata
- Download URL: temporal_reasoning-0.4.0-py3-none-any.whl
- Upload date:
- Size: 43.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c913fba5a9feba70a512930473d5dea304100d50959315c9e10b29b83cb61ef1
|
|
| MD5 |
8ef8b4f010905b6683986b0023dc6bcc
|
|
| BLAKE2b-256 |
4e2a9ea0d2d2f19f8bae6cd55280ccea5ded222415216863aa112f8aa83b7146
|
Provenance
The following attestation bundles were made for temporal_reasoning-0.4.0-py3-none-any.whl:
Publisher:
release.yml on project-minigraf/temporal_reasoning
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
temporal_reasoning-0.4.0-py3-none-any.whl -
Subject digest:
c913fba5a9feba70a512930473d5dea304100d50959315c9e10b29b83cb61ef1 - Sigstore transparency entry: 2065772439
- Sigstore integration time:
-
Permalink:
project-minigraf/temporal_reasoning@87cbcc1a3f9c57a44c7ba9c21533398c81f5bc78 -
Branch / Tag:
refs/tags/v0.4.0 - Owner: https://github.com/project-minigraf
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@87cbcc1a3f9c57a44c7ba9c21533398c81f5bc78 -
Trigger Event:
push
-
Statement type: