Loading...
Loading...
AI-powered social media agent with real browser automation for autonomous account operation
npx skill4agent add aradotso/ai-agent-skills locoagent-social-media-automationSkill by ara.so — AI Agent Skills collection.
# Install Bun runtime
curl -fsSL https://bun.sh/install | bash
# Install agent-browser CLI
npm install -g @vercel/agent-browsergit clone https://github.com/LocoreMind/locoagent.git
cd locoagent
bun install.env# OpenRouter (recommended - access 200+ models)
CLAUDE_CODE_USE_OPENAI=1
OPENAI_API_KEY=sk-or-v1-...
OPENAI_BASE_URL=https://openrouter.ai/api/v1
OPENAI_MODEL=anthropic/claude-sonnet-4.5
# Required for automated mode
SKIP_PERMISSIONS=1# DeepSeek (with thinking mode)
CLAUDE_CODE_USE_OPENAI=1
OPENAI_API_KEY=<DEEPSEEK_API_KEY>
OPENAI_BASE_URL=https://api.deepseek.com
OPENAI_MODEL=deepseek-v4-flash
# Ollama (local models)
CLAUDE_CODE_USE_OPENAI=1
OPENAI_API_KEY=ollama
OPENAI_BASE_URL=http://localhost:11434/v1
OPENAI_MODEL=llama3.2
# Anthropic direct (native SDK)
ANTHROPIC_API_KEY=<ANTHROPIC_API_KEY># One-time: copy Chrome profile and launch with CDP
bun run setup-chrome
# Connect agent-browser to running Chrome
agent-browser connect 9222# Start interactive session
bun start
# Load X.com skill and execute task
> /x-com open home timeline, like first 3 posts about AI
# Check operation history
> /operation-log recent --limit 20# Single query execution
bun start -p "open X.com and like the first post about AI agents"
# With specific model
bun start --model anthropic/claude-sonnet-4.5 -p "/x-com like 5 posts about LLMs"
# Platform-specific task
bun start -p "/x-com like 5 posts about 'large language models', then follow the authors"# Interactive
> /x-com open home timeline, like first 3 posts about AI, reply to the best one
# Headless
bun start -p "/x-com like 5 posts about 'machine learning', follow authors with >1k followers"skills/linkedin/SKILL.md---
description: "LinkedIn platform operations playbook"
allowed-tools:
- Bash
user-invocable: true
---
# LinkedIn Operations
## 1. Navigation
### Open Home Feed
```bash
agent-browser open https://www.linkedin.com/feedagent-browser open "https://www.linkedin.com/search/results/content/?keywords=AI%20agents"
agent-browser snapshot -i -c -s 'div[data-post-id]'agent-browser snapshot -ibutton[aria-label*="Like"]agent-browser click @e<ref>div[role="textbox"]agent-browser click @e<ref>agent-browser fill @e<ref> "Insightful post!"
Load the skill:
```bash
bun start
> /linkedin search for posts about 'AI safety', like top 3# List all workflows
bun run workflow list
# Run once (blocking)
bun run workflow run --id hf-papers-to-x
# Run once (background)
bun run workflow start --id hf-papers-to-x
# Daemon mode (every 3 minutes)
bun run workflow daemon --id x-search-reply --interval 3
# Stop workflow
bun run workflow stop --id x-search-reply
# View status
bun run workflow status
# View execution history
bun run workflow history --id hf-papers-to-xworkflows/linkedin-engagement.json{
"id": "linkedin-engagement",
"name": "LinkedIn Daily Engagement",
"description": "Search for AI posts on LinkedIn and engage",
"schedule": "daily",
"executor": "executors/linkedin-engagement.ts",
"config": {
"searchQuery": "artificial intelligence",
"maxPosts": 5,
"cdpPort": 9222
}
}workflows/executors/linkedin-engagement.ts#!/usr/bin/env bun
import { execSync } from 'node:child_process'
// Parse config from workflow engine
const configArg = process.argv.find((_, i, a) => a[i - 1] === '--config')
const config = JSON.parse(configArg!)
// agent-browser helper
function ab(cmd: string): string {
return execSync(`agent-browser --cdp ${config.cdpPort} ${cmd}`, {
encoding: 'utf-8',
timeout: 30000,
}).trim()
}
// Helper to check operation log
function hasEngaged(postUrl: string): boolean {
try {
execSync(`bun run scripts/log-operation.ts check --platform linkedin --action like --url "${postUrl}"`, {
encoding: 'utf-8',
stdio: 'ignore'
})
return true // exit 0 = already done
} catch {
return false // exit 1 = not done
}
}
// Helper to log operation
function logOperation(postUrl: string, action: string, status: string, note: string) {
execSync(`bun run scripts/log-operation.ts add --platform linkedin --action ${action} --url "${postUrl}" --status ${status} --note "${note}"`, {
encoding: 'utf-8',
stdio: 'inherit'
})
}
console.error('[linkedin-engagement] Starting workflow...')
// Step 1: Navigate to search
console.error(`[linkedin-engagement] Searching for: ${config.searchQuery}`)
const searchUrl = `https://www.linkedin.com/search/results/content/?keywords=${encodeURIComponent(config.searchQuery)}`
ab(`open "${searchUrl}"`)
ab('wait 3000')
// Step 2: Get posts
console.error('[linkedin-engagement] Getting posts...')
const snapshot = ab('snapshot -i -c -s \'div[data-post-id]\'')
const posts = JSON.parse(snapshot)
let engaged = 0
const stepsTotal = Math.min(posts.length, config.maxPosts)
// Step 3: Engage with posts
for (let i = 0; i < stepsTotal; i++) {
const post = posts[i]
const postUrl = post.attributes?.['data-urn'] || `post-${i}`
// Check if already engaged
if (hasEngaged(postUrl)) {
console.error(`[linkedin-engagement] Already engaged with ${postUrl}, skipping`)
continue
}
// Find like button
const likeButton = post.children?.find((el: any) =>
el.attributes?.['aria-label']?.includes('Like')
)
if (likeButton?.ref) {
ab(`click ${likeButton.ref}`)
logOperation(postUrl, 'like', 'success', `Workflow: ${config.searchQuery}`)
engaged++
console.error(`[linkedin-engagement] Liked post ${i + 1}/${stepsTotal}`)
ab('wait 2000') // Rate limiting
}
}
// Output final summary (required)
console.log(JSON.stringify({
stepsCompleted: engaged,
stepsTotal,
searchQuery: config.searchQuery
}))bun run workflow run --id linkedin-engagementimport { execSync } from 'node:child_process'
function hasLiked(postUrl: string): boolean {
try {
execSync(`bun run scripts/log-operation.ts check --platform x --action like --url "${postUrl}"`, {
encoding: 'utf-8',
stdio: 'ignore'
})
return true // exit 0 = already done
} catch {
return false // exit 1 = not done
}
}
const url = "https://x.com/user/status/123"
if (hasLiked(url)) {
console.log("Already liked this post")
} else {
// Perform like action
execSync(`agent-browser click @e5`)
// Log operation
execSync(`bun run scripts/log-operation.ts add --platform x --action like --url "${url}" --status success --note "AI research post"`)
}# Check if operation was performed (exit 0 = done, exit 1 = not done)
bun run scripts/log-operation.ts check \
--platform x \
--action like \
--url "https://x.com/user/status/123"
# Record operation
bun run scripts/log-operation.ts add \
--platform x \
--action like \
--url "https://x.com/user/status/123" \
--status success \
--note "AI agents research post"
# View recent operations
bun run scripts/log-operation.ts recent --limit 20
# 30-day summary
bun run scripts/log-operation.ts summary --days 30persona/operation-log.jsonpersona/tasks.md## Daily Tasks
1. Engage with AI research content (like 5-10 posts)
2. Monitor project mentions and respond
3. Leave 1-2 technical comments on relevant posts
## Weekly Tasks (Monday)
4. Follow 3-5 relevant researchers or developers
5. Post 1 original tweet about recent findings
## Session Constraints
| Action | Max per session |
|----------|----------------|
| Likes | 10 |
| Comments | 2 |
| Follows | 5 |
| Posts | 1 |# Execute today's tasks
bun run run-tasks
# Preview prompt without running
bun run run-tasks:dry
# Restrict to one platform
bun run run-tasks -- --platform x--print# Terminal 1: start monitor
bun run tail
# Terminal 2: run agent
bun start -p "/x-com open timeline, like first post"═══ New Task ═══
/x-com open timeline, like first post
[6:30:47 PM] ⚡ Bash: agent-browser connect 9222
[6:30:47 PM] ✓ Result: Done
[6:31:10 PM] ⚡ Bash: agent-browser open https://x.com/home
[6:31:27 PM] ⚡ Bash: agent-browser snapshot -i -c -s 'article'
[6:31:44 PM] ● Agent: Found first post, like button ref=e136
[6:31:44 PM] ⚡ Bash: agent-browser click e136
[6:31:45 PM] ✓ Result: Done# Replay latest session from beginning
bun run tail:history
# List recent sessions
bun run tail:list
# Watch specific session
bun run tail <session-id>#!/usr/bin/env bun
import { execSync } from 'node:child_process'
function ab(cmd: string): string {
return execSync(`agent-browser --cdp 9222 ${cmd}`, {
encoding: 'utf-8',
timeout: 30000,
}).trim()
}
function hasEngaged(platform: string, action: string, url: string): boolean {
try {
execSync(`bun run scripts/log-operation.ts check --platform ${platform} --action ${action} --url "${url}"`, {
stdio: 'ignore'
})
return true
} catch {
return false
}
}
function logEngagement(platform: string, action: string, url: string, note: string) {
execSync(`bun run scripts/log-operation.ts add --platform ${platform} --action ${action} --url "${url}" --status success --note "${note}"`, {
stdio: 'inherit'
})
}
// Navigate to page
ab('open https://x.com/search?q=AI%20agents&f=live')
ab('wait 3000')
// Get posts
const snapshot = JSON.parse(ab('snapshot -i -c -s \'article\''))
const posts = snapshot.slice(0, 5)
for (const post of posts) {
const postUrl = post.attributes?.['data-testid'] || `post-${Math.random()}`
// Skip if already engaged
if (hasEngaged('x', 'like', postUrl)) {
console.error(`Already liked ${postUrl}`)
continue
}
// Find like button
const likeBtn = post.children?.find((el: any) =>
el.attributes?.['data-testid'] === 'like'
)
if (likeBtn?.ref) {
ab(`click ${likeBtn.ref}`)
logEngagement('x', 'like', postUrl, 'AI agents search result')
ab('wait 2000') // Rate limiting
}
}#!/usr/bin/env bun
import { execSync } from 'node:child_process'
import { writeFileSync, readFileSync, existsSync } from 'fs'
const CHECKPOINT_FILE = '/tmp/workflow-checkpoint.json'
function loadCheckpoint(): any {
if (existsSync(CHECKPOINT_FILE)) {
return JSON.parse(readFileSync(CHECKPOINT_FILE, 'utf-8'))
}
return { step: 0, data: {} }
}
function saveCheckpoint(step: number, data: any) {
writeFileSync(CHECKPOINT_FILE, JSON.stringify({ step, data }))
}
const checkpoint = loadCheckpoint()
let currentStep = checkpoint.step
// Step 1: Fetch data
if (currentStep === 0) {
console.error('[workflow] Step 1: Fetching data...')
const data = { papers: ['paper1', 'paper2', 'paper3'] }
saveCheckpoint(1, data)
currentStep = 1
}
// Step 2: Process data
if (currentStep === 1) {
console.error('[workflow] Step 2: Processing data...')
const { data } = loadCheckpoint()
// Process papers
saveCheckpoint(2, { ...data, processed: true })
currentStep = 2
}
// Step 3: Post to social
if (currentStep === 2) {
console.error('[workflow] Step 3: Posting to social...')
const { data } = loadCheckpoint()
// Post each paper
saveCheckpoint(3, data)
currentStep = 3
}
// Cleanup checkpoint on success
if (existsSync(CHECKPOINT_FILE)) {
execSync(`rm ${CHECKPOINT_FILE}`)
}
console.log(JSON.stringify({ stepsCompleted: 3, stepsTotal: 3 }))# Check if Chrome is running with CDP
ps aux | grep chrome | grep remote-debugging-port
# Kill existing Chrome and restart
pkill -f chrome
bun run setup-chrome
# Verify CDP port is accessible
curl http://localhost:9222/json/version# Check log file exists and is readable
cat persona/operation-log.json
# Reset log if corrupted
echo '[]' > persona/operation-log.json
# Verify log script works
bun run scripts/log-operation.ts recent --limit 5# Check workflow status
bun run workflow status
# View detailed logs
bun run workflow history --id <workflow-id>
# Run with debug output
DEBUG=1 bun run workflow run --id <workflow-id>
# Check executor is executable
chmod +x workflows/executors/<executor>.ts# Verify API key is set
echo $OPENAI_API_KEY
# Test connection
curl -H "Authorization: Bearer $OPENAI_API_KEY" \
$OPENAI_BASE_URL/models
# Check model availability
bun start --model <model-name> -p "test"# Get detailed snapshot
agent-browser snapshot -i -c -s 'article' > snapshot.json
# Check element refs are valid
cat snapshot.json | jq '.[] | .ref'
# Try broader selector
agent-browser snapshot -i -c -s 'div'
# Wait for page to load
agent-browser wait 5000
agent-browser snapshot -i