engram
Enables fully local LLM and embedding support using models like llama3.2 and nomic-embed-text, allowing offline operation of the memory system.
Provides LLM and embedding capabilities for extracting facts from text, synthesizing observations, and evaluating evidence in the memory system.
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@engramRemember that Alice works at Acme Corp."
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
engram
A learning agent memory system with reflect loop. Engram stores facts extracted from text, links them to entities and relationships, and periodically synthesizes higher-order observations (patterns, preferences, insights).
Engram is two things:
A TypeScript library — import
Engramand callretain(),recall(),reflect()directly.An MCP server — run
engram-mcpand connect any MCP client (Claude Desktop, Claude Code, etc.) over stdio.
Both use the same SQLite-backed engine. No external database required.
How It Works
text
|
v
+---------+---------+
| retain() | Extract facts, entities, relations via LLM
| (or retainDirect)| Store with embeddings + FTS index
+---------+---------+
|
v
+---------+---------+
| recall() | Hybrid search: semantic + keyword + temporal
| | Rank fusion with configurable weights
+---------+---------+
|
v
+---------+---------+
| reflect() | Cluster facts by shared entities
| | Synthesize observations (patterns/preferences/insights)
+---------+---------+retain — Ingests text, extracts atomic facts and entities via an LLM, embeds them, and stores everything in SQLite. Duplicate facts are skipped via content-hash deduplication. Optionally evaluates new facts against existing observations to adjust confidence.
retainDirect — Same as retain but accepts pre-extracted facts/entities/relations, skipping the LLM extraction call entirely. Useful when you already have structured data.
recall — Hybrid retrieval combining semantic similarity (sqlite-vec), keyword matching (FTS5), and temporal filtering (chrono-node). Scores are fused with configurable weights and observations get a priority boost.
reflect — Groups facts by shared entities into clusters, then synthesizes higher-order observations using an LLM. Duplicate observations merge evidence and bump confidence. Low-confidence observations are archived.
forget — Removes facts by ID, age, or TTL expiry. FTS5 and entity links are cleaned up automatically.
Quick Start
Programmatic (TypeScript library)
import { Engram, createProviders } from 'engram';
const { embedding, llm } = await createProviders({ provider: 'openai' });
const engram = new Engram(
{ dbPath: './memory.sqlite' },
embedding,
llm,
);
// Store information (LLM extracts facts automatically)
await engram.retain({
text: 'Alice prefers TypeScript over JavaScript. She works at Acme Corp.',
source: 'conversation',
});
// Store pre-extracted facts (no LLM call)
await engram.retainDirect({
facts: [{ content: 'Bob uses vim keybindings' }],
entities: [{ name: 'Bob', entityType: 'person' }],
});
// Search memory
const results = await engram.recall({
query: 'What does Alice prefer?',
limit: 5,
});
// Synthesize observations
await engram.reflect({ force: true });
// Remove a fact
engram.forget({ factId: 'some-fact-id' });
// Remove old facts
engram.forget({ olderThan: 30 * 24 * 60 * 60 * 1000 }); // 30 days
// List all memory banks
const banks = engram.listBanks();
engram.close();MCP Server
# Set your API key
export OPENAI_API_KEY=sk-...
# Start the MCP server
npx engram-mcpAdd to your MCP client configuration (e.g. Claude Desktop):
{
"mcpServers": {
"engram": {
"command": "npx",
"args": ["engram-mcp"],
"env": {
"OPENAI_API_KEY": "sk-..."
}
}
}
}MCP Tools Reference
Tool | Description | Key Parameters |
| Store text, extracting facts via LLM |
|
| Store pre-extracted facts (no LLM) |
|
| Search memory with hybrid retrieval |
|
| Synthesize observations from facts |
|
| View stats, look up entities, or list banks |
|
| Remove facts from memory |
|
Concepts
Banks
Memory banks provide isolated namespaces. Every fact, entity, and observation is scoped to a bank. Use banks to separate contexts (e.g. "work" vs "personal", or per-user).
engram.createBank('work', 'Work', 'Professional context');
await engram.retain({ text: '...', bankId: 'work' });
const banks = engram.listBanks(); // discover all banksFacts vs Observations
Facts are atomic pieces of information extracted from input text. Each has a content hash for deduplication, a confidence score, and optional entity links.
Observations are higher-order insights synthesized by the reflect loop. Types:
pattern,preference,insight. They track confidence that adjusts as new evidence arrives.
Confidence
Facts start at confidence 1.0.
Observations start at 0.5 and adjust based on supporting/contradicting evidence.
Observations below 0.1 confidence are archived during reflect.
Entity Resolution
Entities are deduplicated by (name, type, bank). When the same entity name appears again, metadata is merged rather than creating a duplicate.
Configuration
interface EngramConfig {
dbPath: string; // SQLite database path (default: ':memory:')
defaultBankId?: string; // Default bank ID (default: 'default')
defaultBankName?: string; // Default bank name (default: 'Default')
reflectThreshold?: number; // Facts before auto-reflect triggers (default: 20)
reflectInterval?: number; // Min ms between reflects (default: 86400000 / 24h)
autoReflect?: boolean; // Enable auto-reflect after retain (default: true)
skipEvidenceEvaluation?: boolean; // Skip LLM evidence eval on retain (default: false)
reflectMaxClusters?: number; // Max clusters to process per reflect (default: 0 = unlimited)
}Environment Variables
Variable | Description | Default |
| SQLite database file path |
|
| AI provider: |
|
| Override embedding model name | Provider default |
| Override language model name | Provider default |
| Default bank ID |
|
| Default bank name |
|
| Set to |
|
| Set to |
|
| Max clusters per reflect cycle | unlimited |
Provider Support
OpenAI (default)
export OPENAI_API_KEY=sk-...
export ENGRAM_PROVIDER=openaiEmbedding:
text-embedding-3-small(1536 dimensions)Language:
gpt-4o-mini
Anthropic
export OPENAI_API_KEY=sk-... # Still needed for embeddings
export ANTHROPIC_API_KEY=sk-ant-...
export ENGRAM_PROVIDER=anthropicEmbedding: OpenAI
text-embedding-3-small(1536 dimensions)Language:
claude-sonnet-4-5-20250929
Ollama (fully local, no API key)
Run everything locally with no API keys or cloud calls:
ollama pull nomic-embed-text
ollama pull llama3.2
ENGRAM_PROVIDER=ollama npm run mcpEmbedding:
nomic-embed-text(768 dimensions)Language:
llama3.2
Custom Providers
Implement the EmbeddingProvider and LLMProvider interfaces:
import type { EmbeddingProvider, LLMProvider } from 'engram';
class MyEmbedding implements EmbeddingProvider {
readonly dimensions = 1536;
async embed(text: string): Promise<Float32Array> { /* ... */ }
async embedMany(texts: string[]): Promise<Float32Array[]> { /* ... */ }
}
class MyLLM implements LLMProvider {
async extractFacts(text: string, bankContext?: string) { /* ... */ }
async synthesizeObservations(facts, existing, bankContext?) { /* ... */ }
async evaluateEvidence(fact, observation) { /* ... */ }
}
const engram = new Engram(config, new MyEmbedding(), new MyLLM());Cost Controls
The reflect loop and evidence evaluation make LLM calls that can add up. Two config options help control costs:
skipEvidenceEvaluation: true — Skips the fire-and-forget LLM call that evaluates each new fact against every active observation. This removes O(facts * observations) LLM calls per retain. Observation confidence will only change during reflect.
reflectMaxClusters: N — Limits reflect to the N largest entity clusters (by fact count). This caps the number of synthesis LLM calls per reflect cycle.
const engram = new Engram({
dbPath: './memory.sqlite',
skipEvidenceEvaluation: true, // No evidence eval LLM calls
reflectMaxClusters: 5, // Max 5 clusters per reflect
}, embedding, llm);Multi-Context via Banks
Banks are isolated memory namespaces. Use them for multi-user, multi-project, or multi-domain scenarios:
// Create isolated banks
engram.createBank('user-alice', 'Alice', 'Alice personal memory');
engram.createBank('user-bob', 'Bob', 'Bob personal memory');
// Store to specific banks
await engram.retain({ text: '...', bankId: 'user-alice' });
// Search within a bank
await engram.recall({ query: '...', bankId: 'user-alice' });
// Discover all banks
const banks = engram.listBanks();TTL and Forgetting
Facts can be given a time-to-live (TTL) so they expire automatically:
// Facts expire after 1 hour
await engram.retain({ text: '...', maxAge: 3600000 });
// Pre-extracted facts with TTL
await engram.retainDirect({ facts: [...], maxAge: 86400000 }); // 24h
// Manually prune expired facts
engram.pruneExpired();
// Forget specific fact
engram.forget({ factId: 'abc-123' });
// Forget facts older than 30 days
engram.forget({ olderThan: 30 * 24 * 60 * 60 * 1000 });
// Forget facts before a date
engram.forget({ before: new Date('2024-01-01') });Expired facts are also opportunistically pruned at the start of each retain() and retainDirect() call.
Testing and Development
Requirements: Node >= 20
Project Structure
src/
engram.ts Main orchestrator (Engram class)
types.ts All domain interfaces
index.ts Public API exports
store/
schema.ts SQLite DDL (tables, FTS5, triggers, vec0)
migrations.ts Forward-only schema migrations
store.ts MemoryStore — SQLite persistence layer
pipelines/
retain.ts Ingestion: extract → embed → store
recall.ts Retrieval: hybrid search + rank fusion
reflect.ts Synthesis: cluster → synthesize → archive
extract.ts LLM prompt builders
providers/
types.ts EmbeddingProvider + LLMProvider interfaces
ai-sdk.ts Vercel AI SDK implementation
defaults.ts Provider factory (openai/anthropic/ollama)
mcp/
server.ts MCP server (6 tools + 1 resource)
utils/
hash.ts SHA-256 content hashing
temporal.ts Natural language date parsing
entity-resolver.ts Entity dedup/merge
rank-fusion.ts Weighted score fusion
bin/
engram-mcp.ts CLI entry point for MCP server
tests/
helpers/
mock-providers.ts Deterministic 4D mock embedding + mock LLM
unit/ Store, hash, temporal, rank-fusion, entity-resolver
integration/ Full retain → recall → reflect cyclesRunning Tests
Tests use mock providers — deterministic 4-dimensional embeddings (MD5-based) and a mock LLM that does simple sentence splitting. No API key or network calls needed.
# Run all tests (57 tests, ~1s)
npm test
# Watch mode
npm run test:watchBuilding
npm run build # TypeScript → dist/
npm run dev # TypeScript watch modeTesting the MCP Server Locally
Option 1: With the MCP Inspector (interactive web UI)
npm run build
npx @modelcontextprotocol/inspector node dist/bin/engram-mcp.jsThis opens a browser UI where you can call retain, recall, reflect, forget, etc. interactively.
Option 2: With a real provider
export OPENAI_API_KEY=sk-...
npm run mcpThis starts the stdio MCP server. Connect any MCP client to it.
Option 3: Fully local with Ollama (no API key)
ollama pull nomic-embed-text && ollama pull llama3.2
ENGRAM_PROVIDER=ollama npm run mcpWriting Tests
Tests follow existing patterns in tests/. Use the mock providers from tests/helpers/mock-providers.ts:
import { Engram } from '../../src/engram.js';
import { MockEmbeddingProvider, MockLLMProvider } from '../helpers/mock-providers.js';
const embedding = new MockEmbeddingProvider(); // 4D deterministic vectors
const llm = new MockLLMProvider(); // Sentence splitting, no network
const engram = new Engram({ dbPath: ':memory:' }, embedding, llm);The MockLLMProvider accepts a custom extractFn if you need to control extraction behavior for a specific test.
This server cannot be installed
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/ReallyArtificial/engram'
If you have feedback or need assistance with the MCP directory API, please join our Discord server