name: auto-apply
description: Search a job board and autonomously apply to matching jobs one at a time, until paused, exhausted, or the max-applications cap is hit.
argument-hint: "<search_query --board [--min-score N] [--max-apps N]> OR 'resume' OR 'retry-failed '"
Auto-apply — Search + Apply On Demand
Keep the chosen board open in tab 1; for each result that qualifies, apply in a second tab, then close it and move to the next job. No batch pre-discovery and no per-job approval — launching the campaign is the confirmation. Pause only for 2FA / payment. A CAPTCHA is not a pause — attempt the solve-captcha skill; if unsolved, skip the job (never pause) and the user finishes it later via the apply skill. Live view at http://localhost:8000/campaigns/<campaign-id>.
Setup
JOBPILOT_API=http://localhost:8000
Follow ../shared/setup.md. Read data.autoApply (defaults applied per field):
| Setting | Default | Notes |
|---|---|---|
minMatchScore |
60 | Qualification threshold (0–100). Inline --min-score overrides. |
maxApplicationsPerCampaign |
null (unlimited) |
Stop after this many successful applies. Inline --max-apps overrides; omit → unlimited. |
defaultStartDate |
"2 weeks notice" |
Default start-date answer. |
Inline argument overrides take precedence. --board <domain> is required unless the argument is resume or retry-failed <campaign-id>.
Campaign Modes
"resume"→ list incomplete campaigns (GET /api/campaigns?status=in_progress), ask which to resume, replay the apply loop on remainingapplying/approved/pendingjobs."retry-failed <campaign-id>"→ fetch the campaign; for everyfailedjob, PATCH back toapproved, readretryNotes, then replay the apply loop on them.- Otherwise → search query → Phase 0.
To recover wrongly-skipped jobs, use the dedicated rescan-skipped skill (it re-scores and promotes to approved; apply them afterward).
Phase 0: Existing Campaign Check + Create
curl -fsS "$JOBPILOT_API/api/campaigns?status=in_progress"
curl -fsS "$JOBPILOT_API/api/campaigns?status=paused"
curl -fsS "$JOBPILOT_API/api/campaigns?status=interrupted"
If any matches, ask "Found an incomplete campaign from <startedAt> (status: <status>). Resume or start fresh?" Resume → inject the resume skill with that campaignId.
Otherwise the web UI already created the campaign row when the user submitted /campaigns/new — confirm it exists and use that campaignId. If invoked manually (rare), create one:
SLUG=$(echo "<query>" | tr '[:upper:]' '[:lower:]' | sed 's/[^a-z0-9]/-/g; s/-\+/-/g; s/^-//; s/-$//')
CAMPAIGN_ID=$(date -u +%Y-%m-%dT%H-%M-%S_${SLUG})
# maxApplications is OPTIONAL — omit the field entirely for unlimited mode.
curl -fsS -X POST "$JOBPILOT_API/api/campaigns" \
-H 'content-type: application/json' \
-d "$(jq -n --arg id "$CAMPAIGN_ID" --arg q "<query>" --arg board "<domain>" \
--argjson minScore <n> \
'{campaignId:$id, query:$q, source:"auto-apply", config:{board:$board, minScore:$minScore}}')"
Surface live view: http://localhost:8000/campaigns/<CAMPAIGN_ID>.
Phase 1: Open the Board (tab 1)
1.1 Parse Query
Extract title/role, keywords, location, preferences. If vague, ask before searching.
1.2 Search the Chosen Board
Resolve the board:
curl -fsS "$JOBPILOT_API/api/job-boards" | jq --arg d "<domain>" '.data[] | select(.domain == $d)'
If no row matches, PATCH the campaign to failed with failReason:"Board <domain> not configured" and stop.
browser_navigatetosearchUrl(this is tab 1 — keep it open for the whole campaign).- Follow
../shared/auth.md— logs in, and registers a new account when none exists, without asking. - Fill the search fields and submit.
- Take a
browser_snapshotnarrowed to the results list (per../shared/browser-tips.md) to read{ title, company, location, url }per row. This is only the first viewport — scroll/paginate to load more rows as the loop drains them (see 2.5); never treat the first ~15 as all jobs.
Phase 2: Apply Loop (on demand)
Walk the tab-1 results top to bottom; at the last loaded row, scroll/page for more (1.2 step 4) before concluding. For each result:
2.1 Pre-filter (no tab)
Dedupe in-board by normalized title+company. Then check previously-applied:
URL_ENCODED=$(jq -rn --arg v "<job-url>" '$v|@uri')
TITLE_ENCODED=$(jq -rn --arg v "<title>" '$v|@uri')
COMPANY_ENCODED=$(jq -rn --arg v "<company>" '$v|@uri')
curl -fsS "$JOBPILOT_API/api/applied/check?url=$URL_ENCODED&title=$TITLE_ENCODED&company=$COMPANY_ENCODED"
If data.applied, add the job with status:"skipped", skipReason:"Already applied (<kind>)" and move on — don't open a tab.
2.2 Score
If the listing row lacks enough detail, read it from the tab-1 snapshot (don't navigate away). Build the digest — { title, company, techStack[], requirements[], responsibilities[], yearsExperience, descriptionExcerpt }; always populate techStack (it drives the score, empty → low score/confidence) — then score server-side:
FIT=$(curl -fsS -X POST "$JOBPILOT_API/api/score-fit" \
-H 'content-type: application/json' \
-d "$(jq -n --argjson digest "$DIGEST" '{digest:$digest}')")
SCORE=$(echo "$FIT" | jq -r '.data.score')
CONF=$(echo "$FIT" | jq -r '.data.confidence')
If CONF >= 0.7 and SCORE is ≥10 from minMatchScore either side, use it directly; otherwise rescore using strongMatches/partialMatches/gaps. A thin/generic row is not a skip — read the full posting (tab-1 detail or briefly in tab 2), rebuild the digest, and rescore first. Below minMatchScore after a fair read → add with status:"skipped", skipReason:"Below minimum match score ($SCORE < $MIN_SCORE)" and move on (no tab). Otherwise add it (status applying) and apply (2.3):
DIGEST=<stringified digest>
curl -fsS -X POST "$JOBPILOT_API/api/campaigns/$CAMPAIGN_ID/jobs" \
-H 'content-type: application/json' \
-d "$(jq -n --arg key "<stable-id>" --arg title "<title>" --arg company "<company>" \
--arg location "<location>" --arg url "<url>" --arg board "<board>" \
--arg matchReason "<one line>" --argjson score <0-100> --arg digest "$DIGEST" --arg desc "<posting text, when read>" \
'{key:$key, title:$title, company:$company, location:$location, url:$url, board:$board, matchScore:$score, matchReason:$matchReason, status:"applying", digest:$digest, description:$desc}')"
2.2a Eligibility — what is (and isn't) a skip
Valid skip reasons (exact phrasing):
Already applied (<kind>)— dedupe (2.1).Below minimum match score (X < Y)— only after reading the actual posting.- A hard requirement the JD itself states the user can't meet, e.g.
US citizenship required,Active security clearance required. Never infer from industry or company name. CAPTCHA — apply manually via the apply skill/Payment required(surface during apply, not scoring).
Never skip for: onsite/hybrid/other city when willingToRelocate is true or preferredLocations is empty/"Anywhere" (score on fit, not geography); a sparse JD (read and rescore first); 1099/contractor work; defense/federal industry absent a JD-stated citizenship/clearance requirement; a role below your level (Junior/Mid when your résumé is Senior) or one asking fewer years than you have — over-qualification is full marks on experience; judge on tech-stack fit.
2.3 Apply (tab 2)
Open a second tab: browser_tabs(action:"new"), then browser_navigate it to the job URL. Tab 1 stays on the results.
- Find Apply —
browser_snapshotthe header,browser_clickthe Apply / Easy Apply control'sref.browser_wait_for. If a new tab appeared (ATS portal),browser_tabs(action:"select", index:<new>). - Authentication — if a login/registration wall appears, follow
../shared/auth.md. On unrecoverable login failure for the domain: POST/resultoutcome:"failed",failReason:"Login failed for <domain>", close apply tab(s), continue. - CAPTCHA gate —
browser_snapshotthe apply form before tailoring or filling. If it shows a CAPTCHA (reCAPTCHA / hCaptcha / Turnstile, "I'm not a robot", image/puzzle), invoke thesolve-captchaskill. Solved → continue. Unsolved → skip: POST/resultoutcome:"skipped",skipReason:"CAPTCHA — apply manually via the apply skill", close apply tab(s), continue. Checking here avoids wasted tailoring/filling. - Tailor Resume — invoke the
tailor-resumeskill with$DIGEST(empty → fall back to the job URL). Capture the variant id + PDF URL. No usable base → POST/resultoutcome:"failed",failReason:"No tailorable resume base", close apply tab(s), continue. - Fill Forms — follow
../shared/form-filling.md. Upload the tailored variant. If the form has a cover-letter field (textarea or file upload), generate one via thecover-letterskill with$DIGEST(passsource:auto-apply) and fill it per form-filling.md (paste text, or upload a generated PDF). UseautoApply.defaultStartDate; ask once for salary expectation and remember it for the campaign. - Submit — submit autonomously,
browser_wait_for, then a narrowedbrowser_snapshot: a success confirmation = applied; a populated error = failure with that message asfailReason. A CAPTCHA at this stage → invokesolve-captcha; unsolved is a skip (2.4), not a failure.
2.4 Record + Close Tab
POST to /api/campaigns/$CAMPAIGN_ID/jobs/<key>/result (atomically updates the Job, creates Application on applied, marks the queue, recomputes summary):
NOW=$(date -u +%Y-%m-%dT%H:%M:%SZ)
# applied
jq -n --arg t "$NOW" --argjson score <0-100> '{outcome:"applied", appliedAt:$t, matchScore:$score}'
# skipped (CAPTCHA appeared — leave for manual apply via the apply skill)
jq -n '{outcome:"skipped", skipReason:"CAPTCHA — apply manually via the apply skill"}'
# failed (unexpected page, validation, crash)
jq -n --arg r "<reason>" --arg notes "<actionable retry context>" '{outcome:"failed", failReason:$r, retryNotes:$notes}'
Close all tabs with index ≥ 1: browser_tabs(action:"close", index:<i>) descending, then browser_tabs(action:"select", index:0). Pace 3–5s before the next job.
2.5 Stop Conditions
The loop ends only on one of these. Before picking the next result, refetch the campaign (GET /api/campaigns/<CAMPAIGN_ID>):
status === "paused"→ POST/resultoutcome:"skipped",skipReason:"Campaign paused by user"for any in-flightapplyingjob, exit.config.maxApplicationsset ANDsummary.applied >= config.maxApplications→ end. Unset (default) = no cap; keep going.- Board exhausted — only when a scroll/next-page attempt surfaces no new rows. First page done ≠ exhausted.
Phase 3: Summary
NOW=$(date -u +%Y-%m-%dT%H:%M:%SZ)
curl -fsS -X PATCH "$JOBPILOT_API/api/campaigns/$CAMPAIGN_ID" \
-H 'content-type: application/json' \
-d "$(jq -n --arg t "$NOW" '{status:"completed", completedAt:$t}')"
Print a summary table, link to http://localhost:8000/campaigns/<CAMPAIGN_ID>, suggest retry-failed <CAMPAIGN_ID>, the rescan-skipped skill on <CAMPAIGN_ID> to recover dropped jobs, or a new search.
Rules
- Autonomous after launch. No per-job or batch confirmation; the UI launch is the approval.
- Account handling — follow
../shared/auth.md: register when missing (without asking), forgot-password via theget-codeskill when stale. - Never process payments — POST
/resultoutcome:"failed",failReason:"Payment required". - Email codes — fetch automatically via the
get-codeskill for<board-domain>(see../shared/auth.md); only ask the user when it returns nothing. 2FA — pause and ask (one-time per board). CAPTCHA — never pause: detect up front per step 2.3.3, attempt thesolve-captchaskill, and skip the job for manual apply later only if it returns unsolved. - One job per tab. Board stays in tab 1; each application runs in tab 2, which is closed before the next job.
- Deduplicate within the board and against previously-applied before opening a tab.
- Pace 3–5s between submissions on the same domain.
- Audit trail. PATCH non-terminal transitions; POST
/resultfor terminal outcomes. - Respect pause. Re-read the campaign between jobs;
status === "paused"→ exit cleanly. - Missing resume file → PATCH campaign to
paused, ask the user to re-upload. - Eligibility — follow 2.2a. Location/onsite, thin JDs, 1099 work, and below-your-level/seniority are never skip reasons; only a JD-stated citizenship/clearance requirement disqualifies.
- Never skip silently. Every
skippedwrite carries a non-emptyskipReason(2.2a). No valid reason → not a skip.