personal-context
Enables LLMs to access a user's personal writing context—voice, style, opinions, expertise, projects, and communication patterns—via curated markdown files, helping the LLM match the user's voice when generating written content.
README
Personal Context
A minimal system for giving LLMs your writing voice, opinions, and style. Curated markdown files served via MCP using FastMCP.
Inspired by Karpathy's LLM Wiki and nlwhittemore's Personal Context Portfolio.
More background on the why behind this project here.
How It Works
context/*.md → FastMCP server → MCP → Claude Code (or any MCP client)
You maintain 6 markdown files about yourself. A FastMCP server exposes them as tools. When you ask an LLM to write something, it can pull your context and match your voice.
The 6 Context Files
| File | What It Captures |
|---|---|
identity.md |
Background, career arc, personal details |
writing-style.md |
Voice, tone, sentence patterns, vocabulary, annotated examples |
opinions.md |
Stances on topics you write about |
expertise.md |
Domains of deep knowledge |
projects.md |
Current and notable past projects |
communication.md |
How you communicate in different contexts (Slack, email, docs) |
Each file has YAML frontmatter tracking last_updated and source_refs (which source materials informed the content).
Getting Started
Use this as a template
- Fork or clone this repo
- Delete everything in
context/— those are my files, not yours - Install dependencies and start filling in your own context
Setup
git clone <this-repo> personal-context
cd personal-context
uv venv && source .venv/bin/activate
uv pip install -e ".[dev]"
Bootstrap from existing writing (optional)
If you have a folder of markdown blog posts or writing samples, the ingest script can generate draft context files:
# Copy or symlink your markdown files
ln -s /path/to/your/blog/posts sources/blogs
# Generate drafts (--cutoff filters by date in frontmatter)
python scripts/ingest.py sources/blogs/ --cutoff 2024-01-01 --output drafts
This creates draft files in drafts/ with excerpts grouped by category. Review them, extract what's useful, and write your curated versions into context/.
The ingest script expects markdown files with YAML frontmatter containing title, pubDate, and optionally categories. If your files don't have frontmatter, you can skip this step and write context files manually.
Write your context files manually
If you don't have existing writing to ingest, just create the 6 files in context/ yourself. Use this template:
---
last_updated: 2026-05-30
source_refs: []
---
# Writing Style
## Voice
[How do you write? Conversational? Formal? Technical? Direct?]
## Patterns
[Recurring structures, transitions, vocabulary choices]
## Examples
[2-3 representative excerpts from your actual writing, with annotations]
A good approach: start a Claude Code session and ask it to interview you. Share writing samples and let it draft context files for you to review and edit.
Connect to Claude Code
The MCP server runs entirely locally — Claude Code spawns it as a subprocess on your machine, and it just reads markdown files from disk. No data is sent to external services beyond the normal Claude API calls.
Register the server as a user-scoped MCP so it's available in every Claude Code session, regardless of which project you're working in:
claude mcp add --scope user personal-context -- /absolute/path/to/personal-context/.venv/bin/python /absolute/path/to/personal-context/server.py
Restart Claude Code. The MCP tools will be available in every session.
Setting up on another machine
To use the same context on a work machine or second computer:
- Clone the repo:
git clone <your-fork> personal-context - Install:
cd personal-context && uv venv && source .venv/bin/activate && uv pip install -e . - Register the MCP server (update paths to match where you cloned it):
claude mcp add --scope user personal-context -- /absolute/path/to/personal-context/.venv/bin/python /absolute/path/to/personal-context/server.py - Restart Claude Code
That's it — same context files, same tools, works in any repo you open. If you keep your context files committed, git pull on either machine keeps them in sync.
Make it automatic
By default, Claude Code won't call MCP tools unless you ask. To have it pull your context automatically when drafting written content, add a rule to your global ~/.claude/CLAUDE.md:
## Personal Context (MCP)
When drafting any written content (emails, messages, docs, social posts, bios, etc.), call `get_writing_style` from the `personal-context` MCP server first to match my voice and tone. For tasks that benefit from broader context (introductions, project summaries, etc.), use `get_all_context` instead.
Test it
In a new Claude Code session:
- Ask Claude to use your writing style to draft something
- It should automatically call
get_writing_styleorget_all_context - Compare the output to how you actually write and iterate on your context files
Adding Context Over Time
This system is manually curated, you update the files, this is not an automated pipeline.
When to update
- After publishing new writing — review if it reveals patterns not yet captured in
writing-style.md - After changing jobs/projects — update
projects.mdandidentity.md - After noticing the LLM gets your voice wrong — the gap between output and expectation tells you what's missing
- After adding new source material — drop files in
sources/private/(gitignored) orsources/blogs/, re-run ingest if helpful
How to update
- Edit the relevant
context/*.mdfile directly - Update the
last_updateddate in frontmatter - Add any new source refs to
source_refs - Commit
Private sources
Work emails, Slack exports, or other private writing go in sources/private/ which is gitignored. You can reference them in source_refs for provenance without committing the content.
Since sources/private/ is gitignored, these files are device-specific — they won't sync when you git pull on another machine. If you need the same private sources on multiple machines, copy them manually or sync via something outside git (e.g., iCloud, Dropbox).
Private served context
sources/private/ holds raw source material for ingest — it is not read by the MCP at runtime. If you have curated context that the MCP should serve but that must stay out of a public repo, put it in context/private.md, which is gitignored.
The server globs context/*.md, so context/private.md is returned by get_all_context() and context://private.md locally, but git never commits it. Like sources/private/, it's device-specific — copy it manually if you run the MCP on another machine.
Re-running ingest
If you add new blog posts or writing samples to sources/blogs/:
python scripts/ingest.py sources/blogs/ --cutoff 2024-01-01 --output drafts
This regenerates drafts (in drafts/, also gitignored). Review the new excerpts and fold anything useful into your context files.
MCP Tools
The server exposes:
| Tool/Resource | What It Does |
|---|---|
get_writing_style() |
Returns your writing-style.md — the most commonly needed file |
get_all_context() |
Returns all context files as a dict (includes a local private.md if present) |
context://{filename} |
Resource access to any individual file by name |
Project Structure
personal-context/
├── context/ # Your curated context files (the product)
│ └── private.md # Optional private served context (GITIGNORED)
├── sources/
│ ├── blogs/ # Public writing samples (committed or symlinked)
│ └── private/ # Private writing samples (GITIGNORED)
├── scripts/
│ └── ingest.py # Bootstrap drafts from existing writing
├── drafts/ # Generated drafts from ingest (GITIGNORED)
├── tests/ # Tests for ingest script and server
├── server.py # FastMCP MCP server
├── pyproject.toml
└── README.md
Running Tests
python -m pytest -v
Security
This repo is designed to be public, but remember you are putting personal information in it. A few things to know:
- Path traversal protection. The
get_contextresource handler validates that requested filenames resolve inside thecontext/directory. Traversal attempts like../../etc/passwdare rejected. - Private sources are gitignored.
sources/private/is in.gitignoreso work emails, Slack exports, etc. stay local. But be careful withsource_refsin frontmatter — the filenames are committed even if the files aren't. Use opaque names likework-email-1.mdinstead of descriptive titles. - Private served context is gitignored.
context/private.mdis in.gitignorefor curated context the MCP should serve locally but never commit (e.g. work-sensitive notes). It's the runtime-served counterpart tosources/private/. - Review your context files before committing. These files are meant to be public, but watch for details you didn't intend to share: financial specifics, internal company information, health details, or anything useful for phishing. If in doubt, leave it out.
.envis gitignored. If you extend this with API keys, they won't be committed accidentally.
Philosophy
- Start minimal. 6 files was enough for me as a starting point. Add complexity only when you outgrow it.
- Curate manually. You know your voice better than any automated pipeline. The LLM can help draft, but you decide what stays.
- Iterate from use. The best edits come from noticing when the LLM gets something wrong about your writing.
- Keep private things private. The
sources/private/directory exists so you can reference work writing without committing it.
Recommended Servers
playwright-mcp
A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.
Magic Component Platform (MCP)
An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.
Audiense Insights MCP Server
Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.
VeyraX MCP
Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.
graphlit-mcp-server
The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.
Kagi MCP Server
An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.
E2B
Using MCP to run code via e2b.
Neon Database
MCP server for interacting with Neon Management API and databases
Exa Search
A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.
Qdrant Server
This repository is an example of how to create a MCP server for Qdrant, a vector search engine.