Claude-skill-inception ghost-admin-api-html-source
install
source · Clone the upstream repo
git clone https://github.com/strataga/claude-skill-inception
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/strataga/claude-skill-inception "$T" && mkdir -p ~/.claude/skills && cp -r "$T/ghost-admin-api-html-source" ~/.claude/skills/strataga-claude-skill-inception-ghost-admin-api-html-source && rm -rf "$T"
manifest:
ghost-admin-api-html-source/SKILL.mdsource content
Ghost Admin API HTML Source Parameter
Problem
When creating posts via the Ghost Admin API with HTML content, posts are created successfully (201 response, post ID returned) but the HTML content is empty. Only the title appears in the Ghost editor.
Context / Trigger Conditions
- POST to
returns 201 success/ghost/api/admin/posts/ - Post is created with correct title
- HTML content in response/editor is empty (0 chars)
- You're sending
field in the JSON bodyhtml - You may have tried adding
to the JSON body (this doesn't work)source: "html"
Root Cause
Ghost's Admin API requires
source=html as a query parameter, not in the JSON
request body. The Ghost JavaScript SDK handles this automatically, but raw HTTP
implementations need to add it manually.
The JS SDK call:
api.posts.add({ title, html }, { source: "html" })
Translates to the raw API as:
POST /ghost/api/admin/posts/?source=html
Solution
Correct Approach (Query Parameter)
POST /ghost/api/admin/posts/?source=html Content-Type: application/json Authorization: Ghost {jwt_token} { "posts": [{ "title": "My Post", "html": "<h2>Content here</h2><p>This will work!</p>", "status": "draft" }] }
Incorrect Approach (Body Parameter - Does NOT Work)
POST /ghost/api/admin/posts/ Content-Type: application/json { "posts": [{ "title": "My Post", "html": "<p>Content</p>" }], "source": "html" // <-- This is IGNORED! }
Implementation Examples
Python (requests):
url = f"{ghost_url}/ghost/api/admin/posts/?source=html" response = requests.post(url, json={"posts": [post_data]}, headers=headers)
Rust (reqwest):
let url = format!("{}/ghost/api/admin/posts/?source=html", base_url); client.post(&url).json(&request_body).send().await?;
For updates (PUT), also use the query parameter:
PUT /ghost/api/admin/posts/{id}/?source=html
Verification
After creating a post, fetch it back and check the HTML length:
get_url = f"{ghost_url}/ghost/api/admin/posts/{post_id}/?formats=html" response = requests.get(get_url, headers=headers) html = response.json()["posts"][0].get("html", "") print(f"HTML length: {len(html)}") # Should be > 0
Example
Before fix:
Created post: 6970a142... HTML length: 0 # Empty!
After fix:
Created post: 6970a3f0... HTML length: 938 # Content preserved!
Notes
- This applies to both POST (create) and PUT (update) operations
- The
query parameter on GET requests is separate - that's for specifying the return formatformats=html - Ghost internally converts HTML to its mobiledoc/lexical format for storage
- The conversion is "lossy" - some HTML elements may not be preserved exactly
- For complex HTML, consider using mobiledoc format directly