Job-search-email-system-claude

skill
Guvenlik Denetimi
Basarisiz
Health Uyari
  • License — License: MIT
  • Description — Repository has a description
  • Active repo — Last push 0 days ago
  • Low visibility — Only 5 GitHub stars
Code Basarisiz
  • Hardcoded secret — Potential hardcoded credential in config.template.py
Permissions Gecti
  • Permissions — No dangerous permissions requested

Bu listing icin henuz AI raporu yok.

SUMMARY

Automated daily job search pipeline — aggregates company tracking, sends styled HTML email digests, tailors resumes with Gemini AI, drafts LinkedIn outreach messages, and scans remote job APIs for given location. Hot jobs for the given role and many features are made as claude skills too newly.

README.md

Claude Job Search Agent 🎯

An end-to-end automated job search pipeline that handles everything from finding opportunities to preparing applications. Sends daily styled HTML email digests from your Excel tracker, scans remote job APIs for EMEA-compatible roles, tailors resumes per company using Gemini AI, sends HR outreach emails with attachments, and drafts LinkedIn outreach messages — all on autopilot via Windows Task Scheduler.

Heavily driven by Claude Code Skills — 9 plain-language voice commands cover the entire workflow: adding jobs, rejecting, opening hot jobs, emailing HR, tailoring resumes, and more. No scripts to remember, no flags to type.

Features

  • Daily Email Reports - Styled HTML emails at 11:00 AM CET with companies grouped by role and status
  • Hot Jobs Section - Sticky listings from LinkedIn + Welcome to the Jungle + BuiltIn (fully configurable role categories, 5–8 slots per category) that persist until you add a company to your tracker, then backfill just that slot — 1 slot per category is always reserved for BuiltIn, with automatic fallback to WTTJ/LinkedIn if none found — each job shows a coloured source badge (🔵 LI / 🟢 WTTJ / 🟠 BuiltIn) — WTTJ is prioritised first within each location tier (hitsPerPage 30, source priority sort)
  • Remote Job Scanner - Fetches from RemoteOK, Remotive, We Work Remotely, Jobicy, LinkedIn (France/Global), and Bluedoor (free public ATS-aggregator API — Greenhouse/Lever/Ashby/Workday + 27 more) every 2 days, filters for EMEA-compatible roles. Bluedoor jobs are scoped to EMEA countries at the source, then description-verified (drops hard US-only roles, and annotates when a job's real scope is broader than its country tag, e.g. "Tagged Poland → actually global remote")
  • Fit Scoring - Every digest job gets a Gemini-scored fit badge (Strong/Good/Moderate/Weak %) against your resume, shown inline in both the daily and remote emails
  • Resume Tailor - Per-company tailored resumes using Gemini 2.5 Flash (free tier) — never fabricates, only reorders and surfaces existing skills
  • Outreach Drafter - Auto-generates short/medium/long LinkedIn message templates for each applied company
  • HR Outreach Emails - CLI script to send personalised cold outreach emails to HR contacts with resume + portfolio attached — auto-detects role from tracker, always previews before sending
  • Excel Integration - Reads your application tracker daily for statuses, role links, and HR contacts
  • HR Contact Management - Maintains recruiter contacts with clickable LinkedIn hyperlinks
  • Platform Aggregators - Curated search links (Glassdoor, LinkedIn, WelcomeToTheJungle, etc.)
  • Windows Automation - Runs on autopilot via Task Scheduler
  • Claude Code Skills (9) - Plain-language commands for every step: add jobs, reject, open hot jobs, email HR, tailor resume, run search — no flags, no scripts to remember, just say what you want

Email Previews

All data shown below is fictional.

Daily Job Report (11:00 AM)

Daily Email Sample

Companies grouped by status with clickable role links, HR contacts (purple), and quick search links. Platform aggregator links appear under each role category.

Remote Job Scanner (every 2 days, 12:00 PM)

Remote Email Sample

EMEA-compatible remote roles from RemoteOK, Remotive, and Arbeitnow APIs, filtered by your configured role keywords and location preferences.

What You Get

Roles Tracked

Roles are defined in daily_job_search.py — edit them to match your job search. The default setup includes Java backend and product management roles, but any stack or function works. Each role maps to a set of job board search links and hot jobs queries.

Status Tracking

  • ⬜ Not Contacted
  • 🕐 Under Review / In Progress
  • ✅ Applied
  • ⏸️ No Jobs Available
  • ❌ Rejected

🌍 Important: Customize for Your Location

This tool comes with sample job search links. Before using:

  1. Open daily_job_search.py
  2. Update the location variables at the top (lines 15-16):
    LOCATION_CITY = "YourCity"      # Change from "Paris"
    LOCATION_COUNTRY = "YourCountry" # Change from "France"
    
  3. Optionally update the job board URLs to match your location (see Customize URLs below)

Prerequisites

  • Python 3.7+
  • Gmail account with App Password enabled
  • Excel tracker (optional but recommended)
  • Windows OS (for Task Scheduler automation)

Quick Start

1. Install Python Dependencies

pip install openpyxl requests beautifulsoup4 python-docx

2. Configure Your Settings

IMPORTANT: Create your configuration file:

  1. Copy config.template.py to config.py
  2. Edit config.py with your details:
EMAIL_CONFIG = {
    'smtp_server': 'smtp.gmail.com',
    'smtp_port': 587,
    'sender_email': '[email protected]',        # ← Your Gmail
    'sender_password': 'YOUR_APP_PASSWORD',         # ← Gmail App Password (16 chars)
    'recipient_email': '[email protected]',      # ← Recipient email
}

TRACKER_FILE = r'C:\Path\To\Your\Tracker.xlsx'     # ← Your Excel tracker path

3. Get Gmail App Password

  1. Go to Google Account Security
  2. Enable 2-Step Verification (if not already enabled)
  3. Go to App passwords (at the bottom)
  4. Select "Mail" and your device
  5. Copy the 16-character password (no spaces)
  6. Paste it in config.py as sender_password

4. Test Run

python daily_job_search.py

Check your email - you should receive the job search report!

5. Schedule Daily Emails (11:00 AM)

Easy Setup (Recommended):

  1. Right-click setup_task_admin.bat
  2. Select "Run as administrator"
  3. Click "Yes" when prompted
  4. You should see "SUCCESS! Scheduled task created"

Alternative Methods:

  • Run setup_scheduled_task.ps1 (PowerShell version)
  • Manual setup via Task Scheduler GUI (see SETUP_INSTRUCTIONS.txt)

Excel Tracker Setup

Option 1: Use the Template (Recommended)

  1. Copy the template:

    # The repository includes tracker_template.xlsx
    # Copy it and customize with your companies
    
  2. Update config.py with your tracker path:

    TRACKER_FILE = r'C:\Path\To\Your\tracker.xlsx'
    

Option 2: Use Your Existing Tracker

Format your Excel file like this:

Company Role Role link status of application potentialHR contact Other comments
abc Backend general specialist https://... In progress Jane Smith
xyz Engineering Manager Java https://... Rejected
hogwarts Senior Software Engineer https://... done John Doe

Required Columns:

  • Column A: Company name (required)
  • Column B: Role title (can be empty, script will use "Various")
  • Column C: Application/job posting link (clickable in email report)
  • Column D: Status - use one of these:
    • "done" or "applied" → Shows as ✅ Applied
    • "In progress" or "Under Review" → Shows as 🕐 Review
    • "Rejected" → Shows as ❌ Rejected (apply strikethrough formatting)
    • "Not available" or "Nothing to apply" → Shows as ⏸️ No Jobs Available
    • Empty or anything else → Shows as ⬜ Not Contacted

Optional Columns:

  • Column E: potentialHR contact - recruiter/TA names with LinkedIn hyperlinks (see HR Contacts below)
  • Column F: Other comments

Tips:

  • You can add section headers (e.g., "Product Owner Roles") in Column A
  • Use strikethrough formatting on rejected companies (the script will detect it)
  • The script automatically categorizes roles based on keywords in Column B
  • Close Excel file before running the script to avoid permission errors

HR Contact Lookup

The system supports a potentialHR contact column (Column E) in your Excel tracker, showing recruiter/talent acquisition contacts for each company with clickable LinkedIn links in the email report.

How to Populate HR Contacts

Option A: Bulk populate with script (recommended for first setup)

  1. Copy update_hr_contacts.template.py to update_hr_contacts.py
  2. Research recruiters for your target companies (search LinkedIn for "CompanyName" "talent acquisition" OR "recruiter" YourCity site:linkedin.com/in)
  3. Fill in the HR_CONTACTS dictionary with (name, linkedin_url) tuples
  4. Run: python update_hr_contacts.py

This adds a potentialHR contact column to your Excel with clickable names.

Important: Web-scraped contacts are suggestions only — they may be outdated, in the wrong location, or no longer at the company. Always verify on LinkedIn before reaching out.

Option B: Add contacts directly in Excel

Simply type recruiter names into Column E of your Excel tracker. To make them clickable:

  • Select the cell, right-click → Link → paste the LinkedIn profile URL
  • The name will appear as a clickable link in the email report

Append-Only Workflow

The update_hr_contacts.py script uses append-only logic — it reads existing cell content first and only adds contacts that aren't already there. This means you can freely mix manual Excel edits with script updates:

Action What to do
Add a contact in Excel Just type it in Column E — the script will never overwrite it
Add a new company row Just add it in Excel — the script leaves unknown companies alone
Script finds new contacts New contacts are added to the HR_CONTACTS dict, then the script appends only the new ones
Remove a contact Delete from Excel and remove from the HR_CONTACTS dict in update_hr_contacts.py

The script is idempotent — running it multiple times won't create duplicates. Only truly new contacts get appended.

Note: Removal is the only case where you need to update both places (Excel + script). For everything else, your manual Excel edits are always preserved.

Auto-Update in Daily Email

The daily email script reads the potentialHR contact column fresh from your Excel every time it runs. This means:

  • Any contacts you add or edit in Column E are automatically included in the next email
  • Deleted contacts are removed from the next email
  • New company rows with HR contacts are picked up automatically

Note: update_hr_contacts.py is gitignored because it contains real contact data.

HR Outreach Emails

Send personalised cold outreach emails directly to HR/Talent Acquisition contacts — with your resume and portfolio attached — via a single CLI command.

python send_outreach_emails.py --name "Andrea Smith" --email "[email protected]" --company "Acme Corp"
# Optional CC:
python send_outreach_emails.py --name "Andrea Smith" --email "[email protected]" --company "Acme Corp" --cc "[email protected]"
# Optional explicit role (skips tracker lookup):
python send_outreach_emails.py --name "Andrea Smith" --email "[email protected]" --company "Acme Corp" --role "Senior Backend Engineer"

How it works:

  1. Role resolution (in priority order):
    • --role "..." if you pass it explicitly, or
    • the applied role looked up from your Excel tracker, or
    • if the company isn't tracked → falls back to the spontaneous template (candidature spontanée — no specific open role)
  2. Fills placeholders ({first_name}, {company}, {role}) in the chosen template
  3. Shows a full preview of the email in the terminal
  4. Asks for yes/no confirmation before sending — nothing goes out without your approval
  5. Attaches your resume and portfolio PDFs

Dead-link guard: if the job posting URL returns 404 / is taken down, the role no longer exists — don't send the role-specific outreach; switch to the spontaneous template or skip (the send-outreach skill enforces this).

Templates live in an emailoutreach/ folder (local only, gitignored — personal content):

  • cold_outreach_template.txt — rich outreach with bullet points about your background (default when a role is known)
  • cold_outreach_spontaneous_template.txt — candidature spontanée for companies with no tracked open role
  • followup_template.txt — short 3-line nudge (for second follow-up after no reply)

Setup — paths to configure in send_outreach_emails.py:

# Folder containing your resume and portfolio PDFs
RESUME_DIR = os.path.join(BASE_DIR, 'resume')

# Folder containing your email templates
TEMPLATE_DIR = os.path.join(BASE_DIR, 'emailoutreach')

# Your actual PDF filenames
ATTACHMENTS = [
    os.path.join(RESUME_DIR, 'your_resume.pdf'),
    os.path.join(RESUME_DIR, 'your_portfolio.pdf'),
]

Note: emailoutreach/ is gitignored — create it locally and add your own templates. resume/ is also gitignored — copy your PDFs there.


Outreach Drafter

After sending the daily email, the system automatically generates LinkedIn outreach drafts for all applied companies (status = "done") that have HR contacts.

Features:

  • Short messages (max 300 chars) per HR contact — for LinkedIn connection requests
  • Long messages per company — for InMail or email
  • Skip logic — won't regenerate drafts if roles haven't changed
  • Stable filenames — one file per company, overwritten only when roles change
  • All personal info comes from USER_PROFILE in config.py (no hardcoded data in code)

Output: Drafts are saved to the output folder configured in outreach_drafter.py.

Standalone run:

python outreach_drafter.py

Hot Jobs

The daily email includes a Hot Jobs section that surfaces fresh LinkedIn listings you haven't seen or applied to yet. It shows 5 jobs per role category (fully configurable), sorted by location priority (City > Country > Region).

How it works:

  • Fetches from LinkedIn's guest API, Welcome to the Jungle (Algolia backend), and BuiltIn EU/France — all three run per query for France/Paris searches
  • 1 slot per category is always reserved for BuiltIn — if no BuiltIn job is found, the slot falls back to WTTJ/LinkedIn
  • Company names for BuiltIn jobs are resolved by fetching the individual job page (page title extraction), and tracker-matched companies (already applied) are skipped with automatic fallback to the next BuiltIn candidate
  • Each job shows a coloured source badge in the email: LI (blue) for LinkedIn, WTTJ (green) for Welcome to the Jungle, BuiltIn (orange) for BuiltIn
  • Maintains a sticky list — the same jobs persist across runs until you act on them
  • When you add a company to your Excel tracker, that job drops out and gets backfilled with a fresh one
  • Only fetches when there are empty slots to fill (fast repeat runs)
  • Shown as an orange-themed section at the top of the daily email

Standalone check (no email sent):

python daily_job_search.py --hot-jobs

Remove a bad listing (e.g. closed, wrong language) and blocklist it:

python daily_job_search.py --hot-jobs --remove "Company" "Role title"

Only that specific company+role pair is blocklisted. Other roles from the same company can still appear.

Remove via Claude Code (recommended) — just say it in plain language:

"remove all senior java hot jobs, I applied to Theodo"

The remove-hot-job skill handles clearing slots and blocklisting dismissed jobs automatically.

Refresh all categories (clear and re-fetch):

python daily_job_search.py --hot-jobs --refresh

Refresh one category:

python daily_job_search.py --hot-jobs --refresh "Senior Java"

Customize queries in config.py (optional — defaults are used if omitted):

HOT_JOB_QUERIES = {
    'Senior Backend': [
        ('senior+backend+developer', 'YourCity, YourCountry'),
        ('senior+software+engineer', 'YourCountry'),
    ],
    'Product Manager': [
        ('product+manager', 'YourCity, YourCountry'),
    ],
}

Per-category title filter (HOT_JOB_TITLE_FILTERS in daily_job_search.py): Optionally restrict a category to only show titles containing specific keywords. Useful for broad searches (e.g. "project manager") where you only want IT/tech results:

HOT_JOB_TITLE_FILTERS = {
    'Project Manager': ['java', 'it', 'software', 'data', 'digital', 'cloud'],
}

Note: The LinkedIn guest API only returns card-level data (title, company, location, URL) — it does not read full job descriptions. Some irrelevant listings may appear; use --remove to blocklist them.

Resume Tailor

Automatically generates per-company tailored resumes using Gemini 2.5 Flash (free tier, $0 cost). Runs at the end of the daily pipeline, after outreach drafts.

What it does:

  • Reads applied companies (status = "done") with non-LinkedIn role links from the Excel tracker
  • Fetches job descriptions from those URLs (supports Workday, WelcomeToTheJungle, Salesforce, and other ATS sites via JSON-LD extraction)
  • Sends the resume + JD to Gemini to get minimal tailoring suggestions
  • Applies changes to a copy of your base DOCX resume and saves per-company files
  • Prints a bullet-point diff summary showing exactly what changed vs the original

Rules (hard-coded):

  • NEVER fabricates experience — only reorders skills and adds keywords from existing experience
  • Keeps the original job title heading unchanged
  • Removes the "Open Minded" line, keeps "Open to remote/hybrid"
  • Maximum 3 bullet tweaks per resume to keep changes minimal

Setup:

  1. Get a free Gemini API key at https://aistudio.google.com/apikey
  2. Add to config.py:
    GOOGLE_API_KEY = 'your_gemini_api_key'
    BASE_RESUME_PATH = r'C:\Path\To\Your\base_resume.docx'
    RESUME_OUTPUT_DIR = r'C:\Path\To\Your\resume_adjusted'
    
  3. Install dependency: pip install google-generativeai (optional, only needed for the SDK — the script uses the REST API directly)

Batch run (from tracker):

python resume_tailor.py

Single job (pass URL + company name):

python resume_tailor.py "https://company.workdayjobs.com/job/..." "Company Name"

Idempotent: Existing resume files are skipped. Delete a file to regenerate it.

Output:

  • Tailored DOCX: resume_{company_name}.docx
  • Change summary: summary_changes_resume_{company_name}.md — documents all changes applied (tagline, skills reorder, bullet tweaks, summary) plus raw Gemini suggestions

Auto-outreach: When running in single mode, outreach drafts are automatically generated after a successful resume tailor (no need to run outreach_drafter.py separately).

Claude Code Skills

Skills are the primary interface for this project. Instead of remembering script names, paths, and flags, you just describe what you want in plain language — Claude picks the right skill and executes the full workflow.

This project includes 9 skills covering every stage of the job search:

Tracker Management

Skill What it does Say something like...
new-job Adds a new job to List.xlsx with company, role, URL, and status "new job https://..." · "I applied to Qonto" · "add to tracker"
mark-rejected Marks a company as Rejected in the tracker "Deel rejected" · "mark Natixis as rejected"
search Checks if a company or URL is already in the tracker "have I applied to Datadog?" · "is this url in my tracker?"

Hot Jobs

Skill What it does Say something like...
open-hot-jobs Opens all (or a category of) hot job links in the browser "open all hot jobs" · "open backend java hot jobs"
remove-hot-job Removes a job from the hot list and blocklists it "remove senior java hot jobs" · "I applied to Theodo from hot jobs"
reject-job Blocklists a company/role from appearing in future hot jobs "hide this job" · "don't show Capgemini again"

Outreach & Applications

Skill What it does Say something like...
send-outreach Sends a cold outreach email to an HR contact — test preview first, then real send "send outreach to Andrea at Natixis" · "email HR at CFM" · "reach out to Mistral"
update-hr Searches LinkedIn for HR contacts at a company and adds them to the tracker "find HR for BNP" · "add recruiter for Société Générale"
resume-tailor Fetches the job description and generates a tailored resume via Gemini AI "tailor resume for Filigran" · "adapt my resume to this role"

Pipeline Control

Skill What it does Say something like...
test-run Manually triggers the daily email or remote job scanner "run daily jobs now" · "send the digest" · "test remote search"

How skills work: Each skill is a SKILL.md file in .claude/skills/ that encodes the full workflow — which script to run, what args to pass, what to check, what can go wrong. Claude reads it and follows it exactly. No slash commands, no flags to remember. Just say what you want.

Skills are version-controlled with the project and load automatically in every Claude Code session.

Architecture

Excel Tracker (List.xlsx)
    │
    ├── daily_job_search.py ──→ HTML email (11:00 AM daily)
    │       │
    │       ├── Hot Jobs ──→ LinkedIn guest API + Welcome to the Jungle (Algolia) + BuiltIn EU
    │       ├── outreach_drafter.py ──→ LinkedIn message drafts
    │       └── resume_tailor.py ──→ Tailored DOCX resumes
    │               │
    │               └── Gemini 2.5 Flash API (free tier)
    │
    └── remote_search/
            ├── remote_job_search.py ──→ HTML email + remote.xlsx append (every 2 days)
            │       │
            │       ├── RemoteOK + Remotive + WWR + Jobicy APIs
            │       ├── LinkedIn France (remote f_WT=2 filter)
            │       ├── LinkedIn Global (EMEA description verification)
            │       ├── Bluedoor (ATS-aggregator API, EMEA-scoped + description-verified)
            │       ├── fit_scorer.py ──→ Gemini fit badges (batched, thinking disabled)
            │       └── rejected_remote.json (filtered out before email/Excel)
            └── reject_remote.py ──→ CLI to manage rejected_remote.json

File Structure

claude-job-agent/
├── daily_job_search.py                # Main daily pipeline (hot jobs + email + outreach + resume)
├── daily_hot_jobs.json                # Sticky hot jobs state (auto-generated, gitignored)
├── outreach_drafter.py                # LinkedIn outreach draft generator
├── resume_tailor.py                   # Per-company resume tailoring via Gemini AI
├── send_outreach_emails.py            # CLI: send cold outreach emails to HR with PDF attachments
├── remote_search/
│   ├── remote_job_search.py           # Remote job API scanner (EMEA filter + Excel dump)
│   ├── reject_remote.py               # CLI to manage rejected jobs list
│   ├── rejected_remote.json           # Reviewed & rejected (company, title) pairs
│   ├── previous_jobs.json             # Last run's job keys for new-job detection (auto-generated)
│   └── run_remote_job_search.bat      # Scheduler wrapper for remote search
├── emailoutreach/                     # Email templates — LOCAL ONLY (gitignored, create your own)
│   ├── cold_outreach_template.txt     # Rich cold outreach with bullet points (default)
│   └── followup_template.txt          # Short follow-up nudge (second contact)
├── resume/                            # Resume + portfolio PDFs — LOCAL ONLY (gitignored)
│   ├── your_resume.pdf
│   └── your_portfolio.pdf
├── config.template.py                 # Configuration template (copy to config.py)
├── config.py                          # Your private configuration (gitignored)
├── update_hr_contacts.template.py     # HR contacts updater template
├── update_hr_contacts.py              # Your HR contacts data (gitignored)
├── run_daily_job_search.bat           # Scheduler wrapper for daily email
├── setup_task_admin.bat               # Easy setup for scheduled task
├── setup_scheduled_task.ps1           # PowerShell setup script
├── SETUP_INSTRUCTIONS.txt             # Detailed setup guide
├── tracker_template.xlsx              # Excel tracker template
├── job_search_log.txt                 # Execution log (auto-generated)
├── .gitignore                         # Excludes private files from Git
├── README.md                          # This file
└── .claude/
    └── skills/
        ├── new-job/SKILL.md           # Auto-triggered: add new job to tracker
        ├── mark-rejected/SKILL.md     # Auto-triggered: mark company as rejected
        ├── search/SKILL.md            # Auto-triggered: search tracker by company or URL
        ├── reject-job/SKILL.md        # Auto-triggered: reject/hide remote jobs
        ├── open-hot-jobs/SKILL.md     # Auto-triggered: open hot job links in browser
        ├── remove-hot-job/SKILL.md    # Auto-triggered: remove & blocklist hot jobs
        ├── test-run/SKILL.md          # Auto-triggered: manually run daily or remote search
        ├── update-hr/SKILL.md         # Auto-triggered: find and add HR contacts
        ├── resume-tailor/SKILL.md     # Auto-triggered: tailor resume for a company
        └── send-outreach/SKILL.md     # Auto-triggered: send cold outreach email to HR contact

How It Works

Daily Pipeline (11:00 AM)

  1. Read Excel - Loads your application tracker for statuses, role links, and HR contacts
  2. Fetch Hot Jobs - Queries LinkedIn + Welcome to the Jungle + BuiltIn for fresh listings per role category, filters out tracker companies, keeps a sticky list per category (5–8 slots, 1 always reserved for BuiltIn), only backfills when a slot opens — each job tagged with a coloured source badge in the email
  3. Organize - Groups companies by role and status (Not Contacted / Review / Applied / Rejected)
  4. Send Email - Styled HTML report via Gmail SMTP with hot jobs section + clickable links and HR contacts
  5. Outreach Drafts - LinkedIn message templates for applied companies with HR contacts
  6. Resume Tailor - Tailored DOCX resumes via Gemini AI for companies with fetchable job links

Remote Job Scanner (every 2 days, 12:00 PM)

  1. Fetch - Pulls listings from RemoteOK, Remotive, WWR, Jobicy, LinkedIn France, LinkedIn Global (India/Boston/NY), and Bluedoor (EMEA countries, remote-only)
  2. Filter - Matches role keywords + location-compatible positions (configurable in config.py)
  3. Dedup - Removes duplicates by company+title across sources
  4. EMEA Verification - For LinkedIn Global jobs, fetches each job's full description and checks for explicit EMEA timezone signals (emea, cet, work from anywhere, any timezone, etc.). Rejects US-only or no-timezone-info jobs. Bluedoor survivors get the same treatment via ?include=description — hard US-only clauses are dropped, and a digest note flags when the true scope is broader than the country tag (e.g. "Tagged Poland → actually global remote").
  5. Fit Scoring - One batched Gemini call scores every surviving job against your resume; badges render inline in the email.
  6. Excel Dump - Appends new jobs to remote_search/remote.xlsx (append-only, never touches List.xlsx)
  7. Send Email - Styled HTML table sorted by location tier (Paris → France → EMEA → UK → Global), new jobs highlighted in green

Reject remote jobs you've reviewed using reject_remote.py — entries are stored in rejected_remote.json and filtered out on every future run:

# Reject a single job
python remote_search/reject_remote.py "company name" "job title"

# Bulk-reject everything from the last run (after reviewing the email)
python remote_search/reject_remote.py --all

# Restore a job for review
python remote_search/reject_remote.py --remove "company name" "job title"

# List all rejections
python remote_search/reject_remote.py --list

# Show all commands
python remote_search/reject_remote.py --help

Test runs — use --no-save to avoid consuming the NEW flag before the scheduled run:

python remote_search/remote_job_search.py --no-save

Excel still gets appended; previous_jobs.json is left untouched.
Leave title as "" to reject all roles from a company. Entries are matched as lowercase substrings.

Excel dump — each run appends new remote jobs to remote_search/remote.xlsx (standalone file, never touches List.xlsx). Columns: Company, Role, URL, Source, Location, Tags, Posted, New?. Append-only — existing rows are never removed, deduped by company+title.

Customization

Customize Remote Job Filters

Edit config.py to match your skills and target region:

REMOTE_ROLE_KEYWORDS = ['java', 'backend', 'software engineer', 'devops', 'python']

REMOTE_LOCATION_INCLUDE = [
    'worldwide', 'anywhere', 'emea', 'europe', 'remote', 'global',
    'yourcountry', 'yourcity',  # ← Add your country/city
]

REMOTE_LOCATION_EXCLUDE = ['us only', 'us timezone', 'americas only']

If omitted, sensible defaults are used (see config.template.py).

Add More Companies

Simply add them to your Excel tracker! The script automatically:

  • Reads all companies from your Excel file
  • Categorizes them by role (Java Developer, Backend Developer, Product Owner)
  • Merges them with platform aggregator links
  • Updates daily with latest statuses

No code changes needed - just update your Excel file and the next email will include the new companies.

Customize for Your Location

Quick Setup (Just 2 Variables!):

Open daily_job_search.py and update lines 15-16:

LOCATION_CITY = "YourCity"         # e.g. "London", "Berlin", "Amsterdam"
LOCATION_COUNTRY = "YourCountry"   # e.g. "UK", "Germany", "Netherlands"

This automatically updates:

  • ✅ Email header location display
  • ✅ All job search link labels

Optional: Update Job Board URLs

If you want to customize the actual URLs for your location:

  1. Find the PLATFORM_AGGREGATORS section (around line 35)
  2. Update URLs for your region's job boards
  3. Add job boards popular in your country

Tip: The link labels already use your location variables, so you only need to update the URLs themselves.

Change Email Time

To change from 11:00 AM to another time:

  1. Delete existing task: schtasks /delete /tn "DailyJobSearch" /f
  2. Run setup_task_admin.bat again after editing the time in the file
  3. Or manually update via Task Scheduler GUI

Modify Email Format

Edit the HTML/CSS in the create_job_report() function in daily_job_search.py.

Troubleshooting

Email Not Sending

  • ✅ Check Gmail App Password is correct (16 characters, no spaces)
  • ✅ Verify 2-Step Verification is enabled in Google Account
  • ✅ Make sure config.py exists (copy from config.template.py)
  • ✅ Test with manual run: python daily_job_search.py

Excel Not Reading

Common Error: Permission denied: 'C:\\...\\List.xlsx'

This happens when Excel file is open. Solution:

  1. Close the Excel file before running the script
  2. ✅ Make sure Excel isn't running in the background
  3. ✅ If error persists, restart your computer

Other Excel issues:

  • ✅ Verify Excel file path in config.py is correct (use raw string: r'C:\Path\...')
  • ✅ Check the file exists at that path
  • ✅ Ensure openpyxl is installed: pip install openpyxl
  • ✅ Make sure file is .xlsx format (not .xls or .csv)

Empty Email Received?

  • If email is empty/shows 0 companies, the Excel file couldn't be read
  • Check the log: job_search_log.txt will show "Warning: Could not read tracker"
  • Close Excel file and run again

Task Scheduler Not Running

  • ✅ Check task exists: schtasks /query /tn "DailyJobSearch"
  • ✅ Verify task is enabled in Task Scheduler (taskschd.msc)
  • ✅ Run run_daily_job_search.bat manually to test
  • ✅ Check log file: job_search_log.txt
  • ✅ Make sure computer is ON at 11:00 AM

Script Errors

  • ✅ Make sure config.py exists (not just config.template.py)
  • ✅ Verify all paths use raw strings: r'C:\Path\...'
  • ✅ Check Python version: python --version (need 3.7+)

Verify It's Working

Check Scheduled Task:

Get-ScheduledTask -TaskName "DailyJobSearch"

Manual Test Run:

python daily_job_search.py

Check Logs:

type job_search_log.txt

Contributing

If you'd like to improve this project:

  1. Fork the repository
  2. Create a feature branch
  3. Make your changes
  4. Submit a pull request

Security Notes

⚠️ IMPORTANT:

  • Never commit config.py to Git (it's in .gitignore)
  • Never commit update_hr_contacts.py to Git (contains real recruiter names/LinkedIn URLs)
  • Keep your Gmail App Password secure
  • Don't share your Excel tracker if it contains private data

License

MIT License - Feel free to use and modify for personal use.

Support

For issues or questions:

  • Check the Troubleshooting section above
  • Review logs in job_search_log.txt
  • See SETUP_INSTRUCTIONS.txt for detailed setup help
  • Test manually first before debugging scheduler

Built with Claude Code | Last Updated: 2026-04-20


Claude Model Compatibility

This project was built and tested with Claude Code using:

Model Best for
Claude Opus 4.6 (claude-opus-4-6) Complex multi-step tasks: designing new features, debugging logic, writing EMEA verification, refactoring filter chains
Claude Sonnet 4.6 (claude-sonnet-4-5-20250929) Routine edits: blocklist entries, README updates, config tweaks, quick JSON changes

Both models work fully with this codebase. Switch between them in Claude Code with /model.

Yorumlar (0)

Sonuc bulunamadi