Skip to content

Memory System

Hermes Agent has bounded, curated memory that persists across sessions. This lets it remember your preferences, your projects, your environment, and things it has learned.

Two files make up the agent’s memory:

FilePurposeChar Limit
MEMORY.mdAgent’s personal notes — environment facts, conventions, things learned2,200 chars (~800 tokens)
USER.mdUser profile — your preferences, communication style, expectations1,375 chars (~500 tokens)

Both are stored in ~/.hermes/memories/ and are injected into the system prompt as a frozen snapshot at session start. The agent manages its own memory via the memory tool — it can add, replace, or remove entries.

At the start of every session, memory entries are loaded from disk and rendered into the system prompt as a frozen block:

══════════════════════════════════════════════
MEMORY (your personal notes) [67% — 1,474/2,200 chars]
══════════════════════════════════════════════
User's project is a Rust web service at ~/code/myapi using Axum + SQLx
§
This machine runs Ubuntu 22.04, has Docker and Podman installed
§
User prefers concise responses, dislikes verbose explanations

The format includes:

  • A header showing which store (MEMORY or USER PROFILE)
  • Usage percentage and character counts so the agent knows capacity
  • Individual entries separated by § (section sign) delimiters
  • Entries can be multiline

Frozen snapshot pattern: The system prompt injection is captured once at session start and never changes mid-session. This preserves the LLM’s prefix cache for performance. When the agent adds/removes memory entries during a session, the changes are persisted to disk immediately but won’t appear in the system prompt until the next session starts.

The agent uses the memory tool with these actions:

  • add — Add a new memory entry
  • replace — Replace an existing entry with updated content (uses substring matching via old_text)
  • remove — Remove an entry that’s no longer relevant (uses substring matching via old_text)

There is no read action — memory content is automatically injected into the system prompt at session start.

The replace and remove actions use short unique substring matching — you don’t need the full entry text:

# If memory contains "User prefers dark mode in all editors"
memory(action="replace", target="memory",
old_text="dark mode",
content="User prefers light mode in VS Code, dark mode in terminal")

For information the agent needs to remember about the environment, workflows, and lessons learned:

  • Environment facts (OS, tools, project structure)
  • Project conventions and configuration
  • Tool quirks and workarounds discovered
  • Completed task diary entries
  • Skills and techniques that worked

For information about the user’s identity, preferences, and communication style:

  • Name, role, timezone
  • Communication preferences (concise vs detailed, format preferences)
  • Pet peeves and things to avoid
  • Workflow habits
  • Technical skill level

The agent saves automatically — you don’t need to ask. It saves when it learns:

  • User preferences: “I prefer TypeScript over JavaScript” → save to user
  • Environment facts: “This server runs Debian 12 with PostgreSQL 16” → save to memory
  • Corrections: “Don’t use sudo for Docker commands, user is in docker group” → save to memory
  • Conventions: “Project uses tabs, 120-char line width, Google-style docstrings” → save to memory
  • Completed work: “Migrated database from MySQL to PostgreSQL on 2026-01-15” → save to memory
  • Explicit requests: “Remember that my API key rotation happens monthly” → save to memory
  • Trivial/obvious info: “User asked about Python” — too vague to be useful
  • Easily re-discovered facts: “Python 3.12 supports f-string nesting” — can web search this
  • Raw data dumps: Large code blocks, log files, data tables — too big for memory
  • Session-specific ephemera: Temporary file paths, one-off debugging context
  • Information already in context files: SOUL.md and AGENTS.md content

Memory has strict character limits to keep system prompts bounded:

StoreLimitTypical entries
memory2,200 chars8-15 entries
user1,375 chars5-10 entries

When you try to add an entry that would exceed the limit, the tool returns an error. The agent should then:

  1. Read the current entries (shown in the error response)
  2. Identify entries that can be removed or consolidated
  3. Use replace to merge related entries into shorter versions
  4. Then add the new entry

Best practice: When memory is above 80% capacity, consolidate entries before adding new ones.

Beyond MEMORY.md and USER.md, the agent can search its past conversations using the session_search tool:

  • All CLI and messaging sessions are stored in SQLite (~/.hermes/state.db) with FTS5 full-text search
  • Search queries return actual messages from the DB — no LLM summarization, no truncation
  • The agent can find things it discussed weeks ago, even if they’re not in its active memory
Terminal window
hermes sessions list # Browse past sessions
FeaturePersistent MemorySession Search
Capacity~1,300 tokens totalUnlimited (all sessions)
SpeedInstant (in system prompt)~20ms FTS5 query
CostToken cost in every promptFree — no LLM calls
Use caseKey facts always availableFinding specific past conversations
ManagementManually curated by agentAutomatic — all sessions stored
# In ~/.hermes/config.yaml
memory:
memory_enabled: true
user_profile_enabled: true
memory_char_limit: 2200 # ~800 tokens
user_char_limit: 1375 # ~500 tokens

For deeper, persistent memory that goes beyond MEMORY.md and USER.md, Hermes ships with 8 external memory provider plugins — including Honcho, OpenViking, Mem0, Hindsight, Holographic, RetainDB, ByteRover, and Supermemory.

External providers run alongside built-in memory (never replacing it) and add capabilities like knowledge graphs, semantic search, automatic fact extraction, and cross-session user modeling.

Terminal window
hermes memory setup # pick a provider and configure it
hermes memory status # check what's active

See the Memory Providers guide for full details.