Skip to main content
Glama
alexanderop

memory-plugin

by alexanderop

memory-plugin

Hermes-style bounded, curated persistent memory for Claude Code.

Two files persist across sessions and are injected into context at session start:

File (~/.claude/memory-plugin/)

Purpose

Limit

MEMORY.md

Agent notes — environment, conventions, completed work

2,200 chars

USER.md

User profile — preferences, style, habits

1,375 chars

How it maps to Claude Code

Hermes feature

Here

Inject memory at session start (frozen snapshot)

SessionStart hook → additionalContext (scripts/inject-memory.mjs)

memory tool: add / replace / remove

MCP server mcp/memory-server.mjsmemory_add / memory_replace / memory_remove (+ memory_list)

Char limits + "memory full" error

Enforced in the MCP server

Substring matching for replace/remove

matchOne() — short unique substring, errors if ambiguous

Duplicate prevention

Exact-match no-op on add

Security scanning

lib/security.mjs — injection/exfil/backdoor patterns + invisible Unicode

Background self-improvement review (memory only)

Stop hook → detached claude -p review (scripts/review.mjs, opt-in)

Correction detector

UserPromptSubmit hook → regex (lib/correction.mjs, scripts/detect-correction.mjs)

Skill capture (procedural memory)

Always-on policy injected at session start — agent writes SKILL.md inline

The model decides what to save via skills/memory/SKILL.md; the server enforces how.

Related MCP server: Claude Memory Server

Three learning loops

Modeled on the real pi-hermes-memory implementation (which splits learning into separate loops on purpose):

  1. Background review (memory)Stop hook fires a detached claude -p that mines the finished turn for durable facts/preferences/corrections and saves them via the memory tools. It is explicitly forbidden from writing skills — a stale subprocess with only a transcript snapshot would author bad procedures. Opt-in (token cost).

  2. Correction detectorUserPromptSubmit hook runs a free, two-pass regex over each prompt (strong patterns always fire; weak patterns need a following directive word; negative patterns suppress). On a hit it nudges the agent to persist the lesson before answering. Detection is pure regex — no LLM call.

  3. Skill capture (procedural) — done inline by the main agent, never in a subprocess. The session-start policy tells it to write a structured SKILL.md (## When to Use / ## Procedure / ## Pitfalls / ## Verification) to ~/.claude/skills/ (portable) or .claude/skills/ (repo-specific) the moment it finishes a complex, reusable workflow — while it still has full context. Claude Code's native skill discovery then handles progressive disclosure.

The Curator (skill lifecycle)

Ports Hermes' active → stale (30d) → archived (90d) aging so captured skills don't pile up forever.

  • Use signal — a PostToolUse hook on the Skill tool records real invocations into a .last-used sidecar (scripts/track-skill-use.mjs). "Last used" is max(.last-used, SKILL.md mtime, dir mtime), so a skill that's invoked but never edited isn't falsely aged out, and a fresh skill is never stale.

  • Weekly sweep — a SessionStart hook (scripts/curate.mjs) throttled to once per 7 days via a sentinel. Approximates Hermes' "every 7 days after idle" without a daemon.

  • Safetynotify-only by default: it reports stale/archivable skills in the session context and only moves files when MEMORY_CURATOR_ARCHIVE=1. pinned: true in frontmatter is never touched. Archiving moves (never deletes) into ~/.claude/skills/.archive/ and is fully reversible.

Install (as a Claude Code plugin)

This repo is also a single-plugin marketplace (.claude-plugin/marketplace.json), so installation is two slash commands inside Claude Code. There is no build or npm install step — the MCP server is zero-dependency.

/plugin marketplace add alexanderop/claude-code-memory
/plugin install memory@memory
  • marketplace add <owner>/<repo> registers this GitHub repo as a marketplace.

  • install <plugin>@<marketplace> — both are named memory here (plugin name memory, marketplace name memory).

Claude Code copies the plugin into ~/.claude/plugins/cache and resolves ${CLAUDE_PLUGIN_ROOT} automatically, so the bundled MCP server and hooks just work. They activate on the next turn — run /reload-plugins to pick them up without restarting. Verify with /plugin (shows memory enabled) and /mcp (shows the memory server with its memory_* tools).

Non-interactive / team install

Declare it in .claude/settings.json (project) or ~/.claude/settings.json (global):

{
  "extraKnownMarketplaces": {
    "memory": { "source": { "source": "github", "repo": "alexanderop/claude-code-memory" } }
  },
  "enabledPlugins": { "memory@memory": true }
}

Local development

Point the marketplace at a local checkout instead of GitHub:

/plugin marketplace add ~/Projects/memory-plugin
/plugin install memory@memory

Background review (optional)

Off by default — it spawns a headless claude -p after each turn and costs tokens. Enable by exporting MEMORY_REVIEW_ENABLED=1 in the environment Claude Code runs in. It is recursion-guarded (MEMORY_REVIEW=1 on the child) and never blocks the turn.

Config (env vars)

Var

Default

Effect

MEMORY_PLUGIN_DIR

~/.claude/memory-plugin

Where the two files live

MEMORY_CHAR_LIMIT

2200

MEMORY.md limit

MEMORY_USER_CHAR_LIMIT

1375

USER.md limit

MEMORY_REVIEW_ENABLED

unset

1 enables the Stop-hook review

MEMORY_SKILLS_DIR

~/.claude/skills

Skills dir the Curator manages

MEMORY_CURATOR_ARCHIVE

unset

1 lets the Curator move (not just report) stale skills

MEMORY_CURATOR_STALE_DAYS

30

Unused-days before a skill is "stale"

MEMORY_CURATOR_ARCHIVE_DAYS

90

Unused-days before a skill is archivable

MEMORY_CURATOR_INTERVAL_DAYS

7

Min days between Curator sweeps

Testing

Layered by cost, zero dev dependencies (plain Node — no Bun, no npm install):

npm test                 # unit + integration — zero token, ~1s, run on every edit
npm run test:unit        # manifests, skill frontmatter, and the store/security/correction/curator modules
npm run test:integration # version consistency, hook wiring, MCP path, markdown links, real MCP stdio round-trip
npm run test:e2e         # model-backed: loads the plugin via `claude --plugin-dir` (~$0.01, needs auth)

The unit suite exercises the real logic modules against temp dirs (never your ~/.claude); the integration suite spawns the actual MCP server and drives it over JSON-RPC. See docs/testing-strategy.md for the full layout and CI policy.

Not included (vs Hermes)

  • Write-approval gating — add a PreToolUse hook matching mcp__memory__memory_* that returns {"permissionDecision":"ask"} (or stages to a pending file).

  • Session search — Claude Code already ships a conversation-search skill over ~/.claude/projects/*/*.jsonl; no FTS5 server needed.

A
license - permissive license
-
quality - not tested
A
maintenance

Maintenance

Maintainers
Response time
0dRelease cycle
3Releases (12mo)
Commit activity

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/alexanderop/claude-code-memory'

If you have feedback or need assistance with the MCP directory API, please join our Discord server