linkedin-post-generator
LinkedIn Post Generator
Generate LinkedIn posts from shared source material, written in each user's personal style.
How It Works
- Personal style profile — stored locally at
~/.config/casper/linkedin-style.md(never committed) - Source config — stored locally at
~/.config/casper/linkedin-sources.md(never committed) - Shared source material — meeting transcripts, Slack dumps, docs in
source-material/ - Prompt template — extraction rules, voice guidelines, few-shot examples in
references/prompt-template.md
First Run: Style Setup
Check if ~/.config/casper/linkedin-style.md exists.
If it does NOT exist, run the style setup flow:
- Say: "Welcome to the LinkedIn Post Generator! Before we start, I need to understand your writing style."
- Say: "Share 3 LinkedIn posts that match the style you want. You can either paste the post links (e.g.
https://linkedin.com/posts/...) or paste the text directly." - Wait for the user to provide 3 posts
- If the user provides LinkedIn URLs, fetch the post content using the apify-scrapers skill:
Extract the post text from the JSON output. If a URL fails to fetch, ask the user to paste that post's text instead.python ${CLAUDE_PLUGIN_ROOT}/skills/apify-scrapers/scripts/scrape_linkedin_posts.py search "{url}" --max-posts 1 - Analyze the posts for: tone, sentence length, vocabulary, formatting habits, hook style, CTA style, use of questions, paragraph length, overall energy
- Create
~/.config/casper/directory if it doesn't exist - Save the analysis to
~/.config/casper/linkedin-style.mdusing this format:
# LinkedIn Style Profile
Generated: [date]
## Tone
[analysis]
## Structure Patterns
[paragraph length, line breaks, formatting habits]
## Hook Style
[how they open posts]
## CTA / Closing Style
[how they end posts — questions, challenges, etc.]
## Vocabulary & Phrases
[distinctive phrases, word choices, energy level]
## Sample Posts
[the 3 original posts, for reference]
- Confirm: "Got it! Your style profile is saved. You can update it anytime with
/casper:generate-linkedin-post --setup"
First Run: Source Material Check
After style setup completes (or if style exists but source-material/ is empty), check for source material:
- Check if
source-material/contains any.mdfiles besidesREADME.md - If empty, guide the user:
- Say: "You don't have any source material yet. I need content to generate posts from — meeting transcripts, notes, Slack conversations, etc."
- Present options:
- "Connect integrations" — run the
--setup-sourcesflow to configure Fireflies, Slack, or Google Drive auto-pulling - "Paste something manually" — run the
--add-sourceflow to let the user paste a transcript, notes, or other content
- "Connect integrations" — run the
- Wait for user choice and proceed with the selected flow
- If source material exists, proceed with generation
Normal Run: Post Generation
If style config exists and source material is available, proceed with generation:
- Read
~/.config/casper/linkedin-style.md - Read ALL files in
${CLAUDE_PLUGIN_ROOT}/skills/linkedin-post-generator/source-material/(excluding README.md) - Read
${CLAUDE_PLUGIN_ROOT}/skills/linkedin-post-generator/references/prompt-template.md - Apply the confidentiality rules from the prompt template (no financials, no client names, no pipeline, no team member names)
- Generate 2-4 post options based on the source material, written in the user's personal style
- Present them in a clean, copy-paste-ready format
Flags
| Flag | Behavior |
|---|---|
| (none) | Normal generation flow |
--setup |
Re-run style setup, overwrite existing config |
--setup-sources |
Configure which Fireflies, Slack, and Drive sources to pull from |
--refresh |
Pull fresh source material from configured integrations, then generate |
--view-style |
Read and display ~/.config/casper/linkedin-style.md |
--view-sources |
List and summarize all files in source-material/ |
--add-source |
Prompt user to paste new content, save as new .md file in source-material/ |
Flag Details
--setup-sources
Interactive setup for automatic source pulling. Read references/source-integrations.md for full details.
- Ask: "What's your work email address? This is used to filter transcripts to only meetings you attended."
- Save as
user_emailin the config
- Save as
- Ask: "Which sources do you want to connect?" Present options:
- Fireflies.ai — pulls meeting transcripts (needs
FIREFLIES_API_KEYenv var) - Slack — pulls messages from channels (needs
SLACK_BOT_TOKENenv var) - Google Drive — pulls docs and transcripts (needs OAuth setup via google-workspace skill)
- Fireflies.ai — pulls meeting transcripts (needs
- For each selected source, check if the required env var / credentials exist. If missing, provide setup instructions:
- Fireflies: "Set
FIREFLIES_API_KEYin your environment. Get your API key from https://app.fireflies.ai/api" - Slack: "Set
SLACK_BOT_TOKENin your environment. Create a Slack app at https://api.slack.com/apps" - Google Drive: "Run the google-workspace skill setup to configure OAuth."
- Fireflies: "Set
- For each enabled source, gather configuration:
- Fireflies: search terms (or leave empty for all recent) and days_back
- Slack: which channels to pull from — list available channels if possible, otherwise ask the user
- Google Drive: search terms, days_back
- Save to
~/.config/casper/linkedin-sources.md - Confirm: "Source config saved. Run
/casper:generate-linkedin-post --refreshto pull fresh content."
--refresh
Pull fresh source material from all configured integrations before generating posts. Read references/source-integrations.md for the full integration workflow.
Summary of the flow:
- Read
~/.config/casper/linkedin-sources.md— if missing, run--setup-sourcesfirst - For each enabled source, call the existing Casper skill scripts:
- Fireflies:
python ${CLAUDE_PLUGIN_ROOT}/skills/transcript-search/scripts/fireflies_transcript_search.py "{term}" --days-back {N} --content --json- After fetching, filter results to only transcripts where
user_email(from source config) appears in the transcript'sparticipantsarray
- After fetching, filter results to only transcripts where
- Slack:
python ${CLAUDE_PLUGIN_ROOT}/skills/slack-automation/scripts/slack_search.py read "{channel}" --days {N} - Google Drive:
python ${CLAUDE_PLUGIN_ROOT}/skills/google-workspace/scripts/gdrive_search.py files "{term}" --modified-days {N} --json
- Fireflies:
- Convert JSON output to clean markdown and save to
source-material/:- Fireflies:
fireflies-{YYYY-MM-DD}-{title-slug}.md - Slack:
slack-{channel}-{YYYY-MM-DD}.md - Google Drive:
gdrive-{title-slug}-{YYYY-MM-DD}.md
- Fireflies:
- Proceed with normal generation
--view-style
Read ~/.config/casper/linkedin-style.md and display it. If it doesn't exist, say "No style profile found. Run /casper:generate-linkedin-post --setup to create one."
--view-sources
List all .md files in ${CLAUDE_PLUGIN_ROOT}/skills/linkedin-post-generator/source-material/ (excluding README.md). For each file, show the filename and a 1-line summary of its contents.
--add-source
- Ask: "Paste the content you want to add as source material (meeting transcript, Slack dump, notes, etc.)"
- Ask: "What should I name this source file? (e.g.,
team-standup-jan-2025)" - Save as
${CLAUDE_PLUGIN_ROOT}/skills/linkedin-post-generator/source-material/[name].md - Confirm: "Source material saved. It will be included in future post generation."
Reference Files
| File | When to Read |
|---|---|
references/prompt-template.md |
Every generation run — contains voice rules, few-shot examples, confidentiality rules |
references/source-integrations.md |
When running --refresh or --setup-sources — contains script paths, arguments, output conversion |
references/style-setup.md |
When running --setup — contains analysis framework for style profiling |
source-material/*.md |
Every generation run — raw content to extract post ideas from |