Skill flagged — suspicious patterns detected

ClawHub Security flagged this skill as suspicious. Review the scan results before using.

Ghost Publisher

v1.0.0

Publish markdown articles to any Ghost 5 CMS site. The first reference implementation of Publisher Interface v1 -- a CMS-agnostic contract for editorial work...

0· 64·0 current·0 all-time

Install

OpenClaw Prompt Flow

Install with OpenClaw

Best for remote or guided setup. Copy the exact prompt, then paste it into OpenClaw for machinesofdesire/ghost-publisher.

Previewing Install & Setup.
Prompt PreviewInstall & Setup
Install the skill "Ghost Publisher" (machinesofdesire/ghost-publisher) from ClawHub.
Skill page: https://clawhub.ai/machinesofdesire/ghost-publisher
Keep the work scoped to this skill only.
After install, inspect the skill metadata and help me finish setup.
Required env vars: GHOST_URL, GHOST_ADMIN_API_KEY
Use only the metadata you can verify from ClawHub; do not invent missing requirements.
Ask before making any broader environment changes.

Command Line

CLI Commands

Use the direct CLI path if you want to install manually and keep every step visible.

OpenClaw CLI

Bare skill slug

openclaw skills install ghost-publisher

ClawHub CLI

Package manager switcher

npx clawhub@latest install ghost-publisher
Security Scan
Capability signals
CryptoRequires sensitive credentials
These labels describe what authority the skill may exercise. They are separate from suspicious or malicious moderation verdicts.
VirusTotalVirusTotal
Suspicious
View report →
OpenClawOpenClaw
Benign
high confidence
Purpose & Capability
Name/description, required environment variables (GHOST_URL, GHOST_ADMIN_API_KEY), and included files (publisher.py, tests, INTERFACE.md) all align with a Ghost CMS publishing adapter. The code implements JWT generation, Ghost Lexical conversion, image upload, and the seven declared interface methods — all expected for a Ghost publisher.
Instruction Scope
SKILL.md instructs the agent to read GHOST_URL and GHOST_ADMIN_API_KEY, optionally a JSON config, local markdown files and local image paths or remote image URLs, and to call the Ghost Admin API. It does not instruct reading unrelated system paths or harvesting unrelated environment variables. The CLI examples and Python usage map directly to the published functionality.
Install Mechanism
There is no install spec (instruction-only install); the repository provides source files but nothing is downloaded from arbitrary URLs during install. This reduces install-time risk — code is bundled rather than fetched from third-party servers.
Credentials
The skill requests only GHOST_URL and GHOST_ADMIN_API_KEY, which are appropriate for its function. Note: a Ghost Admin API key is effectively a site-level privileged credential (can create/publish/delete posts and upload media). Although required and expected, this is a powerful secret — treat it with care (use a dedicated integration, rotate keys, avoid using a site owner's universal credential).
Persistence & Privilege
The skill does not request always:true, does not modify other skills or system-wide configs, and is user-invocable; autonomous invocation remains enabled by default but is not combined with other red flags here.
Assessment
This skill appears to do what it says: publishing markdown to Ghost using the Admin API. Before installing, review publisher.py (it is included) to confirm there are no hidden network endpoints beyond your configured GHOST_URL and to understand how images are fetched/uploaded. Create a dedicated Ghost custom integration for this skill rather than using broad or shared credentials, keep the Admin API key secret, and do not commit any config file containing IDs to version control. Because the Admin API key grants publish/delete rights, consider testing in a staging site or with a limited-profile integration first, and rotate the key if you later stop using the skill.

Like a lobster shell, security has layers — review code before you run it.

Runtime requirements

EnvGHOST_URL, GHOST_ADMIN_API_KEY
Primary envGHOST_ADMIN_API_KEY
latestvk970n186dwcbhdtb0rhbp300dn84ze97
64downloads
0stars
1versions
Updated 1w ago
v1.0.0
MIT-0

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 skill
  • wordpress-publisher -- planned
  • substack-publisher -- planned
  • medium-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.com or https://yourpub.ghost.io
  • GHOST_ADMIN_API_KEY -- Admin API key from a Ghost custom integration. Format: <key_id>:<hex_secret>

To get your Admin API key

  1. Go to Ghost Admin -> Settings -> Integrations
  2. Click "Add custom integration"
  3. Name it (e.g. "OpenClaw Publisher")
  4. 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 <post_id> article.md

# 3. Upload and attach a feature image
python3 publisher.py set-image <post_id> https://cdn.example.com/hero.jpg --alt "Hero image" --upload

# 4. Schedule it, or publish it now
python3 publisher.py schedule <post_id> 2026-05-01T09:00:00Z
python3 publisher.py publish <post_id> --newsletter

# Other interface methods
python3 publisher.py get <post_id>
python3 publisher.py delete <post_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="<your_key_id>:<your_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: # H1 through #### 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.

Comments

Loading comments...