A local, persistent memory system for AI coding assistants — an MCP server with a real long-term brain.
Project description
🧠 Gingugu
Your AI forgets everything between sessions. Gingugu fixes that.
Gingugu is a local MCP server that gives AI coding assistants a real long-term brain — persistent, structured, searchable memory that survives across sessions, repos, and projects. No cloud, no API keys, no telemetry. One SQLite file on your machine.
📋 Table of Contents
- Why Gingugu
- Features
- Architecture
- Setup
- Memory Explorer UI
- Configuration
- Usage
- Development
- Troubleshooting
Why Gingugu
Every session with an AI assistant starts from zero. The decisions you made yesterday, the bug you fixed last week, the architecture you settled on a month ago — gone. Existing memory tools dump observations into a flat pile with no structure, no staleness tracking, no relationships, and no sense of what's relevant right now.
Gingugu is designed to be the actual brain — not a junk drawer:
- Remembers across sessions, repos, and projects
- Organizes knowledge by namespace, type, and relationships
- Ranks memories by relevance, freshness, and confidence
- Auto-surfaces relevant context when you start working
- Consolidates duplicate and related knowledge on demand
Where this goes long-term — federated, org-wide agent memory — lives in docs/enterprise-vision.md.
Features
| Feature | Description |
|---|---|
| 🏷️ Namespace Scoping | Memories auto-scoped to repos/projects with cross-repo pattern sharing |
| 🔍 Full-Text Search | SQLite FTS5 with BM25 ranking — fast, local, no API calls |
| ⏰ Temporal Intelligence | Decay scoring, staleness detection, "last confirmed" tracking |
| 🔗 Relationships | Link memories: supersedes, related_to, caused_by, contradicts |
| 🎯 Confidence Levels | verified → inferred → stale → deprecated lifecycle |
| 🧹 Consolidation Tools | Merge duplicates, summarize clusters, deduplicate on demand |
| 🚀 Auto-Context | Surfaces relevant memories on session start — zero manual effort |
| 📊 Health Metrics | Memory stats, staleness reports, namespace overviews |
| 🔐 Credential Vault | Secure service-bundle storage for API keys/tokens via OS Keychain |
| 🌐 Memory Explorer UI | Interactive knowledge graph + dashboard for visualizing memory data |
Architecture
graph TD
A[AI Assistant<br/>any MCP client] -->|MCP Protocol| B[Gingugu Server]
B --> C[Search Engine<br/>FTS5 + BM25]
B --> D[Storage Layer<br/>SQLite + WAL]
B --> E[Decay Engine<br/>Scoring + Pruning]
B --> F[Context Engine<br/>Auto-Retrieval]
B --> H[Consolidation Engine<br/>Merge + Dedupe]
B --> K[Credential Vault]
C --> D
E --> D
F --> D
H --> D
K --> D
K --> J[OS Keychain<br/>via keyring]
D --> G[(~/.local/share/gingugu/memories.db)]
See docs/architecture.md for full technical details.
Setup
Prerequisites
- Python 3.11+
uv(recommended) orpip- macOS, Linux, or Windows — the credential vault uses your OS-native secret
store via
keyring(macOS Keychain, Windows Credential Locker, Linux Secret Service/KWallet). On headless Linux without a Secret Service backend, everything works except storing secrets.
Install
# Clone
git clone https://github.com/gingugu/gingugu.git && cd gingugu
# Install with uv (recommended)
uv sync
# Or with pip
pip install -e .
✅ Phases 1–4 shipped — storage engine, FTS5 search, decay scoring, auto-context, the credential vault, the relationship graph (tags, relations, consolidation), and full namespace + export/import management are built and tested (112 passing). 16 MCP tools live. See
docs/roadmap.md.
Configure Your MCP Client
Gingugu speaks standard MCP over stdio —
it works with any MCP client. All of them boil down to the same thing:
run uv --directory /ABSOLUTE/PATH/TO/gingugu run gingugu.
🏄 Gingugu is built and dogfooded daily in Windsurf — this repo's own memories live in a Gingugu database — but Claude Code, Claude Desktop, Cursor, and Cline are first-class citizens too.
Windsurf
Add to ~/.codeium/windsurf/mcp_config.json — a ready-to-edit template lives
at examples/mcp_config.json:
{
"mcpServers": {
"gingugu": {
"command": "uv",
"args": ["--directory", "/ABSOLUTE/PATH/TO/gingugu", "run", "gingugu"]
}
}
}
⚠️ Windsurf's
mcp_config.jsonis global, not per-workspace, and it only interpolates${env:VAR}/${file:path}— not${workspaceFolder}. So a single server instance serves every repo.
Claude Code
claude mcp add gingugu -- uv --directory /ABSOLUTE/PATH/TO/gingugu run gingugu
Or add the standard mcpServers block (as in the Windsurf example) to
.mcp.json in your project root for a per-repo setup.
Claude Desktop
Add the same mcpServers block to
~/Library/Application Support/Claude/claude_desktop_config.json (macOS) or
%APPDATA%\Claude\claude_desktop_config.json (Windows).
Cursor
Add the same mcpServers block to ~/.cursor/mcp.json (global) or
.cursor/mcp.json in your repo (per-project).
Cline
Cline → MCP Servers → Configure: add the same mcpServers block to
cline_mcp_settings.json.
Anything else
Any client that supports stdio MCP servers works — point it at:
command: uv
args: ["--directory", "/ABSOLUTE/PATH/TO/gingugu", "run", "gingugu"]
Scoping memories per repo: when your client's config is global (it can't
see the active workspace), the assistant passes a namespace argument on each
memory tool call (every tool accepts one). To instead pin a server instance to
a single project, set a static MEMORY_NAMESPACE in the env block. See
docs/architecture.md → Namespace Auto-Detection for the full resolution
order.
Configure Your AI Agent
The MCP server gives your assistant the tools, but it won't use them effectively without instructions. Add the memory protocol below to your agent's rules file so it knows when and how to call them.
Which file? Depends on your IDE / tool:
| IDE / Tool | Rules File | Scope |
|---|---|---|
| Windsurf | .windsurfrules (repo root) |
Per-workspace |
| Cursor | .cursorrules (repo root) |
Per-workspace |
| Cline | .clinerules (repo root) |
Per-workspace |
| Codex / OpenAI | AGENTS.md (repo root) |
Per-repo |
| Any (global) | Your IDE's global rules/system prompt | All workspaces |
Paste this into your rules file (adjust the namespace and tool prefix to match your MCP config name):
## Memory Protocol
**Gingugu** is the persistent memory system for this workspace.
Use namespace `<your-project-name>` on every call.
### Session start
Run `memory_context` (with `namespace="<your-project-name>"` and an
optional `task_hint`) before doing anything else. This surfaces relevant
decisions, bugs, patterns, and state from previous sessions.
### During the session
**Default: save. Immediately.** Gingugu has decay scoring,
consolidation, and staleness detection - volume is its problem, not yours.
Save with `memory_store` whenever you:
- Make or observe a decision, trade-off, or architectural choice
- Hit an error or fix a bug (update the memory when resolved)
- Notice a pattern, convention, or constraint worth remembering
- See a config value, version, path, or credential name that matters
- Complete a task (what you did, why, and the outcome)
Use `memory_recall` before non-trivial work to check what's already known.
Use `memory_update` when something changes - don't leave stale records.
Use `memory_relate` to link connected memories (supersedes, related_to,
caused_by, contradicts, parent_of, child_of).
Set `confidence="verified"` when proven by a test or explicit confirmation.
Use `confidence="inferred"` for conclusions you drew.
### Memory types
- `fact` - concrete state (versions, paths, config values)
- `decision` - trade-offs made, rejected alternatives
- `architecture` - structural choices, module boundaries
- `bug` - issues found and how they were fixed
- `pattern` - recurring approaches worth reusing
- `workflow` - process steps, sequences
- `context` - background explaining *why* something is the way it is
- `preference` - observed working style and tool choices
Tip: A ready-to-use example lives at
.windsurfrulesin this repo (the project dogfoods itself). Copy the## Memory Protocolsection and adapt the namespace.
Memory Explorer UI
A React-based visualization dashboard lives in ui/ for exploring your memory
data interactively.
# Start the API server (reads live from your DB)
uv run python ui/api.py
# In another terminal, start the UI
cd ui && npm install && npm run dev
Open http://localhost:5173 - the UI connects to the API server and shows a green LIVE badge when pulling from your database. Features:
- Knowledge Graph - interactive force-directed graph of memories and relationships
- Dashboard - stats, charts by type/namespace/confidence, tag cloud, timeline
- Refresh - pull fresh data anytime; falls back to static sample when API is offline
Configuration
Environment variables (all optional):
| Variable | Default | Description |
|---|---|---|
MEMORY_DB_PATH |
~/.local/share/gingugu/memories.db (macOS/Linux) · %LOCALAPPDATA%\gingugu\memories.db (Windows) |
Database location |
MEMORY_NAMESPACE |
(unset) | Default namespace for this workspace (recommended per-MCP-entry) |
MEMORY_NAMESPACE_PATH |
(unset) | Alternative: filesystem path; namespace derived from basename |
MEMORY_AUTO_CONTEXT_LIMIT |
10 |
Max memories to surface on auto-context |
MEMORY_DECAY_LAMBDA |
0.05 |
Freshness decay rate in days⁻¹ (higher = faster forgetting) |
MEMORY_W_RELEVANCE |
0.45 |
Composite-score weight for FTS5 relevance |
MEMORY_W_FRESHNESS |
0.25 |
Composite-score weight for freshness |
MEMORY_W_ACCESS |
0.10 |
Composite-score weight for access frequency |
MEMORY_W_CONFIDENCE |
0.20 |
Composite-score weight for confidence |
MEMORY_LOG_LEVEL |
INFO |
Logging verbosity (logs go to stderr — stdout is the MCP transport) |
MEMORY_DEBUG |
false |
Convenience switch for DEBUG logging (MEMORY_LOG_LEVEL wins if also set) |
The four MEMORY_W_* weights are normalized at load (w_i / Σw), so they
need not sum to 1.0 — only their ratios matter. Setting all four to 0 falls
back to the defaults with a logged warning.
See docs/architecture.md → Decay Scoring Algorithm for how the weights combine.
Concurrency
The DB runs in WAL mode, which supports multiple concurrent processes:
any number of readers plus a single writer at a time. Running your IDE or
agent across several workspaces — each spawning its own gingugu process
against the shared DB — is fully supported. Writers serialize via SQLite's write lock and a
busy_timeout; transient DB locked errors under write contention are retried
automatically.
Usage
Once configured, the MCP server exposes these tools to your AI assistant:
| Tool | Purpose |
|---|---|
memory_store |
Save a new memory |
memory_recall |
Search + retrieve (ranked by relevance × freshness) |
memory_context |
Auto-surface relevant memories for current workspace |
memory_update |
Update content, confidence, or metadata |
memory_relate |
Create relationships between memories |
memory_consolidate |
Merge/summarize related memories |
memory_forget |
Deprecate or remove a memory |
memory_namespaces |
List/create/update/delete namespaces |
memory_export |
Export memories + tags + relations to portable JSON |
memory_import |
Restore a JSON export (skip or replace on conflict) |
memory_stats |
Health overview (staleness, counts, coverage); opt-in stale auto-flagging |
memory_search |
Advanced filtered search (type, tags, confidence, dates) |
credential_store |
Store/update a service credential bundle |
credential_get |
Retrieve credentials (secrets from OS Keychain) |
credential_list |
List services + expiry status (no secrets shown) |
credential_delete |
Remove a service or specific credential field |
Development
# Run tests
uv run pytest
# Run with verbose logging
MEMORY_LOG_LEVEL=DEBUG uv run gingugu
# Run specific test suite
uv run pytest tests/test_search.py -v
Troubleshooting
| Issue | Solution |
|---|---|
| DB locked | Expected under heavy concurrent writes — WAL mode supports multiple processes (many readers + one writer). The server retries with a busy_timeout; if it persists, a stuck process holds the write lock. See Concurrency above. |
| Slow search | Run memory_stats to check DB size; consolidate if bloated |
| Stale results | Use memory_update to confirm or deprecate old memories |
| Missing context | Check namespace — memories might be scoped to a different repo |
License
MIT — see LICENSE.
See CHANGELOG.md for release history.
A pirate never forgets where the treasure's buried. 🏴☠️
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 gingugu-0.1.0.tar.gz.
File metadata
- Download URL: gingugu-0.1.0.tar.gz
- Upload date:
- Size: 612.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.8.18
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d61c2c251b677ec010491aadefc35273d5e9dbf344bbb117f2ebbfb622f9f529
|
|
| MD5 |
b9b2cdba7fee97b80815ac96427c15ed
|
|
| BLAKE2b-256 |
4cd18c659659ac77cb9f0d8edf91db2344d19417e91c087f49eea6e7c83141a8
|
File details
Details for the file gingugu-0.1.0-py3-none-any.whl.
File metadata
- Download URL: gingugu-0.1.0-py3-none-any.whl
- Upload date:
- Size: 45.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.8.18
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
842e19eb1b91da6339f95cde08d0ecf723ad8d5d4075caad2b8af7c2128c1f3b
|
|
| MD5 |
11b34d5070773c8cd082b998eb63a871
|
|
| BLAKE2b-256 |
6194ff933ebf377454a89637de8809ba6c04f84419358775edad6fec9b434761
|