medium-ops
Provides tools for reading and writing to Medium, including managing stories, responses, claps, feed, profiles, and stats, with a reply engine and MCP-native drafting loop.
medium-ops
Standalone Medium CLI + 22-tool MCP server. Your IDE drafts the replies. Zero AI API keys.
Stories, responses, claps, feed, profiles, stats, reply engine, MCP server. One Python install, one binary, MIT licensed. Sibling of substack-ops.
TL;DR — MCP-native (no API key, one command)
uvx medium-ops mcp install cursor # or claude-desktop, claude-code, print
# Restart your host. Then in chat:
# "list unanswered responses on post abc123def456"
# "draft a warm reply to response r1"
# "post that draft"Your host's LLM (Cursor's, Claude's) does the drafting via the
propose_reply / confirm_reply tools. No ANTHROPIC_API_KEY /
OPENAI_API_KEY needed.
Why a hybrid
Medium exposes three usable surfaces and we use all of them:
Public RSS (reads, no auth).
medium.com/feed/@{user}returns the author's ~10 most recent stories withbody_html,pubDate,tags, hero image, anddc:creator. Zero credentials, faster than GraphQL, stable. Used by default forlist_posts/get_post/get_post_content. Inspired by Portfolio_V2's blog page.Dashboard GraphQL (authenticated reads).
medium.com/_/graphql+medium.com/_/api/*with thesidcookie. Used as a fallback when you ask for more than ~10 posts, when the post isn't in the RSS window, or for things RSS can't give you (responses, claps, feed, stats, search).Official REST (writes).
api.medium.com/v1/*with an Integration Token. SupportscreatePost,createPostInPublication,getUser,getPublications. That's it.
Force a specific transport with --source rss|graphql|auto on posts list,
posts show, and posts content. The dashboard + GraphQL endpoints are
undocumented and Medium can change them at any time. See
Known gaps.
Setup (dev / from source)
git clone https://github.com/06ketan/medium-ops && cd medium-ops
uv sync
uv sync --extra mcp # mcp SDK for the MCP server (recommended)
uv sync --extra tui # textual for the TUIAuth is read from ~/.cursor/mcp.json's mcpServers.medium-ops.env (or
medium-api / medium). Override with env or .env.
uv run medium-ops auth verify
uv run medium-ops quickstartCommand surface
Every write defaults to --dry-run. Flip with --no-dry-run. All writes
land in .cache/audit.jsonl and are dedup-checked against
.cache/actions.db.
Auth (3)
Command | What it does |
| Probe both integration token (/me) and sid cookie (GraphQL Viewer). |
| Same but exits non-zero on failure (CI-friendly). |
| Interactive: paste token / sid / uid / username to |
Read — Stories (5)
Command | What it does | ||
| Latest stories by a user (default: self). | ||
| Story metadata (title, clap count, response count). | ||
| Body HTML (or Markdown with | ||
| Medium-side full-text search. | ||
`posts publish -t "..." -f body.md [--pub] [--status draft | public | unlisted]` | Publish via integration token. |
Read + Write — Responses (3)
Command | What it does |
| Top-level responses table. |
| Full response + reply tree JSON. |
| Post a response or reply. |
Read + Write — Claps (2)
Command | What it does |
| Total claps. |
| Clap 1-50 times. Dedup-protected. |
Read — Discovery + Profile (5)
Command | What it does |
| Reader feed. |
| Your full profile (GraphQL). |
| Any user's public profile. |
| Per-post views / reads / fans (dashboard scrape). |
| Publications you can publish to (integration token). |
Reply engine (3)
Command | What it does |
| Rule-based replies (no LLM). |
| Draft every response to a file. |
| Post only |
Operations + safety (2)
Command | What it does |
| Query the JSONL audit log. |
| Counts in the dedup SQLite DB. |
MCP server (3)
Command | What it does |
| Auto-merge config into your host. |
| stdio MCP server (22 tools). |
| Print the tool registry. |
Other (1)
Command | What it does |
| Print a quickstart checklist. |
Reply modes
Mode | What it does | Safety |
| YAML keyword rules under | dry-run default |
| LLM drafts every response to | offline review, dedup-checked on send |
| Posts only items with | dry-run default; dedup DB prevents dup replies |
MCP | Host LLM drafts, you approve per-item, token-gated | 5-min token TTL, idempotent, no API key |
MCP server
medium-ops mcp install cursor # auto-add to ~/.cursor/mcp.json
medium-ops mcp install claude-desktop # auto-add to claude_desktop_config.json
medium-ops mcp install claude-code # uses `claude mcp add`
medium-ops mcp install print # print the snippet only
medium-ops mcp serve # stdio server
medium-ops mcp list-tools # 22 toolsManual config snippet:
{
"mcpServers": {
"medium-ops": {
"command": "medium-ops",
"args": ["mcp", "serve"]
}
}
}If the mcp SDK is not installed, the server falls back to a minimal
stdin/stdout JSON-line dispatcher:
echo '{"tool":"list_posts","args":{"limit":3}}' | medium-ops mcp serveMCP-native draft loop (no API key)
The safety + drafting stack that makes the unattended mode safe:
Tool | What it does |
| Worklist — responses where you haven't replied. |
| Dry-run only. Returns a |
| Posts the staged reply by token. Idempotent via dedup DB. Token TTL 5 min. |
| File-based offline review loop. |
| Read the audit log + dedup counts. |
LLM strategy
Two layers, both free:
MCP-native (default). Host LLM drafts via
propose_reply/confirm_reply. No env vars, no API key. Use this for interactive replies.Subprocess CLI (daemon path). For
reply bulkwhen no human is in the loop. Auto-detectsclaude(Claude Code),cursor-agent, orcodexon PATH. Override withMEDIUM_OPS_LLM_CMD.
There is no paid-API-key path.
Auth setup
Medium has two auth layers that map to different feature surfaces:
Integration Token —
Authorization: Bearer <token>. Used againstapi.medium.com/v1/*. Gets you:publish_post,list_own_publications. Token generation at https://medium.com/me/settings → "Integration tokens". Note: Medium stopped issuing new tokens in 2023. If you never generated one, the write path will 401 and you'll have to use the sid-cookie response path for any writes.sid cookie — from
medium.com(Application → Cookies →sid). Used againstmedium.com/_/graphqlandmedium.com/_/api/*. Gets you: all reads (stories, responses, claps, feed, stats, profile), plusclap_postandpost_response(fragile — undocumented).
medium-ops auth verify
medium-ops auth test
medium-ops auth setup
medium-ops auth har ./medium.har # ingest a Chrome devtools HAR exportRefreshing auth from a HAR
When cookies rotate or Medium changes a GraphQL schema, the fastest fix is:
Open
medium.comin Chrome with devtools → Network panel.Reproduce the failing action (publish a draft, post a response, etc.).
Right-click any request → "Save all as HAR with content".
medium-ops auth har ./medium.har
This:
merges fresh
sid,uid,xsrf,cf_clearancecookies into.env(preserving everything else)writes a redacted snapshot to
.cache/har-snapshot.jsonlisting every Medium GraphQL operation observed plus its request-variable / response-data key shapes — useful for diffing against the queries hard-coded inclient.pyto spot schema drift before users hit it.
Env vars (or ~/.cursor/mcp.json → mcpServers.medium-ops.env):
MEDIUM_INTEGRATION_TOKEN=2fb00... # optional, for writes
MEDIUM_SID=1:... # optional, for reads
MEDIUM_UID=... # optional
MEDIUM_USERNAME=yourhandle # optional but recommendedArchitecture
mcp.json | env → auth.py
│
MediumConfig (token? sid? uid? username?)
│
MediumClient (httpx)
┌───────┼──────────┐
▼ ▼ ▼
api.medium.com medium.com/ medium.com/_/
/v1/* (REST) _/graphql api/* (dashboard)
│ │ │
Bearer token sid cookie sid cookie
│
┌──────┬──────┬────────────┬──────┬────────┬──────────┐
▼ ▼ ▼ ▼ ▼ ▼ ▼
posts responses claps profile stats feed reply_engine
│
┌───────────────────┼───────────────┐
▼ ▼ ▼
template ai_bulk MCP propose/confirm
└───────────────────┬───────────────┘
▼
base.post_response
│
┌─────────┼─────────┐
▼ ▼ ▼
dedup audit dry_run
(SQLite) (jsonl)
mcp/server.py ──── 22 tools ─── all share MediumClientEndpoints used
Action | Method + URL |
Auth: integration token |
|
Auth: sid cookie |
|
User profile |
|
List stories |
|
Story metadata |
|
Story body |
|
Story search |
|
Responses |
|
Feed |
|
Publish story |
|
Publish to pub |
|
Own pubs |
|
Clap |
|
Post response |
|
Stats |
|
Tests
uv run pytest -qCoverage: auth loading, client transports + XSSI stripping, dedup DB, audit log search, MCP tool registry + dispatcher, MCP install host-config merging, propose/confirm flow + token expiry, reply-engine template matching + dedup+audit flow, subprocess LLM detection.
Known gaps
Medium stopped issuing new Integration Tokens in 2023. If you never got one,
publish_post/list_own_publicationswill 401. The read + response + clap paths still work via sid. The RSS read path needs no credentials at all.RSS is capped at ~10 posts and lacks clap/response counts and stats. When you need more, pass
--source graphql(requires sid) or set--limit > 10and the client will auto-fall back to GraphQL.GraphQL operation names and schemas change silently. The queries in
client.pymirror what the dashboard uses today — expect breakage every couple of months. Pin this package's version.post_responseuses GraphQLsavePostResponse(deltas: [Delta!]!, inResponseToPostId: ID!). Delta shape is{type: 1, index: N, paragraph: {type: 1, text, markups: []}}(type=1 means insert; paragraph.type=1 is P). Reverse-engineered from error messages.update_draft_contentuses dashboardPOST /p/{id}/deltaswith{baseRev, rev, deltas}. For a brand-new draft,baseRev=-1, rev=0. Subsequent edits should bump both.clap_poststill uses the undocumented/_/api/posts/{id}/clapshape; not yet re-validated against the new GraphQL surface. Dry-run first.Members-only stories return a paywall preview unless the
sidbelongs to a paying member.No "restack" equivalent. Medium doesn't have reshares; the closest is a clap + a response. Use
clap_post+post_responsetogether for that.No notes / short-form. Medium killed short-form in 2018.
Chrome cookie auto-grab (the
auth_chromeflow from substack-ops) is not yet implemented. Paste yoursidinto.envfor now.TUI not yet implemented; the extras pin is there for future work.
License
MIT. See LICENSE.
Maintenance
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/06ketan/medium-ops'
If you have feedback or need assistance with the MCP directory API, please join our Discord server