Ghost Writer MCP
Scans local Git repositories to extract commit history and aggregate engineering activity for LinkedIn post generation.
Fetches commits and merged pull requests from GitHub repositories via the GitHub API for repositories without local clones.
Uses Ollama as a local LLM provider for content classification, draft generation, and confidentiality review.
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., "@Ghost Writer MCPscan my recent commits and draft a LinkedIn post"
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.
Ghost Writer MCP
An MCP server that scans your Git repositories, identifies interesting engineering work, and generates LinkedIn post drafts — with built-in confidentiality sanitisation.
What it does
Your repos → scan activity → classify by content potential → generate draft → sanitise → reviewScan — reads
git logfrom local repo clones (zero API calls, instant)Aggregate — groups commits by conventional-commit prefix and clusters related work
Classify — LLM ranks groups by content potential, assigns pillars and angles
Generate — LLM writes a LinkedIn draft in your chosen format (war story, hot take, tactical howto, TIL, deep dive)
Sanitise — three-gate safety: regex blocklist → LLM review → human review
Quick start
Prerequisites
Python 3.12+
uv (recommended) or pip
An LLM provider: Ollama (free, local) or an Anthropic API key
Local clones of the repos you want to scan
Install
Option A — clone and run (recommended for customisation):
git clone https://github.com/fabdendev/ghost-writer-mcp.git
cd ghost-writer-mcp
uv syncOption B — run directly with uvx (no clone needed):
uvx ghost-writer-mcpConfigure
cp config.example.yaml config.yamlEdit config.yaml with your repos, blocklist, and LLM settings. See config.example.yaml for a fully documented template.
For Ollama (free, local):
llm:
provider: ollama
base_url: "http://localhost:11434/v1"
classifier_model: qwen3:8b
generator_model: qwen3:8b
api_key: "ollama"ollama pull qwen3:8b
ollama serveFor Anthropic (cloud):
llm:
provider: anthropic
classifier_model: claude-haiku-4-5-20251001
generator_model: claude-haiku-4-5-20251001
api_key: "${ANTHROPIC_API_KEY}"echo "ANTHROPIC_API_KEY=sk-ant-..." > .envAdd to Claude Code
Add to your Claude Code MCP settings (~/.claude/settings.json):
If installed from clone:
{
"mcpServers": {
"ghost-writer": {
"command": "uv",
"args": ["run", "--directory", "/path/to/ghost-writer-mcp", "fastmcp", "run", "src/server.py"]
}
}
}If using uvx:
{
"mcpServers": {
"ghost-writer": {
"command": "uvx",
"args": ["ghost-writer-mcp"]
}
}
}Use via MCP
From Claude Code (or any MCP client):
scan_activity # scan all configured repos (last 7 days)
scan_activity(repo="my-project") # scan a single repo
scan_activity(days=30) # look back 30 days
generate_draft(activity_index=1) # generate from top candidate
generate_draft(3, format="hot_take") # override format
edit_draft(1, "make it shorter") # refine with natural language
list_drafts(status="pending") # see saved draftsUse via CLI
You can also test without an MCP client:
uv run python -m src scan --days 14 # scan all repos
uv run python -m src scan --repo my-project # scan one repo
uv run python -m src generate 1 # draft from top result
uv run python -m src generate 3 --format hot_take # override format
uv run python -m src list # list saved draftsTools
Tool | Description |
| Scan repos, aggregate commits, classify and rank by content potential |
| Generate a LinkedIn draft from a classified activity |
| Refine a draft with natural language instructions |
| List saved drafts, optionally filtered by status |
Scanning modes
Ghost Writer supports two scanning backends:
Local git (default) — reads
git logfrom local clones. Zero API calls, instant results. Setlocal_pathon each repo in your config.GitHub API (fallback) — used automatically for repos without
local_path. Requires a GitHub token (github.tokenin config). Fetches up to 30 commits and 20 merged PRs per repo.
You can mix both: some repos with local clones, others via API.
Confidentiality
Ghost Writer uses three safety gates to prevent leaking sensitive information:
Gate 1 — Blocklist: regex-based detection and replacement of company names, client names, product names, infrastructure details, and people names
Gate 2 — LLM Review: the LLM scans generated text for anything that looks confidential and flags it
Gate 3 — Human Review: drafts are saved as
pending— you always get the final say
Configure your blocklist and abstractions in config.yaml:
sanitisation:
blocklist:
company_names: ["Acme Corp"]
client_names: ["Big Client"]
product_names: ["internal-tool"]
infrastructure: ["prod-db-01.internal"]
people: ["John Doe"]
abstractions:
"Acme Corp": "a mid-size tech company"
"internal-tool": "an internal platform"Content pillars
Define what topics you want to post about. The classifier maps activities to pillars:
content:
pillars:
- name: ai_engineering
description: "Building AI agents, LLM integration, prompt engineering"
repo_signals: ["agent", "llm", "prompt"]
weight: 1.0
- name: data_architecture
description: "Data pipelines, ETL, event-driven systems"
repo_signals: ["pipeline", "etl", "kafka"]
weight: 0.8Post formats
Format | Description |
| Problem → 3-5 concrete steps → takeaway |
| Contrarian claim backed by one specific thing you built |
| What broke, what you tried, what worked, the lesson |
| One surprising thing you learned, under 500 chars |
| 3-4 sections with trade-offs and alternatives |
Architecture
src/
├── server.py # FastMCP server (4 tools)
├── cli.py # Standalone CLI for testing without MCP
├── config.py # Pydantic config with env/shell resolution
├── llm_client.py # Unified Anthropic + OpenAI-compatible client
├── scanner/
│ ├── local_git.py # Git CLI scanner (primary)
│ ├── github_client.py # GitHub API scanner (alternative)
│ ├── aggregator.py # Commit grouping and clustering
│ └── activity.py # ActivityItem dataclass
├── content/
│ ├── classifier.py # LLM-based content scoring
│ ├── generator.py # Draft generation with sanitisation
│ ├── abstractor.py # Two-gate confidentiality layer
│ └── prompts/ # System prompts (classifier, generator, reviewer)
└── store/
├── database.py # SQLite persistence
└── blocklist.py # Regex-based blocklistDevelopment
uv sync --extra dev
uv run pytest # run tests
uv run ruff check src/ tests/ # lintLicense
MIT
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/fabdendev/ghost-writer-mcp'
If you have feedback or need assistance with the MCP directory API, please join our Discord server