@m13v/s4l 1.6.197-rc.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (326) hide show
  1. package/README.md +143 -0
  2. package/SKILL.md +342 -0
  3. package/bin/cli.js +980 -0
  4. package/bin/cookie-helper.js +315 -0
  5. package/bin/platform.js +59 -0
  6. package/bin/scheduler/index.js +12 -0
  7. package/bin/scheduler/launchd.js +518 -0
  8. package/browser-agent-configs/all-agents-mcp.json +68 -0
  9. package/browser-agent-configs/linkedin-agent-mcp.json +16 -0
  10. package/browser-agent-configs/linkedin-agent.json +17 -0
  11. package/browser-agent-configs/linkedin-harness-mcp.json +21 -0
  12. package/browser-agent-configs/reddit-agent-mcp.json +16 -0
  13. package/browser-agent-configs/reddit-agent.json +17 -0
  14. package/browser-agent-configs/twitter-harness-mcp.json +18 -0
  15. package/config.example.json +45 -0
  16. package/mcp/dist/index.js +4212 -0
  17. package/mcp/dist/onboarding.js +200 -0
  18. package/mcp/dist/panel.html +176 -0
  19. package/mcp/dist/product-link.html +102 -0
  20. package/mcp/dist/repo.js +222 -0
  21. package/mcp/dist/runtime.js +1079 -0
  22. package/mcp/dist/screencast.js +323 -0
  23. package/mcp/dist/setup.js +545 -0
  24. package/mcp/dist/telemetry.js +306 -0
  25. package/mcp/dist/twitterAuth.js +138 -0
  26. package/mcp/dist/version.js +271 -0
  27. package/mcp/dist/version.json +4 -0
  28. package/mcp/install-runtime.mjs +70 -0
  29. package/mcp/install.mjs +169 -0
  30. package/mcp/manifest.json +80 -0
  31. package/mcp/menubar/dashboard_server.py +213 -0
  32. package/mcp/menubar/s4l_card.py +1314 -0
  33. package/mcp/menubar/s4l_log_relay.py +179 -0
  34. package/mcp/menubar/s4l_menubar.py +2439 -0
  35. package/mcp/menubar/s4l_state.py +891 -0
  36. package/mcp/package.json +34 -0
  37. package/mcp/shared/doctor.cjs +437 -0
  38. package/mcp/shared/onboarding-ledger.cjs +324 -0
  39. package/mcp-servers/browser-harness/server.py +968 -0
  40. package/package.json +160 -0
  41. package/requirements.txt +20 -0
  42. package/scripts/_compute_allowlist.py +58 -0
  43. package/scripts/_db_update.py +20 -0
  44. package/scripts/_filt.py +9 -0
  45. package/scripts/_li_notif_match.py +76 -0
  46. package/scripts/_li_notif_orchestrate.py +126 -0
  47. package/scripts/_lock_preempt_test.py +60 -0
  48. package/scripts/_run_icp_precheck.py +57 -0
  49. package/scripts/a16z_pearx_calendar_reminders.py +99 -0
  50. package/scripts/account_resolver.py +141 -0
  51. package/scripts/active_campaigns.py +114 -0
  52. package/scripts/active_users.py +190 -0
  53. package/scripts/amplitude_24h_signups.py +468 -0
  54. package/scripts/amplitude_signups.py +177 -0
  55. package/scripts/apply_onboarding_selections.py +131 -0
  56. package/scripts/audience_pages.py +243 -0
  57. package/scripts/audit_helper.py +120 -0
  58. package/scripts/author_history_block.py +353 -0
  59. package/scripts/autopilot_stall_watch.py +284 -0
  60. package/scripts/backfill_twitter_attempts_topic.py +81 -0
  61. package/scripts/backfill_twitter_log_post_no_id.py +322 -0
  62. package/scripts/bench_dashboard.sh +138 -0
  63. package/scripts/bh_send.py +39 -0
  64. package/scripts/build_persona.py +409 -0
  65. package/scripts/bulk_icp.py +18 -0
  66. package/scripts/campaign_bump.py +51 -0
  67. package/scripts/capture_thread_media.py +288 -0
  68. package/scripts/check_browser_lock_health.sh +81 -0
  69. package/scripts/check_external_pool_depth.py +253 -0
  70. package/scripts/check_unread_web_chats.py +28 -0
  71. package/scripts/claim_web_chat.py +47 -0
  72. package/scripts/classify_run_error.py +158 -0
  73. package/scripts/claude_job.py +988 -0
  74. package/scripts/clean_stale_singleton.sh +56 -0
  75. package/scripts/cleanup_harness_tabs.py +68 -0
  76. package/scripts/copy_browser_cookies.py +454 -0
  77. package/scripts/counterparty_history.py +350 -0
  78. package/scripts/db.py +57 -0
  79. package/scripts/discover_claude_profiles.py +120 -0
  80. package/scripts/discover_linkedin_candidates.py +984 -0
  81. package/scripts/dm_conversation.py +682 -0
  82. package/scripts/dm_db_update.py +69 -0
  83. package/scripts/dm_engage_helper.py +161 -0
  84. package/scripts/dm_outreach_helper.py +147 -0
  85. package/scripts/dm_outreach_twitter_helper.py +129 -0
  86. package/scripts/dm_send_log.py +106 -0
  87. package/scripts/dm_short_links.py +1084 -0
  88. package/scripts/dump_web_chat_history.py +47 -0
  89. package/scripts/engage_github.py +640 -0
  90. package/scripts/engage_reddit.py +1235 -0
  91. package/scripts/engage_twitter_helper.py +301 -0
  92. package/scripts/engagement_styles.py +1787 -0
  93. package/scripts/enrich_twitter_candidates.py +82 -0
  94. package/scripts/feedback_digest.py +448 -0
  95. package/scripts/fetch_prospect_profile.py +312 -0
  96. package/scripts/fetch_twitter_t1.py +134 -0
  97. package/scripts/find_threads.py +530 -0
  98. package/scripts/follow_gate_log.py +59 -0
  99. package/scripts/funnel_per_day.py +194 -0
  100. package/scripts/generate_daily_human_style.py +494 -0
  101. package/scripts/generation_trace.py +173 -0
  102. package/scripts/get_run_cost.py +107 -0
  103. package/scripts/github_engage_helper.py +93 -0
  104. package/scripts/github_tools.py +509 -0
  105. package/scripts/harness_overlay.py +556 -0
  106. package/scripts/harvest_twitter_following.py +243 -0
  107. package/scripts/heartbeat.sh +70 -0
  108. package/scripts/history_context.py +284 -0
  109. package/scripts/http_api.py +206 -0
  110. package/scripts/human_dm_replies_helper.py +169 -0
  111. package/scripts/identity.py +302 -0
  112. package/scripts/ig_batch_creator.sh +93 -0
  113. package/scripts/ig_post_type_picker.py +243 -0
  114. package/scripts/ig_scrape_transcribe.sh +91 -0
  115. package/scripts/ingest_human_dm_replies.py +271 -0
  116. package/scripts/ingest_web_chat_replies.py +229 -0
  117. package/scripts/install_fleet.py +187 -0
  118. package/scripts/invent_mcp_server.py +350 -0
  119. package/scripts/invent_topics.py +1462 -0
  120. package/scripts/learned_preferences.py +263 -0
  121. package/scripts/li_discovery.py +161 -0
  122. package/scripts/link_edit_helper.py +142 -0
  123. package/scripts/link_tail.py +592 -0
  124. package/scripts/linkedin_api.py +561 -0
  125. package/scripts/linkedin_browser.py +730 -0
  126. package/scripts/linkedin_cooldown.py +128 -0
  127. package/scripts/linkedin_exclusions.py +234 -0
  128. package/scripts/linkedin_killswitch.py +1333 -0
  129. package/scripts/linkedin_search_topic_schema.py +49 -0
  130. package/scripts/linkedin_unipile.py +658 -0
  131. package/scripts/linkedin_url.py +228 -0
  132. package/scripts/log_claude_session.py +636 -0
  133. package/scripts/log_draft.py +143 -0
  134. package/scripts/log_linkedin_search_attempts.py +126 -0
  135. package/scripts/log_post.py +651 -0
  136. package/scripts/log_run.py +364 -0
  137. package/scripts/log_thread_media.py +108 -0
  138. package/scripts/log_twitter_search_attempts.py +150 -0
  139. package/scripts/log_twitter_skips.py +211 -0
  140. package/scripts/lookup_post.py +78 -0
  141. package/scripts/mark_web_chat_processed.py +32 -0
  142. package/scripts/mcp_lock_proxy.py +370 -0
  143. package/scripts/memory_snapshot.py +972 -0
  144. package/scripts/merge_review_queue.py +215 -0
  145. package/scripts/mint_external_pool.py +182 -0
  146. package/scripts/mint_kent_pool.py +249 -0
  147. package/scripts/moltbook_post.py +320 -0
  148. package/scripts/moltbook_tools.py +159 -0
  149. package/scripts/pending_threads.py +188 -0
  150. package/scripts/pick_ig_account.py +177 -0
  151. package/scripts/pick_project.py +208 -0
  152. package/scripts/pick_search_topic.py +771 -0
  153. package/scripts/pick_thread_target.py +279 -0
  154. package/scripts/pick_twitter_thread_target.py +202 -0
  155. package/scripts/podlog_fetch_batch.sh +32 -0
  156. package/scripts/post_github.py +1311 -0
  157. package/scripts/post_reddit.py +2668 -0
  158. package/scripts/precompute_dashboard_stats.py +204 -0
  159. package/scripts/preflight.sh +297 -0
  160. package/scripts/progress.py +88 -0
  161. package/scripts/project_excludes.py +353 -0
  162. package/scripts/project_slugs.py +91 -0
  163. package/scripts/project_stats.py +241 -0
  164. package/scripts/project_stats_json.py +1563 -0
  165. package/scripts/project_topics.py +192 -0
  166. package/scripts/qualified_query_bank.py +436 -0
  167. package/scripts/reap_stale_claude_sessions.py +867 -0
  168. package/scripts/reddit_browser.py +2549 -0
  169. package/scripts/reddit_browser_fetch.py +141 -0
  170. package/scripts/reddit_browser_lock.py +593 -0
  171. package/scripts/reddit_chat_sync.py +710 -0
  172. package/scripts/reddit_query_bank.py +200 -0
  173. package/scripts/reddit_threads_helper.py +151 -0
  174. package/scripts/reddit_tools.py +956 -0
  175. package/scripts/refresh_instagram_tokens.py +280 -0
  176. package/scripts/release-mcpb.sh +497 -0
  177. package/scripts/reply_db.py +334 -0
  178. package/scripts/reply_insert.py +98 -0
  179. package/scripts/reply_risk_digest.py +761 -0
  180. package/scripts/reset-test-machine.sh +602 -0
  181. package/scripts/restore_twitter_session.py +177 -0
  182. package/scripts/ripen_reddit_plan.py +478 -0
  183. package/scripts/run_claude.sh +433 -0
  184. package/scripts/run_moltbook_cycle.py +555 -0
  185. package/scripts/s4l_box_update.sh +226 -0
  186. package/scripts/s4l_channel.py +103 -0
  187. package/scripts/s4l_ctl.sh +75 -0
  188. package/scripts/s4l_env.py +47 -0
  189. package/scripts/saps_activity.py +126 -0
  190. package/scripts/saps_mode.py +328 -0
  191. package/scripts/scan_dm_candidates.py +580 -0
  192. package/scripts/scan_github_replies.py +168 -0
  193. package/scripts/scan_instagram_comments.py +481 -0
  194. package/scripts/scan_moltbook_replies.py +252 -0
  195. package/scripts/scan_pii.py +190 -0
  196. package/scripts/scan_reddit_replies.py +377 -0
  197. package/scripts/scan_twitter_mentions_browser.py +327 -0
  198. package/scripts/scan_twitter_thread_followups.py +299 -0
  199. package/scripts/scan_x_profile.py +384 -0
  200. package/scripts/schedule_state.py +202 -0
  201. package/scripts/scheduled_tasks_snapshot.py +123 -0
  202. package/scripts/score_linkedin_candidates.py +419 -0
  203. package/scripts/score_twitter_candidates.py +718 -0
  204. package/scripts/scrape_linkedin_comment_stats.py +1755 -0
  205. package/scripts/scrape_linkedin_stats_browser.py +52 -0
  206. package/scripts/scrape_reddit_views.py +365 -0
  207. package/scripts/seed_search_queries.py +453 -0
  208. package/scripts/seed_search_topics.py +127 -0
  209. package/scripts/send_web_chat_reply.py +130 -0
  210. package/scripts/sentry_init.py +128 -0
  211. package/scripts/setup_twitter_auth.py +1320 -0
  212. package/scripts/snapshot.py +583 -0
  213. package/scripts/stats.py +2702 -0
  214. package/scripts/stats_helper.py +52 -0
  215. package/scripts/strike_alert.py +783 -0
  216. package/scripts/sweep_post_link_clicks.py +107 -0
  217. package/scripts/sync_ig_to_posts.py +147 -0
  218. package/scripts/test_browser_lock.py +189 -0
  219. package/scripts/test_installation_api.sh +52 -0
  220. package/scripts/test_percard_posting.py +142 -0
  221. package/scripts/top_dud_linkedin_queries.py +71 -0
  222. package/scripts/top_dud_reddit_queries.py +67 -0
  223. package/scripts/top_dud_twitter_queries.py +71 -0
  224. package/scripts/top_dud_twitter_topics.py +102 -0
  225. package/scripts/top_linkedin_queries.py +55 -0
  226. package/scripts/top_omitted_reddit_topics.py +91 -0
  227. package/scripts/top_performers.py +588 -0
  228. package/scripts/top_search_topics.py +180 -0
  229. package/scripts/top_twitter_queries.py +190 -0
  230. package/scripts/twitter_access_check.py +382 -0
  231. package/scripts/twitter_account.py +41 -0
  232. package/scripts/twitter_batch_phase.py +126 -0
  233. package/scripts/twitter_browser.py +2804 -0
  234. package/scripts/twitter_cookie_mirror.py +130 -0
  235. package/scripts/twitter_cycle_helper.py +310 -0
  236. package/scripts/twitter_gen_links.py +287 -0
  237. package/scripts/twitter_post_plan.py +1188 -0
  238. package/scripts/twitter_scan.py +324 -0
  239. package/scripts/twitter_supply_signal.py +57 -0
  240. package/scripts/twitter_threads_helper.py +152 -0
  241. package/scripts/unclaim_web_chat.py +29 -0
  242. package/scripts/update_instagram_stats.py +261 -0
  243. package/scripts/update_linkedin_stats_from_feed.py +328 -0
  244. package/scripts/version.py +72 -0
  245. package/scripts/watchdog_hung_runs.py +343 -0
  246. package/scripts/write_generation_trace.py +73 -0
  247. package/setup/SKILL.md +277 -0
  248. package/skill/amplitude-24h-signups.sh +38 -0
  249. package/skill/archive-old-logs.sh +40 -0
  250. package/skill/audit-dm-staleness.sh +42 -0
  251. package/skill/audit-linkedin.sh +14 -0
  252. package/skill/audit-moltbook.sh +4 -0
  253. package/skill/audit-reddit-resurrect.sh +67 -0
  254. package/skill/audit-reddit.sh +4 -0
  255. package/skill/audit-twitter.sh +4 -0
  256. package/skill/audit.sh +287 -0
  257. package/skill/backfill-twitter-attempts-topic.sh +19 -0
  258. package/skill/backfill-twitter-ghost-posts.sh +24 -0
  259. package/skill/check-external-pool-depth.sh +7 -0
  260. package/skill/check-web-chats.sh +203 -0
  261. package/skill/dm-outreach-linkedin.sh +250 -0
  262. package/skill/dm-outreach-reddit.sh +274 -0
  263. package/skill/dm-outreach-twitter.sh +265 -0
  264. package/skill/engage-dm-replies-linkedin.sh +4 -0
  265. package/skill/engage-dm-replies-reddit.sh +4 -0
  266. package/skill/engage-dm-replies-twitter.sh +4 -0
  267. package/skill/engage-dm-replies.sh +1597 -0
  268. package/skill/engage-linkedin.sh +581 -0
  269. package/skill/engage-moltbook.sh +36 -0
  270. package/skill/engage-reddit.sh +146 -0
  271. package/skill/engage-twitter.sh +467 -0
  272. package/skill/github-engage.sh +176 -0
  273. package/skill/ingest-web-chat-replies.sh +38 -0
  274. package/skill/invent-supply-test.sh +100 -0
  275. package/skill/invent-topics.sh +50 -0
  276. package/skill/lib/linkedin-backend.sh +364 -0
  277. package/skill/lib/platform.sh +48 -0
  278. package/skill/lib/reddit-backend.sh +234 -0
  279. package/skill/lib/twitter-backend.sh +314 -0
  280. package/skill/link-edit-github.sh +136 -0
  281. package/skill/link-edit-moltbook.sh +117 -0
  282. package/skill/link-edit-reddit.sh +201 -0
  283. package/skill/linkedin-presence.sh +182 -0
  284. package/skill/linkedin-recovery.sh +282 -0
  285. package/skill/lock.sh +647 -0
  286. package/skill/memory-snapshot.sh +39 -0
  287. package/skill/precompute-stats.sh +35 -0
  288. package/skill/prewarm-funnel.sh +104 -0
  289. package/skill/refresh-instagram-tokens.sh +57 -0
  290. package/skill/refresh-twitter-following.sh +52 -0
  291. package/skill/reply-risk-digest.sh +31 -0
  292. package/skill/run-cycle-update-guard.sh +44 -0
  293. package/skill/run-draft-and-publish.sh +123 -0
  294. package/skill/run-generate-daily-style.sh +50 -0
  295. package/skill/run-github-launchd.sh +62 -0
  296. package/skill/run-github.sh +102 -0
  297. package/skill/run-instagram-daily.sh +149 -0
  298. package/skill/run-instagram-render.sh +875 -0
  299. package/skill/run-linkedin-launchd.sh +81 -0
  300. package/skill/run-linkedin-unipile.sh +130 -0
  301. package/skill/run-linkedin.sh +1593 -0
  302. package/skill/run-moltbook-launchd.sh +61 -0
  303. package/skill/run-moltbook.sh +38 -0
  304. package/skill/run-overlay-watch.sh +100 -0
  305. package/skill/run-reddit-search-launchd.sh +64 -0
  306. package/skill/run-reddit-search.sh +505 -0
  307. package/skill/run-reddit-threads-double.sh +32 -0
  308. package/skill/run-reddit-threads.sh +847 -0
  309. package/skill/run-scan-moltbook-replies.sh +57 -0
  310. package/skill/run-twitter-cycle-launchd.sh +63 -0
  311. package/skill/run-twitter-cycle-singleton.sh +62 -0
  312. package/skill/run-twitter-cycle.sh +2408 -0
  313. package/skill/run-twitter-threads.sh +592 -0
  314. package/skill/scan-instagram-replies.sh +61 -0
  315. package/skill/scan-twitter-followups.sh +57 -0
  316. package/skill/social-autoposter-update.sh +66 -0
  317. package/skill/stats-instagram.sh +72 -0
  318. package/skill/stats-linkedin.sh +271 -0
  319. package/skill/stats-moltbook.sh +4 -0
  320. package/skill/stats-reddit.sh +4 -0
  321. package/skill/stats-twitter.sh +4 -0
  322. package/skill/stats.sh +521 -0
  323. package/skill/strike-alert.sh +18 -0
  324. package/skill/styles.sh +87 -0
  325. package/skill/sweep-link-clicks.sh +40 -0
  326. package/skill/topics.sh +51 -0
package/README.md ADDED
@@ -0,0 +1,143 @@
1
+ # social-autoposter
2
+
3
+ Open-source repo behind **[S4L (s4lai)](https://s4l.ai)**: an automated social posting pipeline for Reddit, X/Twitter, LinkedIn, and Moltbook. Ships as a Claude Code skill plus a set of standalone Python helpers and macOS launchd jobs.
4
+
5
+ > The hosted managed version is **S4L** (written `s4lai`, domain `s4l.ai`): done-for-you Reddit and Twitter brand-awareness, $1/1K impressions, $50/1K site visits. See https://s4l.ai.
6
+
7
+ State (posts, replies, candidates, stats) is read and written through the hosted S4L HTTP API (`AUTOPOSTER_API_BASE` + an install key in `~/social-autoposter/.env`); no database to provision. Each platform drives its own persistent Playwright MCP browser profile, so logins survive across runs.
8
+
9
+ ## Prerequisites
10
+
11
+ A new machine needs all of these before the pipeline can run end to end:
12
+
13
+ - **macOS** (the launchd plists are mac-only; Linux users can crib the cron snippets from `setup/SKILL.md` Step 7)
14
+ - **Node.js 16+** (for `npx`, the installer, and `@playwright/mcp` at runtime)
15
+ - **Python 3.9+** with `pip3` (helper scripts; deps auto-installed by the installer)
16
+ - **Claude Code CLI** on `PATH` (the cron scripts shell out to `claude -p` with a per-platform MCP config)
17
+ - One Chromium install per platform (created on first run by `@playwright/mcp` against the persistent profile dirs)
18
+
19
+ Optional:
20
+
21
+ - `MOLTBOOK_API_KEY` in `.env` for Moltbook posting and scanning
22
+ - `RESEND_API_KEY` and `NOTIFICATION_EMAIL` in `.env` for DM-escalation emails
23
+
24
+ ## Install
25
+
26
+ ```bash
27
+ npx social-autoposter init
28
+ ```
29
+
30
+ `bin/cli.js` does all of the wiring in one shot:
31
+
32
+ 1. Copies `scripts/`, `skill/`, `setup/`, `SKILL.md`, and `browser-agent-configs/` into `~/social-autoposter/`
33
+ 2. Creates `config.json` from `config.example.json` and writes a blank `.env` template (fill in your S4L API key and optional `MOLTBOOK_API_KEY`)
34
+ 3. Installs the Python helper deps via `pip3` if missing
35
+ 4. Generates launchd plists in `~/social-autoposter/launchd/` with the user's actual `HOME` and `PATH`
36
+ 5. Installs the Playwright MCP configs to `~/.claude/browser-agent-configs/` (twitter, reddit, linkedin) with `__HOME__` and `__NODE_BIN__` placeholders substituted. Existing files are left alone, so any window-position tweaks survive `npx social-autoposter update`.
37
+ 6. Creates empty persistent browser profile dirs at `~/.claude/browser-profiles/{twitter,reddit,linkedin}`
38
+ 7. Symlinks `~/.claude/skills/social-autoposter` and `~/.claude/skills/social-autoposter-setup` to the install dir
39
+
40
+ To refresh code without touching user files (`config.json`, `.env`, `SKILL.md`, or any browser config you customized):
41
+
42
+ ```bash
43
+ npx social-autoposter update
44
+ ```
45
+
46
+ ## Configure
47
+
48
+ Tell your Claude agent: **"set me up on social-autoposter plugin end to end"**. The
49
+ setup skill treats that as a terminal goal:
50
+
51
+ 1. Inspect and repair the owned runtime.
52
+ 2. Auto-detect the best browser profile and connect X/Twitter. macOS may require
53
+ a Safe Storage approval; a logged-out account may require one manual sign-in.
54
+ 3. Scan the X profile, discover and research the user's product, and infer a
55
+ conservative project, ICP, voice, and search topics without an interview.
56
+ 4. Save the project and seed its topics into the backend.
57
+ 5. Run a draft-only cycle to verify the pipeline without posting.
58
+
59
+ The agent pauses only for an unavoidable login or when no product can be
60
+ identified. Autopilot remains off until explicitly requested.
61
+
62
+ ## How the runtime is wired
63
+
64
+ ```
65
+ launchd ──▶ skill/run-{platform}.sh ──▶ claude -p --strict-mcp-config --mcp-config ~/.claude/browser-agent-configs/{platform}-agent-mcp.json
66
+ │ │
67
+ │ └──▶ @playwright/mcp@latest
68
+ │ │
69
+ │ └──▶ ~/.claude/browser-profiles/{platform}/ (persistent userDataDir)
70
+
71
+ ├──▶ scripts/find_threads.py, top_twitter_queries.py (no browser, API dedup)
72
+ ├──▶ scripts/pick_project.py (weighted project rotation)
73
+ ├──▶ scripts/top_performers.py (feedback report from past stats)
74
+ └──▶ S4L HTTP API (AUTOPOSTER_API_BASE in .env)
75
+ ```
76
+
77
+ Each `skill/run-*.sh`:
78
+
79
+ 1. Controlled by launchd (load/unload). Use the dashboard Pause All / Resume All button, or `launchctl unload/load` directly
80
+ 2. Acquires a per-platform lock from `skill/lock.sh` (waits up to 60 min for any prior run)
81
+ 3. Sources `~/social-autoposter/.env`
82
+ 4. Picks a project, builds a feedback report, fetches `llms.txt` for product context
83
+ 5. Calls `find_*.py` for API-side candidates already deduped against the DB
84
+ 6. Spawns a child Claude process with `--strict-mcp-config` so it only sees the one platform's browser MCP
85
+
86
+ The launchd schedules generated by `bin/cli.js` on install:
87
+
88
+ | Job | Cadence |
89
+ |-----|---------|
90
+ | `com.m13v.social-stats` (`stats.sh`) | every 21600 s (6 h) |
91
+ | `com.m13v.social-engage` (`engage.sh`) | every 21600 s (6 h) |
92
+
93
+ All per-platform plists live in `launchd/` (reddit-search, reddit-threads, twitter-cycle, linkedin, moltbook, github, octolens, audit, dm-replies-*, link-edit-*, scan-reddit-replies, scan-moltbook-replies, etc.) and use either `StartInterval` or `StartCalendarInterval` for fixed wall-clock times. Activate any of them with:
94
+
95
+ ```bash
96
+ ln -sf ~/social-autoposter/launchd/com.m13v.social-twitter-cycle.plist ~/Library/LaunchAgents/
97
+ launchctl load ~/Library/LaunchAgents/com.m13v.social-twitter-cycle.plist
98
+ ```
99
+
100
+ ## Skill commands
101
+
102
+ | Command | What it does |
103
+ |---------|-------------|
104
+ | `/social-autoposter` | Comment run: find threads, draft, post, log (cron-safe) |
105
+ | `/social-autoposter post` | Create an original post or thread (manual only) |
106
+ | `/social-autoposter stats` | Update engagement stats via API |
107
+ | `/social-autoposter engage` | Scan and reply to responses on our posts |
108
+ | `/social-autoposter audit` | Full browser audit of all posts |
109
+
110
+ View live stats at `https://s4l.ai/stats/<your-handle>` once posts start landing.
111
+
112
+ ## Repo layout
113
+
114
+ ```
115
+ social-autoposter/
116
+ ├── SKILL.md the playbook (locked, immutable)
117
+ ├── bin/cli.js installer + dashboard launcher
118
+ ├── browser-agent-configs/ Playwright MCP templates (twitter/reddit/linkedin)
119
+ ├── config.example.json config template
120
+ ├── setup/SKILL.md autonomous end-to-end setup skill (locked)
121
+ ├── scripts/ Python and JS helpers (no browser, no LLM)
122
+ ├── skill/ shell wrappers invoked by launchd
123
+ └── launchd/ generated macOS LaunchAgent plists
124
+ ```
125
+
126
+ ## For other AI agents
127
+
128
+ The skill works with any agent that has shell access, browser automation, and an LLM. The Python and JS helpers in `scripts/` handle thread discovery, reply scanning, and stats updates without needing a browser. `SKILL.md` is the playbook; any agent can read it and execute the workflows with its own tools.
129
+
130
+ ## Pause and resume
131
+
132
+ Use the dashboard at `localhost:3141` (Pause All / Resume All button), or manually:
133
+
134
+ ```bash
135
+ # Pause: unload all jobs + kill running processes
136
+ for plist in ~/Library/LaunchAgents/com.m13v.social-*.plist; do launchctl unload "$plist"; done
137
+
138
+ # Resume: reload all jobs
139
+ for plist in ~/social-autoposter/launchd/com.m13v.social-*.plist; do
140
+ ln -sf "$plist" ~/Library/LaunchAgents/
141
+ launchctl load ~/Library/LaunchAgents/$(basename "$plist")
142
+ done
143
+ ```
package/SKILL.md ADDED
@@ -0,0 +1,342 @@
1
+ ---
2
+ name: social-autoposter
3
+ description: "Automate social media posting across Reddit, X/Twitter, LinkedIn, and Moltbook. Find threads, post comments, create original posts, track engagement stats. Use when: 'post to social', 'social autoposter', 'find threads to comment on', 'create a post', 'audit social posts', 'update post stats'."
4
+ user_invocable: true
5
+ ---
6
+
7
+ # Social Autoposter
8
+
9
+ Automates finding, posting, and tracking social media comments and original posts across Reddit, X/Twitter, LinkedIn, and Moltbook.
10
+
11
+ ## Quick Start
12
+
13
+ | Command | What it does |
14
+ |---------|-------------|
15
+ | `/social-autoposter` | Comment run — find threads + post comment + log (cron-safe) |
16
+ | `/social-autoposter post` | Create an original post/thread (manual or cron-driven for Reddit threads) |
17
+ | `/social-autoposter stats` | Update engagement stats via API |
18
+ | `/social-autoposter engage` | Scan and reply to responses on our posts |
19
+ | `/social-autoposter audit` | Full browser audit of all posts |
20
+
21
+ **View your posts live:** `https://s4l.ai/stats/[your_handle]`
22
+ — e.g. `https://s4l.ai/stats/m13v_` (Twitter handle without `@`), `https://s4l.ai/stats/Deep_Ad1959` (Reddit), `https://s4l.ai/stats/matthew-autoposter` (Moltbook).
23
+ The handles come from `config.json → accounts.*.handle/username`. Each platform account has its own URL.
24
+
25
+ ---
26
+
27
+ ## FIRST: Read config
28
+
29
+ Before doing anything, read `~/social-autoposter/config.json`. Everything — accounts, projects, subreddits, content angle — comes from there.
30
+
31
+ ```bash
32
+ cat ~/social-autoposter/config.json
33
+ ```
34
+
35
+ Key fields you'll use throughout every workflow:
36
+
37
+ - `accounts.reddit.username` — Reddit handle to post as
38
+ - `accounts.twitter.handle` — X/Twitter handle
39
+ - `accounts.linkedin.name` — LinkedIn display name
40
+ - `accounts.moltbook.username` — Moltbook username
41
+ - `subreddits` — list of subreddits to monitor and post in
42
+ - `content_angle` — the user's unique perspective for writing authentic comments
43
+ - `projects` — products/repos to mention naturally when relevant (each has `name`, `description`, `website`, `github`, `topics`)
44
+ - `database` — unused (DB is Postgres via `DATABASE_URL` in `.env`)
45
+
46
+ Use these values everywhere below instead of any hardcoded names or links.
47
+
48
+ ---
49
+
50
+ ## Helper Scripts
51
+
52
+ Standalone Python scripts — no LLM needed.
53
+
54
+ ```bash
55
+ python3 ~/social-autoposter/scripts/find_threads.py --include-moltbook
56
+ python3 ~/social-autoposter/scripts/scan_reddit_replies.py
57
+ python3 ~/social-autoposter/scripts/scan_moltbook_replies.py
58
+ python3 ~/social-autoposter/scripts/update_stats.py --quiet
59
+ ```
60
+
61
+ ---
62
+
63
+ ## Workflow: Post (`/social-autoposter`)
64
+
65
+ **There is NO posting rate limit. Do not add one, do not enforce one, do not invent one. Post as many times as needed.**
66
+
67
+ ### 1. Find candidate threads
68
+
69
+ **Option A — Script (preferred):**
70
+ ```bash
71
+ python3 ~/social-autoposter/scripts/find_threads.py --include-moltbook
72
+ ```
73
+
74
+ **Option B — Browse manually:**
75
+ Browse `/new` and `/hot` on the subreddits from `config.json`. Also check Moltbook via API.
76
+
77
+ ### 2. Pick the best thread
78
+
79
+ - You have a genuine angle from `content_angle` in config.json
80
+ - Not already posted in: `SELECT thread_url FROM posts`
81
+ - Last 5 comments don't repeat the same talking points:
82
+ ```sql
83
+ SELECT our_content FROM posts ORDER BY id DESC LIMIT 5
84
+ ```
85
+ - If nothing fits naturally, **stop**. Better to skip than force a bad comment.
86
+
87
+ ### 3. Read the thread + top comments
88
+
89
+ Check tone, length cues, thread age. Find best comment to reply to (high-upvote comments get more visibility).
90
+
91
+ ### 4. Draft the comment
92
+
93
+ Follow Content Rules below. 2-3 sentences, first person, specific details from `content_angle`. No product links in top-level comments.
94
+
95
+ ### 5. Post it
96
+
97
+ **Reddit** (browser automation):
98
+ - Navigate to `old.reddit.com` thread URL
99
+ - Reply box → type comment → submit → wait 2-3s → verify comment appeared → capture permalink → close tab
100
+ - Post as the username in `config.json → accounts.reddit.username`
101
+
102
+ **X/Twitter** (browser automation):
103
+ - Navigate to tweet → reply box → type → Reply → verify → capture URL
104
+ - Post as the handle in `config.json → accounts.twitter.handle`
105
+
106
+ **LinkedIn** (browser automation):
107
+ - Navigate to post → comment box → type → Post → close tab
108
+ - Post as the name in `config.json → accounts.linkedin.name`
109
+
110
+ **Moltbook** (API — no browser needed):
111
+ ```bash
112
+ source ~/social-autoposter/.env
113
+ curl -s -X POST -H "Authorization: Bearer $MOLTBOOK_API_KEY" -H "Content-Type: application/json" \
114
+ -d '{"title": "...", "content": "...", "type": "text", "submolt_name": "general"}' \
115
+ "https://www.moltbook.com/api/v1/posts"
116
+ ```
117
+ On Moltbook: write as agent ("my human" not "I").
118
+ Verify: fetch post by UUID, check `verification_status` is `"verified"`.
119
+
120
+ ### 6. Log + sync
121
+
122
+ ```sql
123
+ INSERT INTO posts (platform, thread_url, thread_author, thread_author_handle,
124
+ thread_title, thread_content, our_url, our_content, our_account,
125
+ source_summary, project_name, engagement_style, feedback_report_used, status, posted_at)
126
+ VALUES (%s, %s, %s, %s, %s, %s, %s, %s, %s, %s, %s, %s, TRUE, 'active', NOW());
127
+ ```
128
+
129
+ Set `project_name` to the matching project name from `config.json` (e.g., 'Fazm', 'Cyrano', 'Terminator'). Every post/comment MUST be labeled with its target project. If engagement is general/unrelated to any project, use 'general'.
130
+
131
+ Set `engagement_style` to the style you chose for this post (e.g., 'critic', 'storyteller', 'pattern_recognizer', 'curious_probe', 'contrarian', 'data_point_drop', 'snarky_oneliner'). Every post MUST have an engagement_style.
132
+
133
+ Use the account value from `config.json` for `our_account`.
134
+
135
+ Posts are written directly to the Postgres database. No separate post-sync step is required.
136
+
137
+ ---
138
+
139
+ ## Workflow: Create Post (`/social-autoposter post`)
140
+
141
+ **Manual only — never run from cron.** Original posts are high-stakes and need human review.
142
+
143
+ ### 1. Cross-posting check
144
+
145
+ ```sql
146
+ SELECT platform, thread_title, posted_at FROM posts
147
+ WHERE source_summary LIKE '%' || %s || '%' AND posted_at >= NOW() - INTERVAL '30 days'
148
+ ORDER BY posted_at DESC;
149
+ ```
150
+
151
+ **NEVER post the same or similar content to multiple subreddits.** Duplicate cross-posts read as spam and get removed. Each post must be unique to its community.
152
+
153
+ ### 2. Pick one target community
154
+
155
+ Choose the single best subreddit from `config.json → subreddits` for this topic. Tailor the post to that community's culture and tone.
156
+
157
+ ### 3. Draft the post
158
+
159
+ **Pre-post checklist** (must pass ALL before posting):
160
+
161
+ - [ ] No em dashes (—). Use regular dashes (-) or commas instead
162
+ - [ ] No markdown headers (##) or bold (**) in Reddit posts
163
+ - [ ] No numbered/bulleted lists — write in paragraphs
164
+ - [ ] No "Hi everyone" or "Hey r/subreddit" openings
165
+ - [ ] Title doesn't use clickbait patterns ("What I wish I'd known", "A guide to")
166
+ - [ ] Contains at least one imperfection: incomplete thought, casual aside, informality
167
+ - [ ] Reads like a real person writing on their phone, not an essay
168
+ - [ ] Does NOT link to any project in the post body — earn attention first
169
+ - [ ] Not too long — 2-4 short paragraphs max for Reddit
170
+
171
+ **Read it out loud.** If it sounds like a blog post or generic AI copy, rewrite it.
172
+
173
+ ### 4. Post it
174
+
175
+ **Reddit**: old.reddit.com → Submit new text post → paste title + body → submit → verify → capture permalink.
176
+
177
+ ### 5. Log it
178
+
179
+ ```sql
180
+ INSERT INTO posts (platform, thread_url, thread_author, thread_author_handle,
181
+ thread_title, thread_content, our_url, our_content, our_account,
182
+ source_summary, project_name, engagement_style, feedback_report_used, status, posted_at)
183
+ VALUES (%s, %s, %s, %s, %s, %s, %s, %s, %s, %s, %s, %s, TRUE, 'active', NOW());
184
+ ```
185
+
186
+ Set `engagement_style` to the style you chose (e.g., 'critic', 'storyteller', 'pattern_recognizer'). Every post MUST have an engagement_style.
187
+
188
+ Set `project_name` to the matching project name from `config.json`. For original posts: `thread_url` = `our_url`, `thread_author` = our account from config.json.
189
+
190
+ ### 6. Mandatory engagement plan
191
+
192
+ After posting, you MUST:
193
+ - Check for comments within 2-4 hours
194
+ - Reply to every substantive comment within 24 hours
195
+ - Replies should be casual, conversational, expand the topic — NOT polished paragraphs
196
+ - If someone asks whether the post is AI or a bot: acknowledge it briefly, do not argue, do not pitch, and prefer disengaging (same policy as the engage pipeline's meta-callout handling)
197
+
198
+ ---
199
+
200
+ ## Workflow: Cron-driven Reddit Threads (`run-reddit-threads.sh`)
201
+
202
+ Daily-cadence original Reddit threads across all products, automated via launchd.
203
+
204
+ **Config lives per-project** under `projects[].threads`:
205
+ - `enabled`: true/false
206
+ - `own_community`: `{subreddit, cadence, floor_days}` (optional). Defaults to 1-day floor.
207
+ - `external_subreddits`: list of external subs (default 3-day floor, override via `external_floor_days`)
208
+ - `topic_angles`: discussion-starter ideas the agent picks from
209
+ - Voice guidance comes from `projects[].voice` (tone, never)
210
+ - `content_sources.guide_dir` / `link_base`: optional source paths/URLs
211
+ - `dynamic_context.day_counter` / `static_facts`: live-calculated facts injected into the prompt
212
+
213
+ **Source research** uses `landing_pages.repo` + `landing_pages.product_source[]` (same schema as the SEO pipeline). The agent is told to read README + product source before drafting so posts ground in real details.
214
+
215
+ **Picker** (`scripts/pick_thread_target.py`): weighted project selection with:
216
+ - Per-sub floor-days filter (queries `posts` table for this account's last original thread)
217
+ - `subreddit_bans` filter: `banned` (can't post or comment) + `skip_threads` (threads blocked, comments OK)
218
+ - Own-community candidates always picked first when eligible
219
+
220
+ **Schedule**: `com.m13v.social-reddit-threads.plist` fires 4x/day at 00:15, 06:15, 12:15, 18:15.
221
+
222
+ **Lock**: the runner calls `acquire_lock reddit-threads` to serialize against the comment pipeline.
223
+
224
+ ---
225
+
226
+ ## Workflow: Stats (`/social-autoposter stats`)
227
+
228
+ ```bash
229
+ python3 ~/social-autoposter/scripts/update_stats.py
230
+ ```
231
+
232
+ After running, view updated stats at `https://s4l.ai/stats/[handle]`. Stats are read from the same Postgres database used by the posting pipeline. Changes appear on the website within ~5 minutes.
233
+
234
+ ---
235
+
236
+ ## Workflow: Engage (`/social-autoposter engage`)
237
+
238
+ ### Phase A: Scan for replies (no browser)
239
+ ```bash
240
+ python3 ~/social-autoposter/scripts/scan_reddit_replies.py
241
+ python3 ~/social-autoposter/scripts/scan_moltbook_replies.py
242
+ ```
243
+
244
+ ### Phase B: Respond to pending replies
245
+
246
+ ```sql
247
+ SELECT r.id, r.platform, r.their_author, r.their_content, r.their_comment_url,
248
+ r.depth, p.thread_title, p.our_content
249
+ FROM replies r JOIN posts p ON r.post_id = p.id
250
+ WHERE r.status='pending' ORDER BY r.discovered_at ASC LIMIT 10
251
+ ```
252
+
253
+ Draft replies: 2-4 sentences, casual, expand the topic. Apply Tiered Reply Strategy.
254
+
255
+ Post via browser (Reddit/X) or API (Moltbook). Update:
256
+ ```sql
257
+ UPDATE replies SET status='replied', our_reply_content=%s, our_reply_url=%s,
258
+ replied_at=NOW() WHERE id=%s
259
+ ```
260
+
261
+ ### Phase C: X/Twitter replies (browser required)
262
+
263
+ Navigate to `https://x.com/notifications/mentions`. Find replies to the handle in config.json. Respond to substantive ones. Log to `replies` table.
264
+
265
+ ---
266
+
267
+ ## Workflow: Audit (`/social-autoposter audit`)
268
+
269
+ Visit each post URL via browser. Check status (active/deleted/removed/inactive). Update engagement metrics. Report summary.
270
+
271
+ ---
272
+
273
+ ## Content Rules
274
+
275
+ ### Tone & Voice
276
+
277
+ 1. **Write like you're texting a coworker.** Lowercase fine. Sentence fragments fine. Never start with "Makes sense" or "The nuance here is." If it sounds like a blog post, rewrite it.
278
+ 2. **First person, specific.** Use concrete details from `content_angle` in config.json. Real numbers, real experiences, not generalizations.
279
+ 3. **Reply to top comments, not just OP.**
280
+ 4. **Only comment when there's a genuine angle from the user's work.** Use `content_angle` from config.json. If the thread doesn't connect, skip it.
281
+ 5. **No product links in top-level comments or original posts.** Earn attention first. Only link in deeper replies when someone asks.
282
+ 6. **On Moltbook, write as an agent.** "my human" not "I".
283
+ 7. **Log everything.**
284
+
285
+ ### Writing style (applies to ALL content)
286
+
287
+ 8. **No em dashes (—).** Use commas, periods, or regular dashes (-) instead.
288
+ 9. **No markdown formatting in Reddit.** No headers (##), no bold (**text**), no numbered lists. Write in plain paragraphs.
289
+ 10. **Never cross-post.** One post per topic per community.
290
+ 11. **Include imperfections.** Contractions, sentence fragments, casual asides, occasional lowercase.
291
+ 12. **Vary your openings.** Don't always start with credentials. Sometimes just jump into the topic.
292
+ 13. **Reply to comments on your posts.** Reply within 24h; an unanswered thread of your own looks neglected.
293
+
294
+ ### Bad vs Good (Comments)
295
+
296
+ BAD: "Makes sense — Claude already tries to tail the output on its own but by then the tokens are already in context."
297
+ GOOD: "gonna try this, I run 5 agents in parallel and my API bill is becoming a second rent payment"
298
+
299
+ BAD: "What everyone here is describing is basically specification-driven development."
300
+ GOOD: "I spend more time writing CLAUDE.md specs than I ever spent writing code. the irony is I'm basically doing waterfall now and shipping faster than ever."
301
+
302
+ ### Bad vs Good (Original Posts)
303
+
304
+ BAD title: "What I Wish I'd Known Before My First Vipassana Retreat: A Complete Guide"
305
+ GOOD title: "just did my 7th course, some things that surprised me"
306
+
307
+ BAD body: Structured with headers, bold, numbered lists, "As a tech founder..."
308
+ GOOD body: Paragraphs, incomplete thoughts, personal details, casual tone, ends with a genuine question
309
+
310
+ ### Bad vs Good (DM Replies)
311
+
312
+ DM replies are texting-style. 1 to 3 sentences. Always reference something specific from the inbound. No unearned call offers, no fabricated links, no time-bound commitments. Booking links only when the matched project has `booking_link_auto_share: true` AND `qualification_status=qualified` on the DM row.
313
+
314
+ BAD: "Hey! I saw your comment on r/startups about agent orchestration. I'd love to share what we're working on, would you be open to a quick call?" (cold-pitch shape, premature call ask)
315
+ GOOD: "yo the point about agents racing on the same file hit home, we solved it with worktrees per agent. what's your setup?"
316
+
317
+ BAD: "Great question! Our product handles exactly that scenario. Check out [link] for more details." (sales register, leading with link in an early DM)
318
+ GOOD: "we hit that too, ended up using the accessibility API route because screenshot-based kept flaking on retina displays"
319
+
320
+ BAD: "Absolutely! Let's do Thursday at 3pm, I'll send an invite." (time-bound commitment, bot has no calendar authority)
321
+ GOOD: "yeah easier to figure it out here, what specifically are you trying to wire up?"
322
+
323
+ BAD: "I totally understand your hesitation. But our solution is different because..." (defensive, pushy rebuttal)
324
+ GOOD: "makes sense, we kicked it around for 6 months before pulling the trigger. what's been the blocker on your end?"
325
+
326
+ ---
327
+
328
+ ## Tiered Reply Strategy
329
+
330
+ **Tier 1 — Default (no link):** Genuine engagement. Expand topic, ask follow-ups. Most replies.
331
+
332
+ **Tier 2 — Natural mention:** Conversation touches a topic matching one of the user's projects (from `config.json → projects[].topics`). Mention casually, link only if it adds value. Triggers: "what tool do you use", problem matches a project topic, 2+ replies deep.
333
+
334
+ **Tier 3 — Direct ask:** They ask for link/try/source. Give it immediately using `projects[].website` or `projects[].github` from config.json.
335
+
336
+ ---
337
+
338
+ ## Database Schema
339
+
340
+ `posts`: id, platform, thread_url, thread_title, our_url, our_content, our_account, project_name, posted_at, status, upvotes, comments_count, views, source_summary, link_edited_at, link_edit_content
341
+
342
+ `replies`: id, post_id, platform, their_author, their_content, our_reply_content, status (pending|replied|skipped|error), depth