Skip to main content

Local archive + web viewer + MCP server for Claude Desktop and Claude Code conversations. Browse, search, export, and let Claude query its own history.

Project description

Claude Explorer

A tool to extract, browse, search, and export your Claude conversation history — even if you've lost access to the email address on your account.

Disclaimer: This is an independent, community-built project. It is not affiliated with, endorsed by, sponsored by, or supported by Anthropic, PBC. "Claude" and "Claude Code" are trademarks of Anthropic, PBC. This project consumes Anthropic's products as a user would — via the same APIs and on-disk file formats the official clients use — but nothing here represents an Anthropic-sanctioned interface, and the formats this project depends on may change without notice.

Quick Start

# install uv if needed: https://docs.astral.sh/uv/getting-started/installation/
uvx claude-explorer serve

That's it. Open http://localhost:8000 in your browser and your Claude Code sessions are visible immediately. Click Refresh in the sidebar to capture credentials and fetch your Claude Desktop history (the UI handles capture via in-process Playwright on first run; no terminal commands needed).

Available on PyPI once V1 ships; until then, see From source (for contributors) below.

Features

  • Browse conversations from both Claude Desktop and Claude Code
  • Full-text search across all messages with instant results
  • Export to Markdown or PDF
  • Dark mode with automatic system preference detection
  • Keyboard navigation with Emacs and Vim modes
  • Message tree visualization for branched conversations
  • Command palette (Cmd+K) for quick navigation
  • Claude Code integration with project grouping and subagent display
  • MCP server exposing your saved sessions to Claude Desktop and Claude Code

Background

Where Claude Desktop Stores Your Data

Claude Desktop is an Electron app that wraps claude.ai. Your conversation history is stored server-side only — there is no meaningful local copy. If you go looking on your Mac, here's what you'll find (and won't find):

Path What's There
~/Library/Application Support/Claude/IndexedDB/https_claude.ai_0.indexeddb.leveldb Chat drafts only — unsent text in the input box. No conversation history.
~/Library/Caches/com.anthropic.claudefordesktop/Cache.db App update URLs only — not conversation data.

The official export path (Settings → Privacy → Export Data) sends a download link to your account email. If you've lost access to that email address, you're stuck.

The Workaround

This tool offers two ways to capture your session credentials:

Method A: Browser Login (Default)

  1. Open a Playwright-controlled browser to claude.ai
  2. Let you log in normally (SSO, email, etc.)
  3. Extract the session cookie (sessionKey) after authentication
  4. Use that session cookie to bulk-fetch all conversations directly from the claude.ai API

Method B: Proxy Interception (--proxy)

If you've lost access to the login (e.g., work SSO disabled) but Claude Desktop is still authenticated:

  1. Run mitmproxy as a local proxy on port 8080
  2. Launch Claude Desktop through the proxy with --proxy-server=127.0.0.1:8080 --ignore-certificate-errors
  3. Watch the intercepted traffic to capture the session cookie (sessionKey)
  4. Use that session cookie to bulk-fetch all conversations directly from the claude.ai API

Claude Desktop does not do SSL certificate pinning, which makes the proxy method possible.


The claude.ai API (Unofficial, Observed)

All endpoints authenticate via a Cookie header:

Cookie: sessionKey=sk-ant-sid01-...

Endpoints

GET /api/organizations/{org_id}/chat_conversations
    ?limit=30&starred=false&offset=0
    → Paginated list of conversation summaries

GET /api/organizations/{org_id}/chat_conversations/count_all
    → { "count": N }

GET /api/organizations/{org_id}/chat_conversations/{uuid}
    ?tree=True&rendering_mode=messages&render_all_tools=true&consistency=strong
    → Full conversation with all messages

Conversation Structure

{
  "uuid": "...",
  "name": "Conversation title",
  "model": "claude-sonnet-4-6",
  "created_at": "2026-02-25T19:14:43Z",
  "updated_at": "2026-02-25T20:30:51Z",
  "is_starred": false,
  "is_temporary": false,
  "current_leaf_message_uuid": "...",
  "chat_messages": [...]
}

Message Structure

{
  "uuid": "...",
  "sender": "human" | "assistant",
  "text": "message text",
  "content": [...],
  "index": 0,
  "created_at": "2026-02-25T19:14:43Z",
  "truncated": false,
  "parent_message_uuid": "...",
  "attachments": [],
  "files": []
}

Message Trees and Branches

Messages form a tree, not a flat list. Each message has a parent_message_uuid linking it to its parent. When you edit a message and regenerate a response, Claude creates a new branch — so a conversation can have multiple alternate paths.

current_leaf_message_uuid on the conversation object points to the tip of the currently active branch. To reconstruct the displayed conversation, walk backward from the leaf following parent_message_uuid links until you reach the root, then reverse the list.

Content Blocks

The content array contains typed blocks:

Type Description
text Regular message text
tool_use A tool call (name + input dict)
tool_result The result of a tool call
image An embedded image

Project Structure

claude-explorer/
├── README.md
├── CLAUDE.md                 # Development guide
├── pyproject.toml            # Python dependencies + CLI entry point
├── PLANS/
│   ├── overview.md           # Project goals and architecture
│   ├── fetcher.md            # Fetcher design and test plan
│   ├── backend.md            # Backend API design and test plan
│   └── frontend.md           # Frontend design and test plan
├── fetcher/
│   ├── cli.py                # CLI entry point (claude-explorer command)
│   ├── playwright_capture.py # Browser-based credential capture (default)
│   ├── mitmproxy_addon.py    # Proxy-based credential capture (--proxy)
│   └── bulk_fetch.py         # Downloads all conversations to local JSON
├── backend/
│   ├── main.py               # FastAPI app
│   ├── models.py             # Pydantic models
│   ├── store.py              # Reads and indexes JSON files from disk
│   ├── search.py             # Full-text search
│   ├── export.py             # Markdown + PDF export
│   └── routers/              # conversations, search, export endpoints
├── frontend/
│   └── src/                  # React 18 + TypeScript + Tailwind + shadcn/ui
└── scripts/
    ├── check-cleanup-period.py        # Inspect/fix Claude Code's cleanupPeriodDays
    └── macos-restore-claude-projects.py  # Recover deleted projects from Time Machine

From source (for contributors)

If you want to hack on the project, run from a git checkout, or you're a contributor: install from source instead.

Prerequisites

# Install uv (Python package manager)
curl -LsSf https://astral.sh/uv/install.sh | sh

# Clone and install
git clone https://github.com/rpeck/claude-explorer
cd claude-explorer
uv sync

# Install Playwright browsers (for browser-based credential capture)
uv run playwright install chromium

Step 1: Capture Your Session Cookie

There are two methods to capture credentials. Choose the one that fits your situation:

Method A: Browser Login (Default)

The simplest approach — opens a browser window where you log into Claude normally:

uv run claude-explorer capture

This opens Chromium, navigates to claude.ai, and waits for you to log in. Once authenticated, credentials are automatically extracted and saved to ~/.claude-explorer/credentials.json.

Options:

  • --timeout N — Max seconds to wait for login (default: 300)

Method B: Proxy Interception (--proxy)

Use this method when you can't log into the web UI but Claude Desktop is still authenticated. This was a lifesaver for recovering conversations from a work account after losing access to the SSO login.

# Terminal 1 — start the proxy (requires ANSI terminal)
uv run claude-explorer capture --proxy

# Terminal 2 — launch Claude Desktop through the proxy
open -a "Claude" --args --proxy-server="127.0.0.1:8080" --ignore-certificate-errors

Click around in Claude Desktop for a few seconds. The addon will print:

✅ Credentials captured! You can now quit mitmproxy (q) and close Claude Desktop.

Options:

  • --port N — Proxy port (default: 8080)

Platform-specific launch commands:

# macOS
open -a "Claude" --args --proxy-server="127.0.0.1:8080" --ignore-certificate-errors

# Windows
"C:\...\Claude.exe" --proxy-server="127.0.0.1:8080" --ignore-certificate-errors

# Linux
claude --proxy-server="127.0.0.1:8080" --ignore-certificate-errors

Note: mitmproxy requires a proper ANSI terminal (Terminal.app, iTerm2, etc.). It will not work in non-TTY environments.


Both methods save credentials to ~/.claude-explorer/credentials.json.

Step 2: Download Your Conversations

uv run claude-explorer fetch

This downloads all your conversations as JSON files to ~/.claude-explorer/conversations/. A rate-limited 0.3s delay between requests keeps things polite. Incremental mode (default) skips conversations you've already downloaded.

Options:

  • --full-refresh — Re-download all conversations
  • --limit N — Download only N conversations
  • --verbose — Show detailed progress

Step 3: Browse and Export

uv run claude-explorer serve

Opens the web app at http://localhost:8000.

One-button Refresh (Build-9)

Once the web app is running, the Refresh button in the sidebar footer owns the entire pipeline. A single click:

  1. Tries an incremental fetch with whatever credentials are on disk.
  2. If credentials are missing, or the fetch fails with 401/403/cf-mitigated, the backend automatically launches the same Playwright login flow as claude-explorer capture — a Chromium window opens, you log in, and the new sessionKey is written to ~/.claude-explorer/credentials.json (0o600, atomically).
  3. The fetch then continues automatically. Existing conversations are preserved (incremental — never --full-refresh automatically).

The toast walks you through each phase:

"Opening browser to log in to Claude…" → "Waiting for you to log in (Ns elapsed)…" → "Credentials captured. Fetching…" → "Fetched +N new conversations."

If the capture window is closed or login times out (5 min default), the toast becomes sticky with a Retry action. You never have to drop to the CLI to re-capture.

Manual override: the Details modal still exposes "Full Refresh" and "Fetch New" buttons that hit the original /fetch/start endpoint without auto-capture.

For development with hot-reload:

# Terminal 1 — backend
DYLD_LIBRARY_PATH=/opt/homebrew/lib uv run uvicorn backend.main:app --reload

# Terminal 2 — frontend
cd frontend && npm run dev

Then open http://localhost:5173.


MCP Server

The project ships with a built-in Model Context Protocol server that lets Claude Desktop or Claude Code query your saved conversations directly. Once configured, you can ask Claude things like "find the session where I debugged the weasyprint install" or "export the ZFS conversation I had last week as markdown" and Claude will use the tools below to answer.

Tools exposed

Tool Purpose
list_sessions Full-text search / list conversation sessions, optionally filtered by source or project
list_projects List distinct projects with session counts
get_session_outline Lightweight per-message summaries (cached in SQLite) for a specific session
get_messages Full message content for specific positions or UUIDs, with optional tool calls/results
export_session Markdown export of a full or partial session

The server runs over stdio (no network port) and reads from the same ~/.claude-explorer/conversations/ directory the web UI uses.

Prerequisites

Make sure the project is installed and conversations have been fetched at least once.

If you installed via PyPI/uvx, you already have everything you need:

uvx claude-explorer serve   # fetch via the Refresh button, then quit

If you're working from a git checkout (contributor flow), the equivalent is:

cd /path/to/claude-explorer
uv sync
uv run claude-explorer capture   # one-time
uv run claude-explorer fetch

The MCP entry point is claude-explorer mcp. You can verify it works standalone:

# PyPI/uvx install:
uvx claude-explorer mcp
# (prints nothing; it's waiting for MCP JSON-RPC on stdin — Ctrl+C to exit)

# From source:
uv run --directory /path/to/claude-explorer claude-explorer mcp

Claude Code setup (all platforms)

The simplest path is the claude mcp add CLI, which writes the config for you:

claude mcp add claude-sessions \
  -- uv run --directory /absolute/path/to/claude-explorer claude-explorer mcp

Use --scope user to make it available in every project, or --scope project (default) to scope it to the current repo (writes to .mcp.json).

Verify with:

claude mcp list

Or edit the config files directly:

  • User scope: ~/.claude.json (key: mcpServers)
  • Project scope: .mcp.json in the project root
{
  "mcpServers": {
    "claude-sessions": {
      "command": "uv",
      "args": [
        "run",
        "--directory",
        "/absolute/path/to/claude-explorer",
        "claude-explorer",
        "mcp"
      ]
    }
  }
}

Note: Use the absolute path to the repo. MCP clients do not inherit your shell's cwd. If uv is not on the default PATH the client sees, replace "uv" with the absolute path from which uv (typically ~/.local/bin/uv or /opt/homebrew/bin/uv).

Claude Desktop setup

Claude Desktop reads from a claude_desktop_config.json file whose location depends on the OS. Create or edit it and add the claude-sessions entry under mcpServers, then fully quit and relaunch Claude Desktop.

macOS

Config path:

~/Library/Application Support/Claude/claude_desktop_config.json
{
  "mcpServers": {
    "claude-sessions": {
      "command": "/opt/homebrew/bin/uv",
      "args": [
        "run",
        "--directory",
        "/Users/YOU/Source/claude-explorer",
        "claude-explorer",
        "mcp"
      ]
    }
  }
}

If you installed uv via the standalone installer instead of Homebrew, it will be at ~/.local/bin/uv. Use which uv to confirm.

Windows

Config path:

%APPDATA%\Claude\claude_desktop_config.json

(typically C:\Users\YOU\AppData\Roaming\Claude\claude_desktop_config.json)

{
  "mcpServers": {
    "claude-sessions": {
      "command": "uv",
      "args": [
        "run",
        "--directory",
        "C:\\Users\\YOU\\Source\\claude-explorer",
        "claude-explorer",
        "mcp"
      ]
    }
  }
}

Use double-backslashes (\\) in JSON string paths. If uv is not on the system PATH as seen by Claude Desktop, use its absolute path (e.g. C:\\Users\\YOU\\.local\\bin\\uv.exe).

Linux

Config path:

~/.config/Claude/claude_desktop_config.json
{
  "mcpServers": {
    "claude-sessions": {
      "command": "/home/YOU/.local/bin/uv",
      "args": [
        "run",
        "--directory",
        "/home/YOU/Source/claude-explorer",
        "claude-explorer",
        "mcp"
      ]
    }
  }
}

Verifying it works

After restarting your client, ask it to search your history, e.g.:

"Use the claude-sessions MCP server to list my 5 most recent sessions."

In Claude Code you can also run /mcp to see the server status and the list of tools it registered.

Troubleshooting

  • "command not found: uv" — the MCP client doesn't see your shell PATH. Use the absolute path to uv in command.
  • "Session not found" / empty results — run uv run claude-explorer fetch first; the MCP server reads from ~/.claude-explorer/conversations/.
  • Need to use a non-default data dir — set CLAUDE_EXPLORER_DATA_DIR via an env block in the MCP config:
    "env": { "CLAUDE_EXPLORER_DATA_DIR": "/path/to/conversations" }
    
  • Stale session outlines after a branch switch — outlines are cached in ~/.claude-explorer/cache.db. Delete that file to force a full rebuild.

Companion Utilities (macOS)

The scripts/ directory ships two standalone utilities that address a Claude Code gotcha you may have hit while using this tool: Claude Code silently auto-deletes session files in ~/.claude/projects/ older than cleanupPeriodDays (default: 30). When a project subdirectory becomes empty, it is removed entirely.

scripts/check-cleanup-period.py — inspect/fix the auto-cleanup

Reports the current cleanupPeriodDays value in ~/.claude/settings.json and (with --set N) atomically updates it while preserving every other key.

# Report only
python3 scripts/check-cleanup-period.py

# Effectively disable auto-cleanup (~10 years)
python3 scripts/check-cleanup-period.py --set 3650

Refuses to set 0 (which silently disables conversation persistence — Claude Code issue #23710). Warns if you set anything shorter than 30 or 365 days.

scripts/macos-restore-claude-projects.py — recover deleted projects from Time Machine

Walks every Time Machine snapshot under /Volumes/.timemachine/<UUID>/, collects the union of every ~/.claude/projects/<name>/ ever seen, diffs against the live directory, and copies the newest backup of each missing dir to ~/.claude/projects-recovered/. Never overwrites anything.

# Preview what's recoverable across ALL backups
sudo python3 scripts/macos-restore-claude-projects.py --dry-run

# Recover all missing projects (default)
sudo python3 scripts/macos-restore-claude-projects.py

# Limit how far back to scan
sudo python3 scripts/macos-restore-claude-projects.py --since 2026-01-01
sudo python3 scripts/macos-restore-claude-projects.py --days 60

# Recover AND auto-move into ~/.claude/projects/ (skips any that already exist)
sudo python3 scripts/macos-restore-claude-projects.py --apply

Requires: Terminal must have Full Disk Access (System Settings → Privacy & Security → Full Disk Access → add Terminal). Without FDA, macOS returns "Operation not permitted" when reading TM snapshot directories. The script detects this and tells you what to do.

After recovery, set a high cleanupPeriodDays with the checker above so this doesn't happen again.


Known Limitations

  • Session key expiry: sessionKey will eventually expire. Re-run the mitmproxy step to get a fresh one.
  • Cloudflare cookie: The cf_clearance cookie may be required on some networks. The fetcher attempts to capture it alongside sessionKey.
  • Truncated messages: The API returns truncated: true for very long messages. A per-message full-content endpoint has not yet been identified.
  • Import: There is no known write API for creating conversations. Migration to a new account is not currently possible programmatically.
  • macOS only (for now): The open -a "Claude" proxy launch command is macOS-specific. Linux and Windows launch commands are documented in fetcher/README.md.

Tech Stack

Layer Technology
Credential Capture Playwright (browser login), mitmproxy (proxy interception)
Fetcher httpx, curl_cffi
Backend FastAPI, uvicorn, uv, weasyprint
Frontend React 18, TypeScript, Vite, Tailwind CSS v4, shadcn/ui, TanStack Query
Search SQLite FTS5 (primary), orjson + FileCache + ThreadPoolExecutor linear-scan fallback
MCP server FastMCP (5 tools: list_sessions, list_projects, get_session_outline, get_messages, export_session)
Export Markdown (built-in), PDF (weasyprint)
Packaging hatchling (PyPI wheel includes pre-built React bundle)

Contributing

PRs welcome — see CONTRIBUTING.md for prerequisites, dev workflow, code style, and the CLA process.

License

This project is licensed under the Apache License 2.0. Contributors agree to the Contributor License Agreement (via CLA Assistant) on their first pull request; the signature applies to all subsequent contributions.

See the disclaimer at the top of this README regarding Anthropic trademarks and the unaffiliated nature of this project.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distribution

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

claude_explorer-0.1.0-py3-none-any.whl (470.0 kB view details)

Uploaded Python 3

File details

Details for the file claude_explorer-0.1.0-py3-none-any.whl.

File metadata

File hashes

Hashes for claude_explorer-0.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 02c5c1b63a01f0fc3551e86852e1e4742440b2c71c94bb7884bfe0e0c51a3195
MD5 4bad0f8a2fcd191bdc18bdc08a2fb8c7
BLAKE2b-256 49deb1b852d43de92b125d2f497277f429c50eca5b7d1aa6863fe65eaad72526

See more details on using hashes here.

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