Career-Ops – AI Job Search Pipeline
Career-Ops – AI Job Search Pipeline
Origin
This system was built and used by santifer to evaluate 740+ job offers, generate 100+ tailored CVs, and land a Head of Applied AI role. The archetypes, scoring logic, negotiation scripts, and proof point structure all reflect his specific career search in AI/automation roles.
The portfolio that goes with this system is also open source: cv-santiago.
It will work out of the box, but it’s designed to be made yours. If the archetypes don’t match your career, the modes are in the wrong language, or the scoring doesn’t fit your priorities – just ask. You (Claude) can edit any file in this system. The user says “change the archetypes to data engineering roles” and you do it. That’s the whole point.
What is career-ops
AI-powered job search automation built on Claude Code: pipeline tracking, offer evaluation, CV generation, portal scanning, batch processing.
Main Files
| File | Function |
|---|---|
data/applications.md | Application tracker |
data/pipeline.md | Inbox of pending URLs |
data/scan-history.tsv | Scanner dedup history |
portals.yml | Query and company config |
templates/cv-template.html | HTML template for CVs |
generate-pdf.mjs | Puppeteer: HTML to PDF |
article-digest.md | Compact proof points from portfolio (optional) |
interview-prep/story-bank.md | Accumulated STAR+R stories across evaluations |
reports/ | Evaluation reports (format: {###}-{company-slug}-{YYYY-MM-DD}.md) |
First Run — Onboarding (IMPORTANT)
Before doing ANYTHING else, check if the system is set up. Run these checks silently every time a session starts:
- Does
cv.mdexist? - Does
config/profile.ymlexist (not just profile.example.yml)? - Does
portals.ymlexist (not just templates/portals.example.yml)?
If ANY of these is missing, enter onboarding mode. Do NOT proceed with evaluations, scans, or any other mode until the basics are in place. Guide the user step by step:
Step 1: CV (required)
If cv.md is missing, ask:
“I don’t have your CV yet. You can either:
- Paste your CV here and I’ll convert it to markdown
- Paste your LinkedIn URL and I’ll extract the key info
- Tell me about your experience and I’ll draft a CV for you
Which do you prefer?”
Create cv.md from whatever they provide. Make it clean markdown with standard sections (Summary, Experience, Projects, Education, Skills).
Step 2: Profile (required)
If config/profile.yml is missing, copy from config/profile.example.yml and then ask:
“I need a few details to personalize the system:
- Your full name and email
- Your location and timezone
- What roles are you targeting? (e.g., ‘Senior Backend Engineer’, ‘AI Product Manager’)
- Your salary target range
I’ll set everything up for you.”
Fill in config/profile.yml with their answers. For archetypes, map their target roles to the closest matches and update modes/_shared.md if needed.
Step 3: Portals (recommended)
If portals.yml is missing:
“I’ll set up the job scanner with 45+ pre-configured companies. Want me to customize the search keywords for your target roles?”
Copy templates/portals.example.yml → portals.yml. If they gave target roles in Step 2, update title_filter.positive to match.
Step 4: Tracker
If data/applications.md doesn’t exist, create it:
# Applications Tracker
| # | Date | Company | Role | Score | Status | PDF | Report | Notes |
|---|------|---------|------|-------|--------|-----|--------|-------|
Step 5: Ready
Once all files exist, confirm:
“You’re all set! You can now:
- Paste a job URL to evaluate it
- Run
/career-ops scanto search portals- Run
/career-opsto see all commandsEverything is customizable — just ask me to change anything.
Tip: Having a personal portfolio dramatically improves your job search. If you don’t have one yet, the author’s portfolio is also open source: github.com/santifer/cv-santiago — feel free to fork it and make it yours.”
Then suggest automation:
“Want me to scan for new offers automatically? I can set up a recurring scan every few days so you don’t miss anything. Just say ‘scan every 3 days’ and I’ll configure it.”
If the user accepts, use the /loop or /schedule skill (if available) to set up a recurring /career-ops scan. If those aren’t available, suggest adding a cron job or remind them to run /career-ops scan periodically.
Personalization
This system is designed to be customized by YOU (Claude). When the user asks you to change archetypes, translate modes, adjust scoring, add companies, or modify negotiation scripts – do it directly. You read the same files you use, so you know exactly what to edit.
Common customization requests:
- “Change the archetypes to [backend/frontend/data/devops] roles” → edit
modes/_shared.md - “Translate the modes to English” → edit all files in
modes/ - “Add these companies to my portals” → edit
portals.yml - “Update my profile” → edit
config/profile.yml - “Change the CV template design” → edit
templates/cv-template.html - “Adjust the scoring weights” → edit
modes/_shared.mdandbatch/batch-prompt.md
Skill Modes
| If the user… | Mode |
|---|---|
| Pastes JD or URL | auto-pipeline (evaluate + report + PDF + tracker) |
| Asks to evaluate offer | oferta |
| Asks to compare offers | ofertas |
| Wants LinkedIn outreach | contacto |
| Asks for company research | deep |
| Wants to generate CV/PDF | pdf |
| Evaluates a course/cert | training |
| Evaluates portfolio project | project |
| Asks about application status | tracker |
| Fills out application form | apply |
| Searches for new offers | scan |
| Processes pending URLs | pipeline |
| Batch processes offers | batch |
CV Source of Truth
cv.mdin project root is the canonical CVarticle-digest.mdhas detailed proof points (optional)- NEVER hardcode metrics – read them from these files at evaluation time
Ethical Use – CRITICAL
This system is designed for quality, not quantity. The goal is to help the user find and apply to roles where there is a genuine match – not to spam companies with mass applications.
- NEVER submit an application without the user reviewing it first. Fill forms, draft answers, generate PDFs – but always STOP before clicking Submit/Send/Apply. The user makes the final call.
- Discourage low-fit applications. If a score is below 3.0/5, explicitly tell the user this is a weak match and recommend skipping unless they have a specific reason.
- Quality over speed. A well-targeted application to 5 companies beats a generic blast to 50. Guide the user toward fewer, better applications.
- Respect recruiters’ time. Every application a human reads costs someone’s attention. Only send what’s worth reading.
Offer Verification – MANDATORY
NEVER trust WebSearch/WebFetch to verify if an offer is still active. ALWAYS use Playwright:
browser_navigateto the URLbrowser_snapshotto read content- Only footer/navbar without JD = closed. Title + description + Apply = active.
Stack and Conventions
- Node.js (mjs modules), Playwright (PDF + scraping), YAML (config), HTML/CSS (template), Markdown (data)
- Scripts in
.mjs, configuration in YAML - Output in
output/(gitignored), Reports inreports/ - JDs in
jds/(referenced aslocal:jds/{file}in pipeline.md) - Batch in
batch/(gitignored except scripts and prompt) - Report numbering: sequential 3-digit zero-padded, max existing + 1
- RULE: After each batch of evaluations, run
node merge-tracker.mjsto merge tracker additions and avoid duplications. - RULE: NEVER create new entries in applications.md if company+role already exists. Update the existing entry.
TSV Format for Tracker Additions
Write one TSV file per evaluation to batch/tracker-additions/{num}-{company-slug}.tsv. Single line, 9 tab-separated columns:
{num}\t{date}\t{company}\t{role}\t{status}\t{score}/5\t{pdf_emoji}\t[{num}](reports/{num}-{slug}-{date}.md)\t{note}
Column order (IMPORTANT – status BEFORE score):
num– sequential number (integer)date– YYYY-MM-DDcompany– short company namerole– job titlestatus– canonical status (e.g.,Evaluated)score– formatX.X/5(e.g.,4.2/5)pdf–✅or❌report– markdown link[num](reports/...)notes– one-line summary
Note: In applications.md, score comes BEFORE status. The merge script handles this column swap automatically.
Pipeline Integrity
- NEVER edit applications.md to ADD new entries – Write TSV in
batch/tracker-additions/andmerge-tracker.mjshandles the merge. - YES you can edit applications.md to UPDATE status/notes of existing entries.
- All reports MUST include
**URL:**in the header (between Score and PDF). - All statuses MUST be canonical (see
templates/states.yml). - Health check:
node verify-pipeline.mjs - Normalize statuses:
node normalize-statuses.mjs - Dedup:
node dedup-tracker.mjs
Canonical States (applications.md)
Source of truth: templates/states.yml
| State | When to use |
|---|---|
Evaluated | Report completed, pending decision |
Applied | Application sent |
Responded | Company responded |
Interview | In interview process |
Offer | Offer received |
Rejected | Rejected by company |
Discarded | Discarded by candidate or offer closed |
SKIP | Doesn’t fit, don’t apply |
RULES:
- No markdown bold (
**) in status field - No dates in status field (use the date column)
- No extra text (use the notes column)
