From compound-engineering
Drafts user-facing launch or promotion copy for a shipped feature, including social posts and changelog blurbs.
How this skill is triggered — by the user, by Claude, or both
Slash command
/compound-engineering:ce-promote [optional: what shipped and/or channels, e.g. 'a tweet thread and a LinkedIn post'][optional: what shipped and/or channels, e.g. 'a tweet thread and a LinkedIn post']The summary Claude sees in its skill listing — used to decide when to auto-load this skill
Turn a feature that just shipped into copy-pasteable, user-facing announcement copy — right inside the engineering workflow.
Turn a feature that just shipped into copy-pasteable, user-facing announcement copy — right inside the engineering workflow.
After you ship, the messaging shouldn't wait for a separate marketing pass. ce-promote figures out what shipped, picks the right channels, and drafts the copy. It is spiral-agnostic by default: with nothing installed it draws on a lite layer of editorial and social-media expertise to produce strong channel-specific copy. When the Spiral CLI (see references/spiral-cli.md) is present and authed, it uses Spiral so the drafts are voice-matched to your brand — a subtle enhancement, never a requirement.
This skill drafts only. It never posts, publishes, commits, or opens PRs. Posting is a human action. The output is always drafts for you to review, edit, and ship yourself.
/ce-promote # Derive what shipped from context, draft defaults
/ce-promote [free-form description] # You describe what shipped
/ce-promote a tweet thread and a LinkedIn post # Request specific channels
/ce-promote 3 tweet options for the new export feature
If the user gave a free-form description of the feature, use it as the source of truth.
Otherwise, derive it from context (use what's available; don't block on any one source):
gh pr view --json title,body,url 2>/dev/null (and gh pr view for the current branch). The title and body usually state the user-facing value.git diff main...HEAD --stat and skim notable changes to ground the claim in what actually changed.[Unreleased] entry in docs/changelog.md, CHANGELOG.md, or similar.git log --oneline -15 for the arc of the change.Then write a 1–3 sentence summary of the user-facing value — what a user can now do that they couldn't before, and why they'd care. Describe the outcome, not the implementation. ("You can now export any report to CSV in one click" — not "Added a CsvSerializer and an export endpoint.")
If you can't confidently tell what shipped, ask the user one short question rather than guessing.
Default to a small, sensible set:
Scale to what the change warrants and to what the user asked for. If they named channels ("LinkedIn", "email", "a blog intro", "a short demo script"), draft those instead of or in addition to the defaults. A small fix needs one or two short drafts; a flagship feature can justify a cross-channel set. Don't force a fixed template.
First, detect Spiral's state with two quick, non-blocking commands:
which spiral
spiral auth status --json 2>/dev/null
Classify into one of three states:
which spiral finds nothing) → Path 0 (install), then Path A if set up, else Path B.spiral auth status --json:
"authenticated": true (equivalently "status": "authenticated") → Path A (voice-matched)."authenticated": false → Path 0, then Path A if the user signs in, else Path B.--json on auth status), fall back to the legacy signal in that same output: ready iff it contains spiral_sk_, else unauthed.Never let a Spiral failure, timeout, or odd output block or slow the skill — when in doubt, treat it as not-ready and continue.
When Spiral isn't ready, offer to set it up once — unless the user previously opted out. The point is one proactive nudge, never a recurring one, and never a blocker: a decline always proceeds to Path B. Any dismissal records the opt-out, so a single first-run decline stops the offer for good in this repo — the user is never asked twice.
Read references/spiral-cli.md for the exact setup prompt (built with the platform's blocking-question tool), the connect/install steps, and how the opt-out is recorded so later runs skip this. In short:
spiral login --json (CLI >= 1.8.0; non-blocking, the API key never passes through the agent). On status: already_authenticated → use Path A. On status: pending → surface the auth_url, the user approves in their browser, then poll spiral auth status --json until authenticated: true → Path A. Never have the user paste a key into chat. (Older CLI without agent login → suggest npm i -g @every-env/spiral-cli@latest, or have the user run spiral login themselves.) Escape hatch: "or the agent can just draft directly, without Spiral's personalization and humanization."Skip Path 0 entirely — straight to Path B — when the opt-out is already recorded, or when running headless / non-interactive (no human to answer). If a human is present but no blocking-question tool is available, do not skip — fall back to a numbered list of the two options in chat and wait for a reply (per the Ask section of references/spiral-cli.md).
Use the Spiral CLI so drafts match the user's brand voice. Read references/spiral-cli.md before composing the prompt — multi-channel vs. single-channel-variations is phrasing-driven (channel keywords / cue words vs. --num-drafts) and getting it wrong silently returns the wrong number or shape of drafts. The exact phrasing rules live there; don't restate them from memory. Essentials:
--instant and --json. Parse drafts[] (each carries its own channel) plus session_id.channel. Spiral decides how many drafts per channel — multi-channel runs often return several per channel — so never assume one-per-channel or drop extras.If the spiral write call errors or returns no usable drafts, silently fall back to Path B for the affected channels.
No Spiral needed — draft strong copy directly using a compact layer of editorial and social-media fundamentals. (The Spiral path goes further: brand-voice matching, humanization, saved styles, and cross-channel campaign orchestration.)
Editorial fundamentals — every channel:
Social fundamentals — distributed channels:
Per channel:
Drafts per channel: one strong draft by default; produce more only when asked ("3 tweet options"), capped ~3.
Show every draft as a clean, copy-pasteable block, labeled by channel. For each:
### X post
<the copy>
session_id and each draft's url so the user can open and tweak them in the Spiral web app.Single-channel variations — "3 tweet options":
User:
/ce-promote 3 tweet options for the new one-click CSV export→ Summarize the value. Spiral path:spiral write "3 tweet options for one-click CSV export" --instant --num-drafts 3 --json(no cue words). No-Spiral path: write 3 distinct tweets directly. Present all three.
Multi-channel set — "a campaign across X, LinkedIn, and email":
User:
/ce-promote draft a launch across X, LinkedIn, and email→ Spiral path:spiral write "announcing one-click CSV export — a launch across X, LinkedIn, and email" --instant --jsonreturns a set of drafts per channel (Spiral decides the count — often several), each carrying itschannel. (--num-draftsignored here.) No-Spiral path: draft one X post, one LinkedIn post, one email directly. Present every returned draft, grouped by channel.
npx claudepluginhub everyinc/compound-engineering-plugin --plugin compound-engineeringGenerates platform-specific social media copy from code changes, marketing briefs, blog posts, or feature descriptions. Tailors copy to each platform's rules and audience.
Use this skill when the user asks to "write a launch announcement", "announce this feature", "write a launch email", "product announcement", "release notes", "launch blog post", "feature announcement to customers", "write the Slack announcement for this launch", or is preparing to communicate a new feature or product to users, customers, or the public.
Takes product update descriptions and auto-generates a dated changelog entry plus a content package with tweet, LinkedIn post, email snippet, and one-liner.