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 gooseworks-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. |
development
End-to-end skill that turns a single reference image into a fully-installed, example-rendered style preset for the goose-graphics composite. Analyzes the image, writes the slim style spec, registers it in styles/index.json, generates all 7 format examples using the standard brief, renders PNGs via Playwright, and updates examples/manifest.json. Invoke with /goose-graphics-create-style.
development
Evaluate YC batch companies for investment — scrapes the YC directory, researches each company and its founders (work history, LinkedIn, website), assesses founder-company fit, and exports to Google Sheets with priority rankings. Use when asked to evaluate YC companies, research a YC batch, screen startups, or do due diligence on YC companies.
tools
Take screenshots of any website using Notte browser automation. Use when asked to screenshot, capture, or snap a webpage.
development
Search the web, platforms, and datasets. Use when asked to search, find, look up, research, or discover information from the web, YouTube, Amazon, eBay, news, academic sources, or any online platform.