skills/capabilities/job-scraper/SKILL.md
Search for job postings across LinkedIn and Indeed. Use when users want to find open roles, monitor hiring signals, identify companies hiring for specific positions, or research competitor hiring activity. Returns job title, company, location, salary, description, seniority level, and direct apply URLs. No login or cookies required.
npx skillsauth add athina-ai/goose-skills job-scraperInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Search for job postings across LinkedIn and Indeed using Apify. Find open roles by keyword, location, company, or job type. Use for hiring signal detection, GTM research, or competitive intelligence.
No LinkedIn cookies. No Indeed login. Just search queries in, structured job data out.
Load this skill when:
Required for both LinkedIn and Indeed scraping. Set in .env:
APIFY_API_TOKEN=your_token_here
No LinkedIn cookies, Indeed login, or any platform credentials needed. That's the only setup.
This skill searches two job platforms via Apify actors:
| Source | Apify Actor | Best For | Cost |
|--------|------------|----------|------|
| LinkedIn | automation-lab/linkedin-jobs-scraper | B2B, tech, SaaS, enterprise roles. Has seniority level, job function, industries. | ~$0.002/job |
| Indeed | borderline/indeed-scraper | Broadest coverage. Richest data — salary, company details, ratings, contacts, street addresses. | ~$0.004/job |
Do NOT ask the user which source to use unless genuinely ambiguous. Decide based on context:
After deciding, tell the user which source(s) you're searching and why. Don't ask — inform.
Extract from the user's message:
If anything is ambiguous, pick reasonable defaults and tell the user what you chose. Do not ask clarifying questions for things you can reasonably infer.
automation-lab/linkedin-jobs-scraperAPI call:
curl -X POST "https://api.apify.com/v2/acts/automation-lab~linkedin-jobs-scraper/runs?token=$APIFY_API_TOKEN" \
-H "Content-Type: application/json" \
-d '{
"searchQuery": "AI engineer",
"location": "San Francisco",
"maxItems": 25
}'
Input fields:
| Field | Type | Description |
|-------|------|-------------|
| searchQuery | string | Job title or keywords (required) |
| location | string | City, state, or country (optional) |
| maxItems | integer | Max jobs to return (default: 50) |
Polling for results:
# Check run status (poll every 10s)
curl "https://api.apify.com/v2/acts/automation-lab~linkedin-jobs-scraper/runs/{RUN_ID}?token=$APIFY_API_TOKEN"
# When status is SUCCEEDED, fetch results
curl "https://api.apify.com/v2/datasets/{DATASET_ID}/items?token=$APIFY_API_TOKEN"
Output fields per job:
title — Job titlecompanyName — Company namecompanyLinkedinUrl — Company LinkedIn pagecompanyLogo — Logo URLlocation — City, statesalary — Salary text (when available)employmentType — Full-time, Part-time, Contract, etc.seniorityLevel — Entry, Mid-Senior, Director, Executive, etc.jobFunction — Engineering, Sales, Marketing, etc.industries — Industry classificationdescriptionText — Full job description (plain text)descriptionHtml — Full job description (HTML)applicantsCount — Number of applicantspostedAt — When posted (e.g., "6 days ago")url — Direct link to the LinkedIn job postingapplyUrl — Direct apply URLborderline/indeed-scraperAPI call:
curl -X POST "https://api.apify.com/v2/acts/borderline~indeed-scraper/runs?token=$APIFY_API_TOKEN" \
-H "Content-Type: application/json" \
-d '{
"query": "AI engineer",
"location": "San Francisco, CA",
"country": "us",
"maxResults": 25
}'
Input fields:
| Field | Type | Description |
|-------|------|-------------|
| query | string | Job title or keywords (required) |
| location | string | City and state (optional) |
| country | string | Lowercase 2-letter country code (required). Common: us, uk, ca, de, fr, in, au |
| maxResults | integer | Max jobs to return |
Important: The country field is required for Indeed. If the user doesn't specify a country, default to us. Use lowercase 2-letter codes only.
Output fields per job:
title — Job titlecompanyName — Company namecompanyDescription — Company descriptioncompanyNumEmployees — Company sizecompanyRevenue — Company revenue rangecompanyUrl — Company Indeed pagelocation — Object with city, postalCode, country, formattedAddressShort, latitude, longitude, streetAddresssalary — Object with salaryCurrency, salaryMin, salaryMax, salaryText, salaryType (hourly/yearly)descriptionText — Full job description (plain text)descriptionHtml — Full job description (HTML)datePublished — Posted date (YYYY-MM-DD)age — Human-readable age ("24 days ago")expired — Whether job is still activeisRemote — Remote flagjobType — Employment typejobUrl — Direct Indeed job URLapplyUrl — Direct apply URLrating — Company rating and review countemails — Contact emails (when available)attributes — Job attributes list (benefits, requirements, etc.)hiringDemand — Urgent hire / high volume hiring flagsIf the user asked for recent jobs, filter results by date:
postedAt field (e.g., "6 days ago") — parse the text to determine recency.datePublished field (YYYY-MM-DD) — compare against today's date.Remove jobs older than what the user requested. If no recency filter specified, still remove jobs older than 30 days by default to avoid stale data.
When searching both LinkedIn and Indeed, the same job may appear on both platforms. Deduplicate by matching:
Show results as a summary table:
Source: LinkedIn + Indeed (deduplicated)
Jobs found: {count}
Location: {location}
Search: "{query}"
| # | Title | Company | Location | Salary | Posted | Source |
|---|-------|---------|----------|--------|--------|--------|
| 1 | AI Engineer | Stripe | SF, CA | $200K-$300K | 3 days ago | LinkedIn |
| 2 | ML Engineer | Meta | Menlo Park, CA | $58.65/hr | Mar 14 | Indeed |
| ... |
After the table:
If the user wants more detail on a specific job, show the full description.
If the user wants to save results:
{search-term}-jobs-{YYYY-MM-DD}.csv
CSV columns:
title, company, location, salary, employment_type, seniority_level, posted_date, job_url, apply_url, description, source
Normalize fields across sources so the CSV has a consistent schema regardless of whether the job came from LinkedIn or Indeed.
| Search | LinkedIn Only | Indeed Only | Both Sources | |--------|-------------|------------|-------------| | 25 jobs | ~$0.05 | ~$0.10 | ~$0.15 | | 50 jobs | ~$0.10 | ~$0.20 | ~$0.30 | | 100 jobs | ~$0.20 | ~$0.40 | ~$0.60 |
LinkedIn is cheaper per job. Indeed returns richer data per job. Both together give the most complete picture.
Hiring signal detection: "Find companies hiring AI engineers in SF" → Search both sources, group by company, rank by number of open roles. Companies with 5+ AI roles are actively building.
Competitive intelligence:
"What is Anthropic hiring for?" → Search LinkedIn with searchQuery: "Anthropic". Shows their open roles, team growth, and strategic priorities.
Salary research: "What do ML engineers make in NYC?" → Search Indeed (richer salary data). Filter to NYC, aggregate salary ranges.
GTM prospecting: "Find companies hiring for VP of Sales" → These companies are scaling their sales org and may need sales tools. Export the company list for outreach.
| Error | Fix |
|-------|-----|
| APIFY_API_TOKEN not set | Ask user to add it to .env |
| Indeed: Missing country input | Add country field with lowercase 2-letter code (default: us) |
| LinkedIn: 0 results | Broaden search query or remove location filter |
| Indeed: 999 results returned | The maxResults field may not cap results. Filter client-side. |
| Apify run fails or times out | Retry once. If still fails, try the other source. |
| Stale results (30+ days old) | Apply recency filter. Warn user about data freshness. |
content-media
Takes an existing screen recording or demo video and adds professional zoom/pan effects synchronized to the narration. Uses transcript-driven zoom targeting and Remotion for rendering. Optionally replaces audio with a soundtrack.
tools
Repurposes long-form video (podcasts, interviews, talks) into short-form vertical clips for Instagram Reels, TikTok, and YouTube Shorts. Handles transcription, moment selection, clip extraction, speaker-tracked reframing (16:9 to 9:16), and animated captions.
development
Creates talking head videos from any source material (docs, changelogs, blog posts, notes, transcripts). Produces multi-scene videos with avatar narration over screenshots/images using HeyGen v2 API. Supports Quick Shot and Full Producer modes.
tools
Generates Instagram-ready product reels from any e-commerce product page URL. Scrapes product images, classifies by type, generates AI-animated clips via Higgsfield API, creates text overlays with style presets, and composes a 15-20 second reel with music. Supports model-based and product-only reels.