FastMCP server for Substack: publish Notes and posts, pull analytics, manage drafts, bridge Obsidian vault drafts to Substack, and generate visual cards with pluggable image generators.
substack__test_connection -- verify auth (start here)
substack__publish_note text=... -- publish a Note immediately
substack__list_vault_drafts -- show drafts from your vault file
substack__publish_vault_draft index=N -- publish one vault draft as a Note
substack__create_draft title=... body=... -- create a post draft
substack__get_dashboard days=30 -- subscriber + view KPIs
substack__capture_analytics_to_vault -- snapshot analytics to vault markdown
test_connection(publication?)-- verify auth, return your profilelist_publications()-- show configured publications and which is defaultpublish_note(text, publication?, attachment_ids?)-- publish a Note immediately; markdown supportedcreate_note_attachment(image_path?, image_url?, link_url?, publication?)-- upload image or register link; returns attachment UUID for use with publish_notelist_my_notes(limit?, publication?)-- your recent Notes with reaction + comment countsreply_to_note(note_id, text, publication?)-- reply to a Note by IDlist_vault_drafts()-- parse vault drafts file, return index + preview for each draftpublish_vault_draft(index, publication?, move_to_published?)-- publish one vault draft as a Notebatch_publish_vault_drafts(indices, publication?)-- publish multiple vault drafts in sequence
create_draft(title, body, subtitle?, audience?, publication?)-- create a post draft; body is markdownupdate_draft(draft_id, title?, body?, subtitle?, audience?, publication?)-- edit an existing draftpublish_post(draft_id, send_email?, audience?, publication?)-- publish a draft live to subscribersschedule_post(draft_id, publish_at, publication?)-- schedule a draft for future publication (ISO 8601)list_drafts(limit?, publication?)-- list unpublished draftslist_published(limit?, publication?)-- list published posts with basic statsget_post(identifier, publication?)-- get full post by slug or numeric IDupload_image(image_path, publication?)-- upload an image to Substack CDN; returns CDN URLreact(post_id, publication?)-- heart a postrestack(post_id, publication?)-- restack a postcomment(post_id, body, publication?)-- comment on a postget_feed(limit?, publication?)-- your reader feed (posts from publications you follow)
get_dashboard(days?, publication?)-- KPIs: total/paid subscribers, views, growth, ARRget_post_stats(post_id, publication?)-- views, opens, clicks, shares, conversions for one postget_subscriber_growth(publication?)-- subscriber count over timeget_growth_sources(publication?)-- subscriber growth by source (search, recommendations, direct, social)get_top_posts(limit?, publication?)-- posts ranked by engagementget_earnings(publication?)-- revenue data for paid publicationsget_recommendation_stats(direction?, publication?)-- recommendation network performancecapture_analytics_to_vault(publication?)-- write a weekly analytics snapshot to your vault
git clone https://github.com/adelaidasofia/substack-mcp
cd substack-mcp
pip3 install -r requirements.txt
python3 -c "import server; print('OK')"Copy config.example.json to config.json and fill in your values:
cp config.example.json config.jsonconfig.json is gitignored. Key fields:
| Field | Description |
|---|---|
publications[].name |
Internal name used in tool calls |
publications[].subdomain |
Your Substack subdomain (e.g. yourname) |
publications[].cookie |
Session cookie (see below) |
default_publication |
Which publication to use when publication arg is omitted |
vault_drafts_path |
Path to your vault drafts markdown file |
image_generator.default |
Active image adapter: pillow_local or canva |
Substack does not have a public API with OAuth. Auth uses your browser session cookie.
- Open Chrome and log in to Substack.
- Open DevTools (F12) and go to the Application tab.
- Under Cookies, find
substack.com. - Copy the value of
substack.sid. - Paste it into
config.jsonunder the matching publication'scookiefield.
The raw value works; the server accepts both abc123 and substack.sid=abc123.
Session cookies expire. If tools return {"error": "Auth failed"}, re-extract the cookie.
Add to your .mcp.json (project-scoped) or via claude mcp add -s user:
{
"mcpServers": {
"substack": {
"command": "python3",
"args": ["/path/to/substack-mcp/server.py"]
}
}
}Restart Claude Code after editing .mcp.json. Verify with claude mcp list.
Set vault_drafts_path in config.json to a markdown file in your Obsidian vault.
Format each draft as a section separated by ---. Sections under ## Ready to Post
are surfaced first by list_vault_drafts. Sections under ## Essay Seeds come next.
After publishing, publish_vault_draft moves the draft to a ## Published section
and appends the Substack URL and timestamp.
Visual card generation is handled by the adapter set in config.json under
image_generator.default.
pillow_local (default) -- pure Pillow, no external API. Renders 1080x1080 PNG
cards locally using fonts from the fonts/ directory. Three pillar templates: warm
mustard with optional figure (P1), deep burgundy bold quote (P2), mustard with section
tag (P3). Use with render_card.py as a standalone CLI.
canva -- Canva MCP choreography. Returns a steps list describing the Canva MCP
tool calls needed to clone a template, replace text, replace the figure, and export PNG.
The Claude session executing the playbook follows these steps. Requires Canva MCP
connected in Claude Code and design IDs filled in under image_generator.canva.pillars.
nano_banana -- Gemini 3 Pro Image via the nano-banana skill. Stub in
image_generators/nano_banana.py. Implement generate() by calling the skill's
image endpoint with a prompt built from the card spec.
midjourney -- Midjourney API (or proxy). Stub in image_generators/midjourney.py.
Implement generate() by submitting a prompt, polling for completion, and downloading
the result.
dalle -- OpenAI DALL-E. Stub in image_generators/dalle.py. Implement generate()
using openai.images.generate.
Subclass ImageGenerator from image_generators.base:
from image_generators.base import ImageGenerator
class MyAdapter(ImageGenerator):
@property
def name(self) -> str:
return "my_adapter"
def generate(self, spec: dict) -> dict:
# spec keys: pillar, quote, handle, figure_path, output_path
# return: {status: "ok", adapter: ..., image_path: ..., width: ..., height: ...}
...Register it in image_generators/__init__.py and add a config block under
image_generator.my_adapter in config.json.
The visual_helper.py script handles the deterministic side of the visual queue
pipeline: parsing a Review Queue markdown file, rotating through figures, and
marking entries published. See visual_playbook.md for the full end-to-end flow
including image generation, upload, and publishing steps.
python3 visual_helper.py peek --lang es # get next approved visual item
python3 visual_helper.py mark --lang es ... # mark published + update log
python3 visual_helper.py rotate-figure --pillar P1-
Cookie expiry.
substack.sidcookies expire after a few weeks. Re-extract from Chrome DevTools whentest_connectionreturns auth errors. -
draft_bylinesis required but undocumented.create_draftwill 400 without it. The server fetches youruser_idfrom/user/profile/selfautomatically and injects it into every draft POST. -
Note attachment endpoint requires trailing slash.
POST /comment/attachment/(with slash) works. Without slash it returns 404. The server handles this correctly. -
Publication-scoped vs global endpoints. Notes use
substack.com/api/v1/comment/feed/. Post drafts use{subdomain}.substack.com/api/v1/drafts. Mixing them returns 404 or 403. -
pledgedArrin the dashboard is a projection, not real pledges. Useget_earningsfor actual revenue data. -
Multi-publication support. Pass
publication="name"to any tool to target a specific publication. Omit it to usedefault_publicationfrom config.
MIT