Skip to main content

Local-first semantic memory for AI agents — MLX (Apple Silicon) or CPU sentence-transformers (Linux/Ubuntu) embeddings + sqlite-vec, MCP server. No cloud, no API keys.

Project description

memo

memo

Local-first semantic memory for AI agents — with time-travel, contradiction radar, and automatic synthesis.

PyPI Python License: MIT MCP

memo gives any MCP-aware agent (Claude Code, Codex, Devin, OpenCode, Cursor, Cline, Continue, …) a long-term memory that runs entirely on your own machinemacOS on Apple Silicon via Apple MLX, or Linux / Ubuntu on a CPU sentence-transformers backend (pipx install "mlx-memo[cpu]", see docs/ubuntu.md). Each memory is a plain Markdown file; embeddings live in a single sqlite file; the embedder, reranker, and LLM run in-process — no Ollama, no Qdrant, no cloud API, no keys. Your prompts and memories never leave the machine.

Save a fact once and any later session recalls it automatically, all stored locally on your own machine.

What makes memo different

Capability memo mem0 letta cognee engram
100% local (no cloud API) ⚠️ ⚠️
Time-machine (rewind corpus to any date)
Contradiction radar (detect + resolve conflicts) ⚠️
Synthesis pipeline (auto-infer cross-cluster insights)
Cross-Mac git sync (shared corpus, no server)
Cloud sync (opt-in replication)
TUI (terminal UI)
Obsidian as source-of-truth
Knowledge graph + entity extraction ⚠️ ⚠️
Eval regression gate (pre-commit wireable)
Multi-modal (images, audio OCR) ⚠️
MCP surface profiles (token economy)
Passive capture (auto-extract from transcripts)
Session timeline (context before/after)

Why it pays for itself — in tokens

memo is built to spend fewer tokens, not more.

  • 92% smaller MCP surface. The default agent profile exposes 10 tools / ~1.2k schema tokens, versus 123 tools / ~15k tokens for the full surface — that overhead is paid every session, in every client. memo trims it to almost nothing.
  • Recall injects the answer instead of re-deriving it. Ambient recall surfaces the top memory before the agent answers, on a tight ~160-token budget. The agent stops re-explaining what it already figured out last week.

On a ~200-memory corpus, memo roi estimates ~80k tokens of model work avoided per session. The number is corpus-specific; it grows as memo learns more.

Technique How to enable Typical saving
Compact recall format export MEMO_RECALL_FORMAT=compact ~65% per injection
Trivial prompt gate On by default ~25% fewer injections
Context file compression memo compress-context CLAUDE.md 30–40% smaller context

Use cases

  • Continuity across sessions. Decide "we use Postgres, not Mongo" today; tomorrow, in a fresh session, the agent recalls it on its own — recall injects the decision before it answers, so you never re-explain it.
  • Shared memory across agents. Save something while working in Claude Code; Codex, Cursor, or Cline pick it up later. They all read the same local store over MCP.
  • Memory that follows you across Macs. Start on the laptop, continue on the desktop. The corpus travels over serverless git sync and the agent starts with the same context on both.
  • Preferences and conventions that stick. "Tests first", "commit messages in English", "don't touch the auth module" — say it once, the agent applies it every future session.
  • Contradiction radar. Change your mind on an old decision and memo flags the now-stale version — the agent won't reintroduce what you already discarded.
  • Time-machine / audit. "What did we know about this bug last month?" Rewind the corpus to any date and see the state of knowledge at that point.
  • Instant project onboarding. A cold agent gets the project's durable decisions, facts, and preferences up front via the session-start briefing.
  • Fewer tokens, not more. Instead of re-deriving what you solved last week, recall injects the answer on a tight budget — and the default MCP surface is ~10 tools, not ~120.

Requirements

  • macOS on Apple Silicon (M1–M4) — MLX is the load-bearing piece and the only path with the reranker + LLM features (ask / synthesize / dream).
  • Linux / Ubuntu / Intel Mac — supported as a standalone install via a CPU sentence-transformers backend (search + recall + save, no MLX). One command: pipx install "mlx-memo[cpu]". See docs/ubuntu.md for what works and the trade-offs.
  • ~8 GB free disk for the default model set (the installer downloads it).
  • Optional: an Obsidian vault. Without one, memo defaults to ~/Documents/memo/.

Python ≥ 3.13 is required if you install without uv. The curl | bash installer handles this automatically — it detects uv and uses its managed Python if no system Python ≥ 3.13 is on PATH.

Install — one step

curl -fsSL https://raw.githubusercontent.com/jagoff/memo/master/install.sh | bash

The installer auto-detects uv (preferred) or falls back to pipx. It downloads MLX models, and wires memo into every agent client it finds (Claude Code, Codex, Devin, OpenCode, Windsurf).

Prefer a manual install? Any of these expose the same two binaries — memo (CLI) and memo-mcp (MCP server):

uv tool install mlx-memo          # recommended
pipx install mlx-memo
brew tap jagoff/memo && brew install mlx-memo

Keep memo isolated as its own tool (uv tool / pipx / Homebrew). Don't vendor it inside another project's .venv. memo doctor --strict-runtime verifies the install.

First install downloads ~8 GB of MLX models (5–15 min); later installs hit the HuggingFace cache. Full installer knobs and "move to a new Mac" steps: docs/reference.md › Install.

Migrating from another Mac? Install first, then restore your corpus:

curl -fsSL https://raw.githubusercontent.com/jagoff/memo/master/install.sh | bash
memo sync bootstrap git@github.com:yourname/memo-sync.git   # restore from git

Hand it to your agent

memo installs itself if you hand the repo (or just the install line) to an AI agent:

curl -fsSL https://raw.githubusercontent.com/jagoff/memo/master/install.sh | bash
memo doctor --strict-runtime     # verify runtime is healthy

After install, tools surface as mcp__memo__memo_* (memo_save, memo_search, memo_ask, memo_get, memo_unified_briefing). Per-client setup (Claude Desktop, Cursor, Cline, Continue, manual JSON) is in docs/reference.md › MCP setup.

Quick start

memo doctor                                            # self-check: models, vault, sqlite-vec
memo save 'MLX prefill ~30% faster than Ollama on M3 Max' --title 'MLX bench' -t mlx -t bench
memo search 'how fast was the MLX benchmark'           # search by meaning, not just keywords
memo list --limit 5                                    # most recent
memo ask 'what changed in the embedder this month?'   # RAG — cites memories by id

Core features

  • Ambient recall — every prompt silently consults memory and injects top hits as context. Warm recall daemon keeps it under <200 ms. No /remember calls.
  • Auto-capture — a Stop hook extracts durable insights from each exchange through a quality gate. The corpus grows on its own.
  • Session briefingSessionStart surfaces open loops, a memory of the day, and one-line crash recovery.

Key capabilities

🕰️ Time-machine

Rewind the corpus to any past date and query it as it was then:

memo as-of ask "what was the deployment strategy?" --date 2026-02-01
memo as-of search "redis config" --date 2026-01-15
memo diff --from 2026-01-01 --to 2026-03-01    # what changed

No other agent-memory system offers this. Full historical reconstruction via reverse-replay of history.db.

⚡ Contradiction radar

memo contradict scan                  # detect conflicting facts corpus-wide
memo contradict triage                # resolve interactively: fuse / newer-wins / dismiss

The LLM classifies each candidate pair. Results persist in contradictions.db; resolved conflicts inform future saves.

🔮 Synthesis pipeline

memo synthesize                       # generate cross-cluster insights (LLM)
memo dream                            # nightly: signal gather → prune → orient

MEMO_SYNTHESIS_ENABLED=1 runs synthesis automatically during memo maintain.

🌐 Cross-Mac git sync

memo sync bootstrap git@github.com:yourname/memo-sync.git   # wire a shared corpus
memo sync once                                                # push/pull now

Pull-rebase-before-push. flock-based single owner per machine. Async debounced hooks keep the corpus current without blocking.

📚 Obsidian vault as source-of-truth

MEMO_MEMORIES_IN_VAULT=1 memo init                # store memories inside your vault
memo migrate --into-vault                          # non-destructive migration

Human edits in Obsidian win on the next memo reindex. The sqlite index is always rebuildable from the .md files.

🕸️ Knowledge graph

memo graph neighbors "MLX"             # what's related
memo graph path "embedder" "reranker"  # how two concepts connect
memo entities                          # list extracted entities
memo links --id abc123                 # backlinks + outlinks

Entity extraction uses a dependency-free regex backend. For code-heavy corpora, memo can merge a codegraph symbol graph as the graph's primary layer (opt-in, MEMO_GRAPH_USE_CODEGRAPH) — callers, callees, and imports become first-class edges, so recall and memo graph path reason over real code structure, not just text similarity. The merged graph also powers the memo_graph MCP tool and the entity-centric "Knowledge map" briefing.

🏥 Health scoring & eval gates

memo health                                         # grounded rate, ROI, usefulness verdict
memo eval recall --labels eval/regression_labels.json --k 5
memo eval recall --gate                             # exit non-zero if precision drops
memo eval recall --update-baseline                  # snapshot current best

Wire --gate into a pre-commit hook to catch retrieval regressions before they ship.

🖼️ Multi-modal ingestion

memo ocr-image screenshot.png               # macOS Vision OCR
memo multimodal add-image photo.jpg --title "whiteboard"
memo search "whiteboard diagram"            # finds it

Daemons

memo runs four background daemons:

Daemon Command Purpose
recall-daemon memo recall-daemon start Warm MLX embedder over socket (<200 ms recall)
idle-daemon auto-started by memo-mcp Auto-capture for MCP-only clients (Devin, OpenCode)
ingest-daemon memo ingest-daemon start Bulk vault ingestion
maint-daemon memo maint-daemon start Background cleanup + synthesis

All 95 CLI commands

Click to expand

Core: save search ask get edit delete list

Recall & Hooks: recall recall-hook briefing continuity prewarm capture-tick capture-stop

Session & History: history as-of diff record-history session resume reflect mine-history

Maintenance: reindex maintain dream consolidate synthesize dedupe cross-dedup retier contradict temporal

Analysis & Quality: health stats doctor lint analytics eval roi token-savings usefulness gaps outcome profile

Knowledge Graph: graph entities entity extract-entities links version

Advanced Search: embed rerank contextual chat chat-ask multimodal repo

Import / Export / Sync: import export backup restore sync ingest

Visualization: tui dashboard map logs hook-log

Setup & Config: init config install-mcp install-watcher uninstall-watcher install-slash install-statusline install-shell-wrapper install-shims startup-banner migrate migrate-vault update watch mcp-command

Daemons: recall-daemon ingest-daemon maint-daemon embed-daemon

MCP surface profiles

Profile Tools Schema tokens Use when
agent (default) 10 ~1.2k Standard agent work — max token economy
core 30 ~2.8k Constrained clients (Codex, OpenCode)
full 123 ~15k Power users, debugging

Set via MEMO_MCP_PROFILE=full or in each client's MCP env config.

Retrieval architecture

Hybrid search: vec leg (MLX embedding on Apple Silicon, CPU sentence-transformers on Linux) + BM25 leg (FTS5/Tantivy, diacritic-folding for Spanish) fused via Reciprocal Rank Fusion → optional MLX cross-encoder rerank (Apple Silicon).

A prompt runs a vector search and a keyword search in parallel; results are fused, reranked, and the top memory is injected.

Markdown is the source of truth. The .md files are canonical; sqlite is a rebuildable index. A hand-edit in Obsidian wins on the next memo reindex. delete() removes the index first, then the file — no silent data loss.

Markdown files are canonical; the sqlite index is derived and can be rebuilt from them at any time.

Embedding models:

Model Dims Disk Use
Qwen3-Embedding-0.6B-4bit 1024 ~0.4 GB Default (fast, good)
Qwen3-Embedding-4B-4bit 2560 ~2.5 GB Higher recall quality
Qwen3-Embedding-8B-4bit 4096 ~5 GB Maximum quality

Switch with MEMO_EMBEDDER_MODEL + MEMO_EMBEDDER_DIMS (requires memo reindex --rebuild).

Documentation

Topic Where
Full install detail, installer knobs, new-Mac migration docs/reference.md › Install
Per-client MCP setup + the /memo slash command docs/reference.md › MCP setup
All MCP tools reference docs/reference.md › MCP tools
Ambient memory, recall daemon, capture & recall tuning docs/reference.md › Ambient memory
Time-machine, session briefing, semantic map docs/reference.md › Surfaces
Full CLI reference + live dashboard (memo tui) docs/reference.md › CLI
All MEMO_* flags, model profiles, upgrading the embedder docs/reference.md › Configuration
Architecture, sync tiers, design notes docs/reference.md › Design & comparison

Contributors: git clone https://github.com/jagoff/memo && cd memo && uv pip install -e '.[dev]'. See CONTRIBUTING.md.

License & provenance

MIT — see LICENSE. Forked philosophically from mem-vault (storage layout + frontmatter schema); the MLX backend pieces are ported from obsidian-rag. memo is one of three sovereign systems in a wider stack (Memflow, Synapse) — the integration is opt-in everywhere; single-Mac users see zero behaviour change.

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

mlx_memo-2.6.4.tar.gz (1.4 MB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

mlx_memo-2.6.4-py3-none-any.whl (838.0 kB view details)

Uploaded Python 3

File details

Details for the file mlx_memo-2.6.4.tar.gz.

File metadata

  • Download URL: mlx_memo-2.6.4.tar.gz
  • Upload date:
  • Size: 1.4 MB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for mlx_memo-2.6.4.tar.gz
Algorithm Hash digest
SHA256 a3fd8e6f93500b8bf6291cc8248bf53720c098bc0a58ff06e84964b83622b0d4
MD5 261b7374fb24778af5f4251dcee712fa
BLAKE2b-256 c5d94beff2e1f96c9408fba18cf076ebef179d6a011d1de20d9c241fafd9312b

See more details on using hashes here.

Provenance

The following attestation bundles were made for mlx_memo-2.6.4.tar.gz:

Publisher: publish.yml on jagoff/memo

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file mlx_memo-2.6.4-py3-none-any.whl.

File metadata

  • Download URL: mlx_memo-2.6.4-py3-none-any.whl
  • Upload date:
  • Size: 838.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for mlx_memo-2.6.4-py3-none-any.whl
Algorithm Hash digest
SHA256 606ea47ba76abf10e119f71297f47b8056aec28aab4fbfc650b40f9a6abf4bce
MD5 5702f5b59c1253e7bc7a0aa612277c67
BLAKE2b-256 ef794ac1bc7bb59cdf4bf3d503da9aa5b6f7ed65f0d4034572eba343d831dd78

See more details on using hashes here.

Provenance

The following attestation bundles were made for mlx_memo-2.6.4-py3-none-any.whl:

Publisher: publish.yml on jagoff/memo

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page