Ghost Publisher
/install ghost-publisher
Ghost Publisher
Publish markdown articles to any Ghost 5 CMS site. The first reference implementation of Publisher Interface v1 -- a CMS-agnostic contract for editorial workflows that makes "change CMS" a config decision, not a rewrite.
Companion skills (all implement the same interface, published separately):
ghost-publisher-- this skillwordpress-publisher-- plannedsubstack-publisher-- plannedmedium-publisher-- planned
About the interface
This skill implements Publisher Interface v1 in full. The interface spec --
the standard post object schema and the seven required methods -- lives in
INTERFACE.md at the skill root. Read that file if you are
building a calling agent or another adapter skill. Everything below this
section is Ghost-specific usage.
The short version: your code (or an editorial workflow orchestrator) calls
publisher.createPost(post), publisher.publishPost(id, opts) and so on.
The adapter translates the standard post object into Ghost-shaped payloads,
handles JWTs, converts markdown to Ghost Lexical, maps author IDs to Ghost
staff members, and uploads images. None of that leaks into caller code.
Setup
Required environment variables
GHOST_URL-- your Ghost site URL, e.g.https://yoursite.comorhttps://yourpub.ghost.ioGHOST_ADMIN_API_KEY-- Admin API key from a Ghost custom integration. Format:\x3Ckey_id>:\x3Chex_secret>
To get your Admin API key
- Go to Ghost Admin -> Settings -> Integrations
- Click "Add custom integration"
- Name it (e.g. "OpenClaw Publisher")
- Copy the Admin API Key -- it is a 24-char key_id, a colon, and a 64-char hex secret (format:
XXXXXXXXXXXXXXXXXXXXXXXX:XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX)
Optional config file (GHOST_PUBLISHER_CONFIG)
For non-secret settings (default author, agent-to-staff mapping, newsletter
ID, max image size), point GHOST_PUBLISHER_CONFIG at a JSON file:
{
"default_author_id": "ghost_staff_id",
"agent_author_map": {
"staff_1": "ghost_staff_id_1",
"staff_2": "ghost_staff_id_2"
},
"newsletter_id": "ghost_newsletter_id",
"max_image_size_mb": 2
}
The agent_author_map lets a caller pass {"author": "staff_1"} and have
the adapter resolve it to the right Ghost staff member without the caller
needing to know internal Ghost IDs. Omit the map and everything falls back
to default_author_id, or to the integration's default if that is also
unset.
Keep this file out of version control if it contains IDs you do not want public. No secret keys go in this file -- secrets come from env vars.
Usage
As a Python module (recommended for agents)
from publisher import GhostPublisher
pub = GhostPublisher() # reads env vars
post_id = pub.createPost({
"title": "A quiet argument about obsolescence",
"excerpt": "Things that still run but no longer serve a purpose.",
"body_md": open("article.md").read(),
"tags": ["culture", "criticism"],
"author": "staff_1",
})
hosted = pub.uploadImage("https://cdn.example.com/hero.jpg", alt="Booth interior")
pub.updatePost(post_id, {"featured_image_url": hosted, "image_alt_text": "Booth interior"})
url = pub.publishPost(post_id, {"send_newsletter": True})
print(f"Live at: {url}")
All seven Publisher Interface v1 methods are available on the
GhostPublisher instance: createPost, updatePost, publishPost,
schedulePost, deletePost, uploadImage, getPost.
As a CLI
Full pipeline (most common)
python3 publisher.py create-publish article.md \
--title "Your Article Title" \
--excerpt "A 1-2 sentence summary for SEO and previews" \
--tags "ai,economics,analysis" \
--image-url "https://cdn.example.com/header.jpg" \
--image-alt "Description of the image" \
--upload-image \
--newsletter
Remove --newsletter to publish without emailing subscribers. Remove
--upload-image if the image URL is already on a CDN you trust and you
don't need Ghost's media store to host it.
Step-by-step
# 1. Create a draft (returns post_id)
python3 publisher.py create-draft --title "Title" --excerpt "Summary" --tags "tag1,tag2"
# 2. Inject markdown body
python3 publisher.py update-content \x3Cpost_id> article.md
# 3. Upload and attach a feature image
python3 publisher.py set-image \x3Cpost_id> https://cdn.example.com/hero.jpg --alt "Hero image" --upload
# 4. Schedule it, or publish it now
python3 publisher.py schedule \x3Cpost_id> 2026-05-01T09:00:00Z
python3 publisher.py publish \x3Cpost_id> --newsletter
# Other interface methods
python3 publisher.py get \x3Cpost_id>
python3 publisher.py delete \x3Cpost_id>
python3 publisher.py upload-image ./local.jpg --alt "alt text"
Worked example
You are publishing a short essay to a Ghost Pro site with a custom integration named "OpenClaw Publisher".
1. Environment
export GHOST_URL="https://yourpub.ghost.io"
export GHOST_ADMIN_API_KEY="\x3Cyour_key_id>:\x3Cyour_hex_secret>"
2. Write article.md
# A quiet argument about obsolescence
Some machines keep running long after they stop serving anyone. The
projector still warms up. The booth still smells of amber lamp and dust.
There is no reel threaded. There is no audience.
## What this piece is about
A meditation on continuity without function -- why certain systems outlive
the problem they were built to solve.
3. Run the pipeline
python3 publisher.py create-publish article.md \
--title "A quiet argument about obsolescence" \
--excerpt "What certain systems look like after their purpose has gone." \
--tags "culture,essay" \
--image-url "https://cdn.example.com/booth.jpg" \
--image-alt "Close-up of a projector booth interior, no reel loaded" \
--upload-image
4. Output
[ok] Draft created: 69d12a34ec4a400445c217ce
[ok] Image uploaded: https://yourpub.ghost.io/content/images/2026/04/booth.jpg
[ok] Feature image set
[ok] Published: https://yourpub.ghost.io/quiet-argument-about-obsolescence/
That is the full pipeline. The same sequence, called from an editorial workflow agent against any Publisher Interface v1 adapter, would produce the same outcome on any CMS.
Markdown formatting supported
- Headings:
# H1through#### H4 - Bold, italic, bold+italic
inline code- Links
- Bullet lists (
-,*,+) - Numbered lists (
1.,2.) - Blockquotes (
>) - Horizontal rules (
---)
The skill converts markdown to Ghost 5 Lexical JSON internally. You never touch the Lexical format directly.
Ghost stores the article title separately from the body. If your markdown
file starts with a # Title line, it is stripped automatically before
injection -- the title you pass via --title (or the title field in a
createPost call) is the one that wins.
Editorial standards
Ghost optimistic concurrency. Every write fetches updated_at first
and passes it back to Ghost. If another writer modifies the post between
your read and your write, Ghost rejects the write. The adapter surfaces
the underlying error -- do not retry blindly.
JWT expiry. JWTs are generated fresh for each API call with a 5-minute expiry. Long-running batch operations do not need to manage tokens.
Image size cap. uploadImage rejects files over the configured
max_image_size_mb (default 2MB) before hitting Ghost. Ghost Pro also has
its own cap; if your image clears this check but Ghost rejects it, raise
the ticket with Ghost Pro support or compress the image first.
Tags. Pass tags as plain strings. The adapter asks Ghost to create any that do not yet exist. Ghost slugs them automatically.
Author mapping. If agent_author_map is configured, pass the
caller-side identifier (e.g. "staff_1") as author. If it is not
configured, the adapter falls back to default_author_id, then to the
integration's default author. An unknown author does not error -- it
silently falls back. Design your workflow to audit bylines post-publish if
that matters to you.
What this skill is not
- Not a headless CMS. Ghost is the CMS; this skill is the plumbing to get content into it.
- Not a markdown renderer. It converts markdown into Ghost Lexical so Ghost can render it. The output lives in Ghost, not here.
- Not Ghost 4.x compatible. Ghost 4 used Mobiledoc; this skill targets Ghost 5's Lexical format.
- Not an editorial workflow. Selecting images, writing copy, scheduling cadence -- that belongs to the calling agent or workflow. This skill does the publish step and does it well.
License
MIT-0 (MIT No Attribution). Use, modify, redistribute, ship commercially.
No attribution required. See LICENSE.
- 确保已安装 OpenClaw(本地或 Docker 部署)
- 在对话框中输入安装命令:
/install ghost-publisher - 安装完成后,直接呼叫该 Skill 的名称或使用
/ghost-publisher触发 - 根据 Skill 的参数说明提供必要输入,即可获得结构化输出
Ghost Publisher 是什么?
Publish markdown articles to any Ghost 5 CMS site. The first reference implementation of Publisher Interface v1 -- a CMS-agnostic contract for editorial work... 它是一个面向 Claude Code / OpenClaw 的 AI Agent Skill 插件,目前累计下载 71 次。
如何安装 Ghost Publisher?
在 OpenClaw 或 Claude Code 对话框中运行命令「/install ghost-publisher」即可一键安装,无需额外配置。
Ghost Publisher 是免费的吗?
是的,Ghost Publisher 完全免费,采用 MIT-0 许可证,可自由下载、安装和使用。
Ghost Publisher 支持哪些平台?
Ghost Publisher 跨平台运行,可在任意部署了 OpenClaw / Claude Code 的环境中使用(cross-platform)。
谁开发了 Ghost Publisher?
由 MachinesOfDesire(@machinesofdesire)开发并维护,当前版本 v1.0.0。