Build Your Own Automated Job Search Alert AI Agent
Introduction
What is this project
Automatically analyze incoming job postings against your personalized profile and sends smart recommendations to Telegram.
Scan new jobs when they appear
AI filters jobs into tiers
You write applications (AI-written resumes get binned anyway)
Tech This Project Uses
How does this project work
ChangeDetection.io monitors job boards
n8n orchestrates AI analysis via Ollama
and Telegram delivers smart notifications sorted with a score
Acknowledging Derivative Work
I admit, we’re not building anything new.
We just want high-quality alerting.
But, traditional job alerts and email blasts are slow and noisy.
Just one quick note, 85% of jobs are filled through networking. So go our there and find your local Linux User Group!
Architecture Overview
I admit we’re mixing a lot of technology, so let me demonstrate with some ASCII:
┌─────────────────────────────────────────────────────────────────────────┐
│ JOB SEARCH AUTOMATION SYSTEM │
└─────────────────────────────────────────────────────────────────────────┘
┌──────────────────┐
│ JOB BOARDS │
│ (hiring.cafe) │
│ Indeed, LinkedIn│
└────────┬─────────┘
│
│ Monitors every 15min
│
┌────────▼─────────┐
│ ChangeDetection │◄──────┐
│ + Browser │ │ JavaScript
│ (Port 5000) │ │ Rendering
└────────┬─────────┘ │
│ │
│ Webhook POST │
│ (new jobs) ┌───┴───────────┐
│ │ Browserless │
┌────────▼─────────┐ │ Chrome │
│ │ │ (Port 3000) │
│ n8n Workflow │ └───────────────┘
│ (Port 5678) │
│ │
│ ┌─────────────┐ │
│ │ User Config │ │ ◄── Your profile, skills,
│ └──────┬──────┘ │ preferences, scoring
│ │ │
│ ┌──────▼──────┐ │
│ │Parse Webhook│ │ ◄── Clean job text
│ └──────┬──────┘ │
│ │ │
│ ┌──────▼──────┐ │
│ │AI Analysis │ │──────┐
│ └──────┬──────┘ │ │
│ │ │ │ Ollama API
└─────────┼────────┘ │ Call
│ │
│ ┌────▼─────────┐
│ │ Ollama │
│ │ (Port 11434) │
│ │ │
│ │ qwen2.5- │
│ │ coder:7b │
│ └────┬─────────┘
│ │
│◄──────────────┘
│ AI Response:
│ FIT SCORE: 8/10
│ RECOMMENDATION: Apply
│
┌─────────▼─────────┐
│ Check │
│ Recommendation │
└─────┬────────┬────┘
│ │
Skip │ │ Apply/Maybe
│ │
┌────────▼────┐ ┌─▼───────────┐
│ Telegram │ │ Telegram │
│ (Rejected) │ │ (Apply) │
│ Channel │ │ DMs │
└────────────┘ └─────────────┘
│ │
Low priority High priority
notifications notifications
Docker Compose Setup
This setup uses docker-compose to get up and running, the images can all be pulled down, but I did not adjust my networking stack.
This is classic works-on-my-machine documentation. You will have to adjust the networking to match your particular setup.
This is on a MacVLAN so each container may have it’s own Layer 2 connection on the host VLAN.
Prerequisites
- Docker and Docker Compose installed
- A domain name (optional, Traefik not included)
- About 6-8GB free disk space (Ollama models require 4-5GB each)
ChangeDetection.io
ChangeDetection.io needs two containers working together:
- changedetection: The main application that monitors websites
- changedetection-browser: A headless Chrome instance for JavaScript-heavy sites
n8n
n8n also needs two containers for this stack:
- n8n: Receives webhooks from ChangeDetection, orchestrates the AI analysis workflow, and sends Telegram notifications
- Ollama: Runs large language models locally to analyze job postings
Docker Components Relationship Map
Docker containers? Oh yeah! I have an ASCII picture for that too!!!
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
DOCKER CONTAINERS
═════════════════
┌─────────────────────────────────────────────────────────────┐
│ Docker Network: br1.224 (10.236.224.0/24) │
│ │
│ ┌────────────────────┐ ┌──────────────────────┐ │
│ │ changedetection │────▶│ changedetection- │ │
│ │ IP: 10.236.224.29 │ │ browser │ │
│ │ Port: 5000 │ │ IP: 10.236.224.30 │ │
│ │ │ │ Port: 3000 │ │
│ │ - Monitors URLs │ │ │ │
│ │ - Sends webhooks │ │ - Headless Chrome │ │
│ │ - Stores state │ │ - Renders JS sites │ │
│ └────────┬───────────┘ └──────────────────────┘ │
│ │ │
│ │ Webhook POST │
│ │ │
│ ┌────────▼───────────┐ ┌──────────────────────┐ │
│ │ n8n │────▶│ ollama │ │
│ │ IP: 10.236.224.26 │ │ IP: 10.236.224.XX │ │
│ │ Port: 5678 │ │ Port: 11434 │ │
│ │ │ │ │ │
│ │ - Workflow engine │ │ - AI model runner │ │
│ │ - Orchestration │ │ - qwen2.5-coder:7b │ │
│ │ - Telegram sender │ │ - Local inference │ │
│ └────────────────────┘ └──────────────────────┘ │
│ │
└─────────────────────────────────────────────────────────────┘
│
│ Internet
│
┌─────────▼──────────┐
│ Telegram Bot API │
│ (External) │
└────────────────────┘
Installation: ChangeDetection.io
Setting up ChangeDetection.io using a docker compose file, there are a lot of labels you may not need.
In addition please be aware of the custom docker network.
env file
Create a directory for your project and add these files:
.env file:
1
2
DOMAIN=example.com
SUBDOMAIN=unraid.example.com
docker-compose.yml for ChangeDetection.io
Replace with your actual domains. If you don’t have domains, you can access via IP addresses instead.
docker-compose.yml:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
services:
changedetection:
image: dgtlmoon/changedetection.io:latest
container_name: changedetection
hostname: changedetection
ports:
- "5000:5000"
volumes:
- /mnt/user/appdata/job_search:/datastore
labels:
- "traefik.enable=true"
- "traefik.http.services.changedetection.loadbalancer.server.port=5000"
- "traefik.http.routers.changedetection.rule=Host(`jobsearch.${DOMAIN}`) || Host(`changedetection.${SUBDOMAIN}`)"
- "traefik.http.routers.changedetection.service=changedetection"
- "traefik.http.routers.changedetection.entrypoints=websecure"
- "traefik.http.routers.changedetection.tls=true"
- "traefik.http.routers.changedetection.tls.certresolver=cloudflare"
- "traefik.http.routers.changedetection.tls.domains[0].sans=*.${DOMAIN}"
- "traefik.http.routers.changedetection.tls.domains[1].sans=*.${SUBDOMAIN}"
environment:
- BASE_URL=http://localhost:5000
- WEBDRIVER_URL=http://changedetection-browser:3000/webdriver
- PLAYWRIGHT_DRIVER_URL=ws://changedetection-browser:3000/websocket
- USE_X_SETTINGS=1
restart: unless-stopped
depends_on:
- changedetection-browser
networks:
br1.224:
ipv4_address: 10.236.224.29
changedetection-browser:
image: browserless/chrome:latest
container_name: changedetection-browser
hostname: changedetection-browser
ports:
- "3000:3000"
environment:
- CHROME_FLAGS=--disk-cache-size=0 --disable-application-cache
- CONCURRENT=10
- TOKEN=
- TIMEOUT=30000
- ENABLE_CORS=true
- DEFAULT_STEALTH=true
- ENABLE_DEBUGGER=false
- MAX_PAYLOAD_SIZE=30000000
- ENABLE_WEBDRIVER=true
restart: unless-stopped
shm_size: 1gb
networks:
br1.224:
ipv4_address: 10.236.224.30
networks:
br1.224:
external: true
name: br1.224
ipam:
config:
- subnet: 10.236.224.0/24
Key configuration points:
The entire stack needs:
- Port 5000: ChangeDetection.io web interface
- Port 3000: Browserless Chrome (internal use only)
- Playwright driver:
ws://changedetection-browser:3000/websocketenables JavaScript rendering - Shared network: Both containers must communicate on the same network
The browserless/chrome container needs sufficient resources:
- Shared memory:
1gb(prevents crashes on complex pages) - Concurrency:
10simultaneous browser sessions - Timeout:
30000ms(30 seconds per page load) - Stealth mode: Enabled (helps avoid bot detection)
These settings ensure reliable monitoring of modern job boards that heavily use JavaScript.
Network Configuration
This setup uses a custom bridge network (br1.224) with static IPs. You have two options for networking:
Option 1: Simple Bridge Network (Recommended for most users)
If you don’t need custom networking, use Docker’s default bridge:
1
2
3
networks:
default:
driver: bridge
Remove the ipv4_address lines from each service and remove the networks section at the bottom. Docker will automatically assign IPs.
Option 2: Custom Network with Static IPs (Advanced)
If you want static IPs for firewall rules or network management, use the configuration shown in the compose files above. This requires you to:
- Create the network first:
docker network create --driver=bridge --subnet=10.236.224.0/24 br1.224 - Adjust the subnet and IPs to match your network setup
- Keep the
ipv4_addressandnetworkssections as shown
For most users, Option 1 (simple bridge) is sufficient. The tutorial examples use Option 2 because they’re on a more complex home network setup.
Deploy the Stack
1
2
3
4
5
6
7
8
# Navigate to your project directory
cd /path/to/changedetection
# Start the containers
docker-compose up -d
# Verify they're running
docker ps | grep changedetection
You should see both changedetection and changedetection-browser running.
Access the Web Interface
Open your browser:
- With domain:
https://jobsearch.example.com - Without domain:
http://localhost:5000
You’ll see the ChangeDetection.io dashboard—empty for now, but ready to monitor job postings.
We’re not going to do anything with it for now, we’re going to finish setting up the rest of the docker components, n8n.
Installation: n8n and Ollama
n8n means “n-eight-n” → “n[odejsautomatio]n”
docker-compose.yml for n8n and Ollama
Create a separate directory for n8n and Ollama (or add to your existing compose file):
docker-compose.yml for n8n + Ollama:
IMPORTANT: This example uses custom networking (network_mode: br1.224). If you prefer simple networking, replace network_mode: br1.224 with a standard networks: section. See the Network Configuration section above for details.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
services:
n8n:
container_name: n8n
network_mode: br1.224 # Change this if using simple bridge networking
environment:
- TZ=America/Denver
- GENERIC_TIMEZONE=America/Denver
- N8N_SECURE_COOKIE=false
- EXECUTIONS_MODE=regular
- N8N_SKIP_WEBHOOK_DEREGISTRATION_SHUTDOWN=true
- WEBHOOK_URL=http://10.236.224.26:5678 # Adjust to your n8n IP/domain
- EXECUTIONS_TIMEOUT=-1
- EXECUTIONS_TIMEOUT_MAX=-1
volumes:
- /mnt/user/appdata/n8n:/home/node/.n8n:rw
restart: unless-stopped
image: n8nio/n8n
ollama:
container_name: ollama
network_mode: br1.224 # Change this if using simple bridge networking
environment:
- TZ=America/Denver
- OLLAMA_ORIGINS=*
- OLLAMA_NO_GPU=1
- OLLAMA_LOAD_TIMEOUT=90m
volumes:
- /mnt/user/appdata/ollama:/root/.ollama:rw
image: ollama/ollama
Alternative for simple bridge networking:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
services:
n8n:
container_name: n8n
ports:
- "5678:5678"
environment:
- TZ=America/Denver
- GENERIC_TIMEZONE=America/Denver
- N8N_SECURE_COOKIE=false
- EXECUTIONS_MODE=regular
- N8N_SKIP_WEBHOOK_DEREGISTRATION_SHUTDOWN=true
- WEBHOOK_URL=http://localhost:5678 # Adjust to your actual domain/IP
- EXECUTIONS_TIMEOUT=-1
- EXECUTIONS_TIMEOUT_MAX=-1
volumes:
- /mnt/user/appdata/n8n:/home/node/.n8n:rw
restart: unless-stopped
image: n8nio/n8n
networks:
- default
ollama:
container_name: ollama
ports:
- "11434:11434"
environment:
- TZ=America/Denver
- OLLAMA_ORIGINS=*
- OLLAMA_NO_GPU=1
- OLLAMA_LOAD_TIMEOUT=90m
volumes:
- /mnt/user/appdata/ollama:/root/.ollama:rw
image: ollama/ollama
networks:
- default
networks:
default:
driver: bridge
Key Configuration Points
n8n settings:
WEBHOOK_URL: Must match your actual n8n address (adjust IP/domain)EXECUTIONS_TIMEOUT=-1: Allows long-running AI analysis without timeout- Port
5678: Web interface and webhook receiver
Ollama settings:
OLLAMA_NO_GPU=1: Uses CPU only (remove if you have GPU)OLLAMA_ORIGINS=*: Allows n8n to connectOLLAMA_LOAD_TIMEOUT=90m: Prevents timeout when loading large models
Deploy and Verify
1
2
3
4
5
# Start the containers
docker-compose up -d
# Check they're running
docker ps | grep -E 'n8n|ollama'
Download AI Models
Ollama needs language models to analyze jobs.
- The 7B model in this guide uses about 4-5GB of disk space and runs well on systems with 8GB+ RAM
This uses less RAM than Chrome with 20 tabs open.
Access the Ollama container and download models:
1
2
3
4
5
6
7
8
# Enter the Ollama container
docker exec -it ollama /bin/bash
# List currently installed models (probably empty)
ollama list
# Pull recommended model for job analysis
ollama pull qwen2.5-coder:7b
Model recommendations:
qwen2.5-coder:7b: Best balance of quality and speed (recommended, ~4.7GB)qwen2.5-coder:1.5b: Faster, less RAM, briefer replies, runs on my GTX950 (~1GB)qwen2.5:14b: Highest quality, requires most resources (~8.5GB)
Note: These are the current model names as of January 2026. Check ollama.com/library for the latest available models.
The 7B model works well for analyzing job postings and runs on no-gpu systems with 8GB+ RAM.
Managing Disk Space
If you download a model that’s too large:
1
2
3
4
5
# Remove unwanted models
ollama rm model-name
# Check available space
df -h
Models range from 1GB (1.5B) to 8GB+ (14B+). Plan accordingly.
Access n8n Interface
Open your browser:
- With domain:
https://n8n.example.com - Without domain:
http://localhost:5678
First visit will prompt you to create an admin account. Complete the setup—you’ll import the workflow in the next section.
Yay! Everything works, now let’s get it to work for us how we need it to!!
Configuring ChangeDetection.io
Now we’ll set up ChangeDetection to monitor job boards and trigger our n8n workflow.
Building Target URLs
The example uses hiring.cafe with specific search parameters. Build your search URLs by:
- Going to your target job board
- Configuring all filters (location, salary, keywords, date range)
- Copying the complete URL from your browser address bar
Example: Cattle rancher jobs in the last 24 hours, near Dodge City, KS (100 mile radius):
1
https://hiring.cafe/?searchState=%7B%22searchQuery%22%3A%22cattle+-sheep%22%2C%22dateFetchedPastNDays%22%3A2%2C%22sortBy%22%3A%22date%22%2C%22maxCompensationLowEnd%22%3A%2244222%22%2C%22locations%22%3A%5B%7B%22id%22%3A%22rhk1yZQBoEtHp_8Uv--X%22%2C%22types%22%3A%5B%22locality%22%5D%2C%22address_components%22%3A%5B%7B%22long_name%22%3A%22Dodge+City%22%2C%22short_name%22%3A%22Dodge+City%22%2C%22types%22%3A%5B%22locality%22%5D%7D%2C%7B%22long_name%22%3A%22Kansas%22%2C%22short_name%22%3A%22KS%22%2C%22types%22%3A%5B%22administrative_area_level_1%22%5D%7D%2C%7B%22long_name%22%3A%22United+States%22%2C%22short_name%22%3A%22US%22%2C%22types%22%3A%5B%22country%22%5D%7D%5D%2C%22geometry%22%3A%7B%22location%22%3A%7B%22lat%22%3A37.7528%2C%22lon%22%3A-100.01708%7D%7D%2C%22formatted_address%22%3A%22Dodge+City%2C+KS%2C+US%22%2C%22population%22%3A27912%2C%22workplace_types%22%3A%5B%22Onsite%22%5D%2C%22options%22%3A%7B%22radius%22%3A100%2C%22radius_unit%22%3A%22miles%22%2C%22ignore_radius%22%3Afalse%2C%22flexible_regions%22%3A%5B%5D%7D%7D%5D%2C%22physicalPositions%22%3A%5B%22Standing%22%5D%7D
Example: Jobs at Agri Beef company:
1
https://hiring.cafe/?searchState=%7B%22dateFetchedPastNDays%22%3A2%2C%22sortBy%22%3A%22date%22%2C%22maxCompensationLowEnd%22%3A%2244222%22%2C%22companyNames%22%3A%5B%22agri+beef%22%5D%2C%22defaultToUserLocation%22%3Afalse%7D
These URLs are long because hiring.cafe includes all search parameters. You may find a job site you’re using doesnt.
ChangeDetection.io inlcudes actions. You can move the mouse, click, login, click a new menu, type words, hit search. Then scrape the page.
Creating a New Watch
- Open ChangeDetection.io web interface (http://localhost:5000 or your domain)
- Click ”+ Add” button in top right
- In the “URL” field, paste your complete job board URL
- Give it a name (e.g., “Cattle Jobs - Dodge City”)
- Add a tag (e.g., “cattle-rancher” or “agri-beef”)
- Now configure the detailed settings below
Request Settings
Navigate to the Request section in your watch settings:
Fetch Method:
- Dropdown selection:
Playwright Chromium/Javascript via 'ws://changedetection-browser:3000/websocket' - This is critical - regular HTTP won’t work for JavaScript-heavy sites
Wait seconds before extracting text:
- Value:
20 - This allows the job board to fully load and render all listings via JavaScript
Filters and Triggers
Navigate to Filters and Triggers section:
Text filtering - Enable these checkboxes:
- ✅ Added lines - captures new job postings
- ✅ Replaced/changed lines - captures updated postings
- ✅ Remove duplicate lines of text - prevents duplicate notifications
Leave unchecked:
- ❌ Removed lines
- ❌ All other options
This configuration ensures you only get notified about new or changed job listings, not removals or other noise.
Fetching Settings
Navigate to Fetching section:
Fetch Method (yes, it appears twice - set both):
- Select:
Playwright Chromium/Javascript via 'ws://changedetection-browser:3000/websocket'
Random jitter seconds ± check:
- Enable the checkbox
- Value:
1200(equals 20 minutes of random variance) - This means checks happen at slightly different times each interval, making the monitoring less predictable and more natural
Global Filters
Navigate to Global Filters:
Ignore whitespace:
- ✅ Enable “Ignore whitespace”
- This prevents false triggers from spacing/formatting changes that don’t affect content
Notifications Setup
This is where the magic happens - connecting ChangeDetection to n8n:
URLs field - Add both of these lines:
1
2
bot - tgram://11111111111:AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA/1111111111?parse_mode=Markdown
n8n - form://10.236.224.26:5678/webhook/job-posting
Important adjustments:
- Telegram line: Replace
11111111111:AAAA...with your actual Telegram bot token and chat ID (or remove this line if you only want n8n notifications) - n8n line: Replace
10.236.224.26:5678with your actual n8n server address and port - The webhook path
/webhook/job-postingmust match what you’ll configure in n8n (covered in next section)
Notification Title:
1
This includes the watch name and tag in the notification, helping you identify which search triggered.
Notification Body:
1
This sends only the new content detected (the actual job posting text), not the entire page.
Notification format:
- Select:
Plain Text - Do not use HTML or Markdown here, as we’ll process it with AI later
Understanding the Notification Flow
When ChangeDetection detects a new job:
- It captures the `` content (new text on the page)
- Sends it to both Telegram (optional) and n8n webhook
- n8n receives the webhook, passes it to Ollama for AI analysis
- n8n sends a smart notification to Telegram with the AI’s recommendation
Test Your Configuration
Before enabling the watch:
- Click “Fetch & Extract” button at the bottom of your watch settings
- Wait for the preview to load (may take 20-30 seconds with Playwright)
- Verify you see job listings in the extracted text
- Check that navigation menus, footers, and ads are NOT captured
- If you see too much noise, adjust CSS selectors or filters
Using Tags for Organization
If monitoring multiple job searches, use tags strategically:
By job type:
- Tag:
cattle-rancher - Tag:
tech-remote - Tag:
construction
By company:
- Tag:
agri-beef - Tag:
tyson-foods
By location:
- Tag:
dodge-city - Tag:
denver
Tags help you quickly identify which search triggered which notification.
Save and Activate
You must save and mark it active:
- Scroll to bottom of watch settings
- Click “Save” button
- In the main dashboard, find your watch
- Toggle the switch to “Active” (green)
- Set check frequency (e.g., check every 4 hours)
You should see “Last Checked” timestamp update after the first check runs. The system is now live.
Setting Up the n8n Workflow
This is where everything comes together. The n8n workflow receives job postings from ChangeDetection, analyzes them with AI, and sends smart notifications to Telegram.
Workflow Structure
The workflow has 8 nodes that process jobs in sequence:
- Webhook - Receives data from ChangeDetection
- User Config - Your profile, skills, preferences (customize this!)
- Parse Webhook Jobs - Cleans up the incoming job text
- AI Analysis - Sends to Ollama for intelligent evaluation
- Clean Response - Removes markdown formatting from AI output
- Check Recommendation - Routes jobs based on AI decision
- Send to Telegram - Apply Jobs - Jobs worth applying to
- Send to Telegram - Rejected Jobs - Jobs that don’t fit
Data Flow Diagram
n8n data ingestion diagram?
I think we’re all sold out; I can check in the back..
Oh wait, here’s one right here!
NEW JOB POSTED ON WEBSITE
│
▼
┌─────────────────────┐
│ ChangeDetection │ Playwright Browser
│ detects new text │ renders JavaScript
└──────────┬──────────┘
│
│ Webhook POST with job text
│ URL: http://n8n:5678/webhook/job-posting
│
▼
┌─────────────────────────────────────────────────────────┐
│ n8n WORKFLOW EXECUTION │
├─────────────────────────────────────────────────────────┤
│ │
│ 1. WEBHOOK RECEIVES: │
│ { │
│ title: "Cattle Jobs - Dodge City", │
│ message: "(added) Ranch Hand\n$45k..." │
│ } │
│ │
│ 2. USER CONFIG LOADS: │
│ - Profile: Ranch Hand, 32, Cheyenne WY │
│ - Skills: Cattle handling, Equipment... │
│ - Target salary: $35k-$55k │
│ - Fit score factors: +2 for target company... │
│ │
│ 3. PARSE WEBHOOK: │
│ Remove "(added)" tags, clean HTML entities │
│ Output: "Ranch Hand\n$45k..." │
│ │
│ 4. AI ANALYSIS: │
│ Send to Ollama with full profile context │
│ ┌────────────────────────┐ │
│ │ Ollama processes: │ │
│ │ - Job description │ │
│ │ - User profile │ │
│ │ - Scoring rules │ │
│ │ Returns: Analysis + Score │ │
│ └────────────────────────────┘ │
│ │
│ 5. CLEAN RESPONSE: │
│ Remove ```markdown``` formatting │
│ │
│ 6. CHECK RECOMMENDATION: │
│ IF contains "RECOMMENDATION: Skip" │
│ → Route to Rejected channel │
│ ELSE │
│ → Route to Apply channel │
│ │
└──────────────┬───────────┬───────────────────────────┘
│ │
▼ ▼
┌──────────────┐ ┌─────────────┐
│ Telegram │ │ Telegram │
│ -100123... │ │ 123456789 │
│ (Group) │ │ (DM) │
└──────────────┘ └─────────────┘
Importing the file for the n8n Workflow
- Open n8n web interface (http://localhost:5678 or your domain)
- Click “Add workflow” in top right
- Click the three dots menu (⋮) → “Import from File”
- Copy the complete JSON from the Appendix at the end of this document
- Paste and click “Import”
The workflow will appear with all nodes pre-configured.
⚠️ CRITICAL: After importing, you MUST update these placeholder values:
- Both Telegram nodes: Replace
YOUR_TELEGRAM_CHAT_IDandYOUR_TELEGRAM_CHAT_ID_FOR_SKIPSwith your actual chat IDs - AI Analysis node: Verify the Ollama base URL matches your setup
- Webhook node: Ensure the path matches your ChangeDetection configuration
Configuring Your Profile
This is the most important customization. Click the “User Config” node to edit these values:
Personal Information:
1
2
3
4
5
profileRole: "Ranch Hand / Cowboy"
profileAge: "32"
profileLocation: "Cheyenne, WY"
salaryMin: "35000"
salaryMax: "55000"
Replace these with your actual information.
Technical Skills:
List your actual skills in ranked order. The AI uses this to evaluate job requirements:
1
2
3
4
5
6
7
technicalSkills: "1. Animal Husbandry: Cattle handling, horseback riding, herd management, calving assistance
2. Equipment Operation: Tractors, ATVs, balers, fence stretchers, basic welding
3. Land Management: Fence repair, irrigation, pasture rotation, hay production
4. Livestock Care: Branding, castration, vaccinations, wound treatment, hoof care
5. Horsemanship: Breaking, training, shoeing basics, trailer loading
6. Maintenance: Equipment repair, building maintenance, vehicle upkeep
7. Weather/Seasons: Working in all conditions, drought management, winter feeding"
Target Job Titles:
List the job titles you’re searching for:
1
2
3
4
5
targetJobTitles: "1. Ranch Hand / Cattle Ranch Worker
2. Cowboy / Working Cowhand
3. Livestock Handler / Cattle Feeder
4. Farm Laborer / Agricultural Worker
5. Feedlot Worker / Cattle Operations"
Target Companies:
Prioritize companies by preference (High/Good/Consider):
1
2
3
4
5
targetCompaniesHigh: "King Ranch, Deseret Ranches, JA Ranch, Pitchfork Ranch, 6666 Ranch (Four Sixes), Padlock Ranch, YO Ranch, TA Ranch, Cheyenne Cattle Company, Wyoming Cattle Company"
targetCompaniesGood: "Local family ranches, Livestock auctions, Feedlot operations, Guest/Dude ranches with working cattle, Agricultural co-ops, Veterinary ranches, Horse breeding operations, USDA/BLM range management"
targetCompaniesConsider: "Rodeo stock contractors, Horse training facilities, Farm equipment dealers, Agricultural supply companies, Ranch management companies"
Key Projects:
Highlight your achievements:
1
2
3
4
5
keyProjects: "- Managed herd of 200+ head through calving season
- Built and maintained 15+ miles of barbed wire fencing
- Operated heavy equipment for hay harvest (500+ bales/season)
- Trained 3 green horses for ranch work
- Assisted with emergency veterinary care during blizzard conditions"
Career Preferences:
Define what you want and don’t want:
1
2
3
careerPreferencesYes: "Outdoor work, working with animals, hands-on physical labor, working independently or small crew, seasonal variety, ranch/rural living"
careerPreferencesNo: "Office work, desk jobs, strict 9-5 schedules, urban locations, corporate environments"
Fit Score Factors:
This controls how the AI scores jobs (point system):
1
2
3
4
5
6
7
8
9
10
fitScoreFactors: "+2 points: High priority ranch/company (King Ranch, Deseret, JA Ranch, Pitchfork, 6666, Padlock, YO Ranch, TA Ranch, Cheyenne/Wyoming Cattle Co)
+1 point: Good fit operation (Family ranches, Feedlots, Working guest ranches, Ag co-ops, Vet ranches, Horse breeding, USDA/BLM)
+2 points: Title matches target roles (Ranch Hand, Cowboy, Livestock Handler, Farm Laborer, Feedlot Worker)
+2 points: Strong skills match (3+ matching: cattle handling, riding, equipment operation, fence/land work, animal care)
+1 point: Rural/ranch location (Wyoming, Montana, Texas, Colorado, Nebraska, South Dakota)
+1 point: Salary in $35k-$55k range (or housing + salary)
+1 point: Livestock/cattle focus, outdoor work
-2 points: Indoor/office only, no animal work
-1 point: Urban location required
-1 point: No housing provided (when needed)"
Enjoyment Factors:
Guide the AI on what makes a job enjoyable for you:
1
2
3
enjoymentFactors: "- High: Working with cattle/horses, outdoor physical work, ranch setting, seasonal variety
- Medium: Mix of animal work and equipment operation, some fence/maintenance work
- Low: Mostly equipment operation only, limited animal contact, indoor work"
Configure the Webhook Node
Click the “Webhook” node:
Settings to verify:
- HTTP Method: POST
- Path:
job-posting - Respond: Immediately
The full webhook URL will be: http://10.236.224.26:5678/webhook/job-posting
This must match what you configured in ChangeDetection’s notification URLs.
Configure the AI Analysis Node
Click the “AI Analysis” node:
Ollama Settings:
- Model:
qwen2.5-coder:7b(or whichever model you downloaded) - Credentials: Click “Create New” and set:
- Base URL:
http://ollama:11434(if containers share a Docker network) orhttp://10.236.224.XX:11434(replace XX with your Ollama container’s IP if using custom networking) orhttp://localhost:11434(if running on same host with port forwarding)
- Base URL:
The AI prompt is already configured in the workflow to produce this output format:
1
2
3
4
5
6
7
8
9
10
11
🎯 Role: [Job Title]
📋 Company: [Company Name]
💰 Salary: [Salary Range or "Not Listed"]
🛠 Skills: [List 3-5 relevant skills from posting]
FIT SCORE: [X/10]
✅ Match: [2-3 brief reasons why good fit]
⚠️ Gaps: [1-2 brief concerns]
Enjoyment: [High/Medium/Low] - [Explanation]
RECOMMENDATION: [Apply/Maybe/Skip] - [Reasoning]
Setting Up Telegram
You need two Telegram destinations:
Node: “Send to Telegram - Apply Jobs”
- Chat ID: Your personal Telegram chat ID (positive number like
1234567890) - YOU MUST REPLACE THE PLACEHOLDER - Credentials: Create Telegram bot credentials
Node: “Send to Telegram - Rejected Jobs”
- Chat ID: A group/channel ID (negative number like
-1001234567890) - YOU MUST REPLACE THE PLACEHOLDER - Credentials: Use same bot credentials
⚠️ IMPORTANT: The imported workflow contains placeholder text YOUR_TELEGRAM_CHAT_ID and YOUR_TELEGRAM_CHAT_ID_FOR_SKIPS. You must replace these with your actual Telegram chat IDs or the workflow will fail.
Getting Telegram Credentials
Create a Telegram Bot:
- Open Telegram and message
@BotFather - Send
/newbotcommand - Follow prompts to name your bot
- BotFather gives you a token like:
111001010:AAAAAADDAAAAAZZAAAAWWWAAAADDDDAAEE - Save this token
Get Your Chat ID (for DMs):
- Message your bot: “Hello”
- Visit:
https://api.telegram.org/bot<YOUR_BOT_TOKEN>/getUpdates - Look for
"chat":{"id":1234567890}in the JSON response - That positive number is your personal chat ID
Get a Group/Channel ID:
- Create a Telegram group or channel
- Add your bot to the group/channel
- Send a message in the group
- Visit:
https://api.telegram.org/bot<YOUR_BOT_TOKEN>/getUpdates - Look for negative number like
"chat":{"id":-1001234567890} - That’s your group/channel ID
Add Telegram Credentials to n8n
- Click either Telegram node
- Click “Credential to connect with” → “Create New”
- Enter your bot token from BotFather
- Save the credential
- Both Telegram nodes can use the same credential
Testing the Workflow
Before activating:
- Click “Execute Workflow” button (top right)
- In the Webhook node, click “Listen for Test Event”
- Trigger a test by clicking “Check now” on a ChangeDetection watch
- Or manually send a POST request to the webhook URL
- Watch each node execute in sequence
- Verify the AI analysis appears correctly
- Check that Telegram messages are sent to correct channels
Test Input Example (if testing manually):
You can use the Python test server from the next section to generate sample job postings.
Save and Activate
- Click “Save” button (top right)
- Toggle workflow to “Active” (switch in top right)
- The webhook is now live and ready to receive jobs from ChangeDetection
Verify End-to-End Integration
- Manually trigger your ChangeDetection watch (click “Check now”)
- Wait for it to detect changes
- Check n8n “Executions” tab for new workflow runs
- Verify Telegram notifications arrive with AI analysis
- Confirm jobs route to correct channels (Apply vs Rejected)
Testing with a Web Server
Before relying on real job boards, test your setup with a controlled environment using a simple Python web server.
Why Test This Way?
- Immediate feedback: Change the text instantly and see if ChangeDetection catches it
- No rate limiting: Test as many times as needed
- Controlled environment: Know exactly what changed and when
- Debug workflow: Verify the entire pipeline works
Create the Test Server
Save this as test-job-server.py:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
#!/usr/bin/env python3
from http.server import BaseHTTPRequestHandler, HTTPServer
# Your block of text (use triple quotes for multi-line)
TEXT_CONTENT = """
Staff Site Reliability Engineer
United States
$129k-$161k/yr Remote Full Time
Ping Identity: A delivering secure and seamless digital experiences through a cloud identity platform.
Experience in scalable distributed systems, cloud platforms, Go, Docker and Kubernetes, automated deployments, Git teamwork, and CS degree or equivalent.
Go, Docker, Kubernetes, Git, CI/CD
"""
class TextHandler(BaseHTTPRequestHandler):
def do_GET(self):
if self.path == '/':
self.send_response(200)
self.send_header('Content-type', 'text/plain; charset=utf-8')
self.send_header('Content-Length', str(len(TEXT_CONTENT)))
self.end_headers()
self.wfile.write(TEXT_CONTENT.encode('utf-8'))
else:
self.send_response(404)
self.end_headers()
if __name__ == '__main__':
server = HTTPServer(('10.236.99.88', 3333), TextHandler)
print("Serving text at http://10.236.99.88:3333 (Ctrl+C to stop)")
server.serve_forever()
⚠️ IMPORTANT Configuration:
- IP:
10.236.99.88- Replace this with your server’s actual IP address. You can find your IP withip addr(Linux) oripconfig(Windows) - Port:
3333- Choose any available port on your system - Content: Replace
TEXT_CONTENTwith your test job posting
Run the Test Server
1
2
3
4
5
# Make it executable
chmod +x test-job-server.py
# Run the server
./test-job-server.py
You should see: Serving text at http://10.236.99.88:3333 (Ctrl+C to stop)
Configure ChangeDetection to Monitor It
- In ChangeDetection, click ”+ Add”
- Enter URL:
http://10.236.99.88:3333(replace with your actual IP) - Name it: “Test Job Server”
- Request settings: Use
Basic fast Plaintext/HTTP Client(no Playwright needed for plain text) - Wait seconds:
0(instant response) - Filters: Same as before (Added lines, Replaced lines, Remove duplicates)
- Notifications: Add your n8n webhook URL
- Check frequency: Set to 1 minute for quick testing
- Save and activate
Test the Complete Pipeline
Step 1: Baseline check
- Click “Check now” in ChangeDetection
- First check establishes the baseline (no notification)
Step 2: Make a change
- Edit
test-job-server.py - Modify the
TEXT_CONTENT(add a new line, change salary, etc.) - Save the file
- Restart the server: Press
Ctrl+C, then run./test-job-server.pyagain
Step 3: Trigger detection
- Click “Check now” again in ChangeDetection
- Within seconds you should see:
- ChangeDetection shows “Changes detected”
- n8n execution appears in “Executions” tab
- Telegram notification arrives with AI analysis
Step 4: Verify the analysis
- Check Telegram for the formatted analysis
- Verify it went to correct channel (Apply vs Rejected)
- Confirm the AI scoring makes sense for the test job
Complete Workflow Execution Timeline
Did you want a visual of what was discussed above?
I actually have a framed picture of one, if you want to come join me in the kitchen.
There it is, next to our 4ft indoor cactus.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
TIME: 00:00:00 ChangeDetection check starts
│
▼
[Fetch page with Playwright]
│
TIME: 00:00:20 Wait for JavaScript to load (20 seconds)
│
▼
[Extract text content]
│
▼
[Compare with previous check]
│
▼──── No changes? ──▶ EXIT (no notification)
│
│ Changes detected!
▼
TIME: 00:00:25 Send webhook to n8n
│
│
┌────────▼──────────────────────────────────────────┐
│ n8n WORKFLOW STARTS │
├────────────────────────────────────────────────────┤
│ TIME: 00:00:25 [Webhook] Receive data │
│ │ │
│ ▼ │
│ TIME: 00:00:26 [User Config] Load profile │
│ │ │
│ ▼ │
│ TIME: 00:00:27 [Parse] Clean job text │
│ │ │
│ ▼ │
│ TIME: 00:00:28 [AI Analysis] Call Ollama │
│ │ │
│ │ ┌──────────────────────────────┐ │
│ │ │ Ollama processes request │ │
│ │ │ Model: qwen2.5-coder:7b │ │
│ │ │ Time: 5-30 seconds │ │
│ │ │ (depends on job length) │ │
│ │ └──────────────────────────────┘ │
│ │ │
│ TIME: 00:00:45 [AI Analysis] Receive response │
│ │ │
│ ▼ │
│ TIME: 00:00:46 [Clean Response] Format output │
│ │ │
│ ▼ │
│ TIME: 00:00:47 [Check] Route by recommendation │
│ │ │
│ ├─────▶ Skip? ──▶ [Telegram: Rejected] │
│ │ │
│ └─────▶ Apply? ──▶ [Telegram: Apply] │
│ │
│ TIME: 00:00:50 [Telegram] Send notification │
│ │
└────────────────────────────────────────────────────┘
│
▼
TIME: 00:00:51 Notification arrives on your phone 📱
TOTAL TIME: ~51 seconds from detection to notification
Troubleshooting
ChangeDetection doesn’t detect changes:
- Verify the web server is running:
curl http://10.236.99.88:3333 - Check ChangeDetection can reach the server (network/firewall)
- Look at “Last check” details for errors
n8n workflow doesn’t trigger:
- Check n8n webhook URL in ChangeDetection notifications
- Verify n8n workflow is Active (not Inactive)
- Look at n8n execution logs for webhook receive confirmation
Telegram notifications don’t arrive:
- Verify bot token is correct in n8n credentials
- Check chat ID is correct (positive for DMs, negative for groups)
- Ensure bot was added to the group/channel
- Test with n8n’s “Test step” button first
AI analysis is wrong:
- Check Ollama container is running:
docker ps | grep ollama - Verify model is downloaded:
docker exec -it ollama ollama list - Review n8n execution logs for Ollama errors
- Adjust the User Config profile to better match test jobs
Troubleshooting and Optimization
The only other project I can recomend for this task is:
- https://github.com/Gsync/jobsync
Common Issues and Solutions
Issue: ChangeDetection detects too many false positives
Symptoms: Getting notifications for minor page changes (timestamps, view counts, ads)
Solutions:
- Add CSS selectors to isolate job listings
- Enable “Ignore text” filters for known noise patterns
- Increase “Random jitter” to reduce check frequency
- Use “Remove lines containing” to filter unwanted patterns
Issue: Jobs are being missed
Symptoms: Job appears on site but no notification received
Solutions:
- Verify “Wait seconds before extracting” is sufficient (increase to 30-40)
- Check if site uses infinite scroll or “Load more” buttons
- Use browser developer tools to verify content is in page source
- Consider using API endpoint instead of web scraping if available
Issue: Ollama is slow or timing out
Symptoms: n8n workflow takes minutes to complete or fails
Solutions:
- Switch to smaller model:
qwen2.5-coder:1.5binstead of7b - Enable GPU support in Ollama (remove
OLLAMA_NO_GPU=1) - Increase system RAM allocated to Docker
- Increase
OLLAMA_LOAD_TIMEOUTin docker-compose
Issue: AI recommendations are inaccurate
Symptoms: Getting “Apply” for poor fits or “Skip” for good opportunities
Solutions:
- Review and refine the User Config profile (skills, preferences, scoring)
- Adjust fit score point values to emphasize what matters most
- Add more specific examples to target companies
- Modify the AI prompt to focus on specific criteria
- Review several AI analyses and identify patterns in errors
Issue: Telegram notifications are noisy
Symptoms: Too many notifications, hard to track what’s important
Solutions:
- Use separate channels for different job types
- Adjust routing logic: only send 8+ fit scores to Apply channel
- Add summary notifications (daily digest instead of instant)
- Use Telegram’s mute feature for Rejected Jobs channel
Performance Optimization
Reduce resource usage:
- Use CPU-only Ollama for most models (adequate for job analysis)
- Set reasonable check intervals (4-6 hours, not every minute)
- Limit concurrent browser sessions in Browserless Chrome
- Archive old n8n executions: Settings → Executions → Retention
Speed up processing:
- Pre-load Ollama models at container startup
- Use keepalive in Ollama API calls to maintain loaded models
- Process jobs asynchronously if monitoring many boards
Improve accuracy:
- Regularly update your User Config as skills/preferences change
- Review false positives/negatives weekly and adjust scoring
- A/B test different AI prompts to find best results
- Fine-tune CSS selectors to capture cleaner job text
System Maintenance
Weekly tasks:
- Review Telegram channels for patterns in AI recommendations
- Check ChangeDetection for any broken watches
- Verify disk space isn’t filling up (Ollama models, n8n logs)
Monthly tasks:
- Update Docker images:
docker-compose pull && docker-compose up -d - Archive or delete old n8n workflow executions
- Review and optimize User Config based on job search results
- Test with new job boards or search parameters
As needed:
- Rotate Telegram bot tokens if compromised
- Backup n8n workflows (export JSON)
- Update Ollama models for improved AI quality
- Expand to additional job boards or company searches
Scaling Up
Monitor multiple job boards:
- Create separate ChangeDetection watches for each board
- Use tags to organize by source, location, or job type
- Consider separate n8n workflows for different industries
- Set staggered check times to distribute load
Support multiple users:
- Duplicate the entire stack per user (full isolation)
- Or: Share ChangeDetection/Ollama, separate n8n workflows per user
- Use different Telegram bots/channels per person
- Adjust Ollama concurrency for multiple simultaneous analyses
Advanced workflows:
- Auto-apply to jobs scoring 9-10 (with approval queue)
- Generate cover letters using AI based on job description
- Track application status and follow-ups
- Build a personal job database with historical fit scores
Conclusion
You now have a fully automated, AI-powered job search assistant that works 24/7 to find opportunities matching your exact profile.
What You’ve Built
✅ Saves time: No more manually reviewing hundreds of irrelevant job postings
✅ Never misses opportunities: Monitors continuously, even while you sleep
✅ Makes smarter decisions: AI analyzes each job against your specific skills and preferences
✅ Maintains privacy: Everything runs on your own hardware, no data sent to third parties
✅ Adapts to you: Fully customizable profile, scoring, and notification preferences
System components:
- ChangeDetection.io: Your tireless web watcher, monitoring job boards for new postings
- n8n: The orchestration engine connecting everything together
- Ollama: Local AI that understands your career goals and evaluates opportunities
- Telegram: Instant notifications wherever you are, organized by priority
Next Steps
Immediate:
- Let the system run for a week
- Review the AI recommendations for accuracy
- Tune the User Config and fit scoring based on results
- Add more job boards or search variations
Short-term:
- Experiment with different AI models for better analysis
- Create specialized workflows for different job types
- Set up daily digest summaries instead of instant notifications
- Build a tracking system for applications sent
Long-term:
- Expand to monitor company career pages directly
- Integrate with job application automation
- Add salary negotiation analysis and recommendations
- Build historical data to identify hiring patterns and best times to apply
Final Thoughts
Job searching is exhausting. This system handles the tedious work—monitoring, filtering, and initial evaluation—so you can focus on what matters: preparing strong applications for truly relevant opportunities.
The time investment (2-3 hours setup) pays off quickly. Most job seekers spend 10-20 hours per week browsing job boards. This system does that work continuously, notifying you only when something actually matches your profile.
Customize it, experiment with it, and make it yours. The stack is open-source and runs entirely under your control. No subscriptions, no data sharing, no limitations.
Happy job hunting! 🎯
Appendix: n8n Workflow JSON
Below is the complete n8n workflow that powers the job analysis system. This JSON file contains all 8 nodes pre-configured and ready to import into your n8n instance.
What’s Included
This workflow contains:
- Webhook receiver configured for
/webhook/job-postingendpoint - User Config node with example ranch hand profile (customize for your needs)
- Parse Webhook Jobs node with JavaScript to clean ChangeDetection output
- AI Analysis node with comprehensive prompt engineering for job evaluation
- Clean Response node to remove markdown formatting
- Check Recommendation node to route jobs by AI decision
- Two Telegram nodes for Apply and Rejected channels
How to Import
- Copy the entire JSON below
- Open your n8n interface (http://localhost:5678)
- Click “Add workflow” → Menu (⋮) → “Import from File”
- Paste the JSON and click “Import”
- Configure your credentials:
- Ollama API credentials (Base URL)
- Telegram bot token and chat IDs
- Customize the User Config node with your profile
- Save and activate the workflow
Important Customization Points
After importing, you must customize:
- User Config node: Replace all profile data with your actual information (role, skills, salary, preferences)
- Telegram - Apply Jobs node: Update Chat ID to your personal Telegram ID
- Telegram - Rejected Jobs node: Update Chat ID to your group/channel ID
- AI Analysis node: Verify Ollama credentials point to your Ollama instance
- Webhook node: Ensure the path matches your ChangeDetection notification URL
The workflow is named “Cowboy Job Search - Telegram - Must Use Single Job” in the example, but you can rename it after import.
n8n Job Search with ChangeDetection.io Workflow JSON
Here is the full Cowboy Job Search json file.
n8n workflow file 🤠
You can download this as file, or just paste it into a newly created workflow inside of n8n.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
{
"name": "Cowboy Job Search - Telegram - Must Use Single Job",
"nodes": [
{
"parameters": {
"httpMethod": "POST",
"path": "job-posting",
"options": {}
},
"type": "n8n-nodes-base.webhook",
"typeVersion": 2.1,
"position": [
-960,
32
],
"id": "cc3d86b6-7be8-49d4-ab9a-a882cb5e36ba",
"name": "Webhook"
},
{
"parameters": {
"values": {
"string": [
{
"name": "profileRole",
"value": "Ranch Hand / Cowboy"
},
{
"name": "profileAge",
"value": "40"
},
{
"name": "profileLocation",
"value": "Cheyenne, WY"
},
{
"name": "salaryMin",
"value": "35000"
},
{
"name": "salaryMax",
"value": "55000"
},
{
"name": "technicalSkills",
"value": "1. Animal Husbandry: Cattle handling, horseback riding, herd management, calving assistance\n2. Equipment Operation: Tractors, ATVs, balers, fence stretchers, basic welding\n3. Land Management: Fence repair, irrigation, pasture rotation, hay production\n4. Livestock Care: Branding, castration, vaccinations, wound treatment, hoof care\n5. Horsemanship: Breaking, training, shoeing basics, trailer loading\n6. Maintenance: Equipment repair, building maintenance, vehicle upkeep\n7. Weather/Seasons: Working in all conditions, drought management, winter feeding"
},
{
"name": "targetJobTitles",
"value": "1. Ranch Hand / Cattle Ranch Worker\n2. Cowboy / Working Cowhand\n3. Livestock Handler / Cattle Feeder\n4. Farm Laborer / Agricultural Worker\n5. Feedlot Worker / Cattle Operations"
},
{
"name": "targetCompaniesHigh",
"value": "King Ranch, Deseret Ranches, JA Ranch, Pitchfork Ranch, 6666 Ranch (Four Sixes), Padlock Ranch, YO Ranch, TA Ranch, Cheyenne Cattle Company, Wyoming Cattle Company"
},
{
"name": "targetCompaniesGood",
"value": "Local family ranches, Livestock auctions, Feedlot operations, Guest/Dude ranches with working cattle, Agricultural co-ops, Veterinary ranches, Horse breeding operations, USDA/BLM range management"
},
{
"name": "targetCompaniesConsider",
"value": "Rodeo stock contractors, Horse training facilities, Farm equipment dealers, Agricultural supply companies, Ranch management companies"
},
{
"name": "keyProjects",
"value": "- Managed herd of 200+ head through calving season\n- Built and maintained 15+ miles of barbed wire fencing\n- Operated heavy equipment for hay harvest (500+ bales/season)\n- Trained 3 green horses for ranch work\n- Assisted with emergency veterinary care during blizzard conditions"
},
{
"name": "careerPreferencesYes",
"value": "Outdoor work, working with animals, hands-on physical labor, working independently or small crew, seasonal variety, ranch/rural living"
},
{
"name": "careerPreferencesNo",
"value": "Office work, desk jobs, strict 9-5 schedules, urban locations, corporate environments"
},
{
"name": "fitScoreFactors",
"value": "+2 points: High priority ranch/company (King Ranch, Deseret, JA Ranch, Pitchfork, 6666, Padlock, YO Ranch, TA Ranch, Cheyenne/Wyoming Cattle Co)\n+1 point: Good fit operation (Family ranches, Feedlots, Working guest ranches, Ag co-ops, Vet ranches, Horse breeding, USDA/BLM)\n+2 points: Title matches target roles (Ranch Hand, Cowboy, Livestock Handler, Farm Laborer, Feedlot Worker)\n+2 points: Strong skills match (3+ matching: cattle handling, riding, equipment operation, fence/land work, animal care)\n+1 point: Rural/ranch location (Wyoming, Montana, Texas, Colorado, Nebraska, South Dakota)\n+1 point: Salary in $35k-$55k range (or housing + salary)\n+1 point: Livestock/cattle focus, outdoor work\n-2 points: Indoor/office only, no animal work\n-1 point: Urban location required\n-1 point: No housing provided (when needed)"
},
{
"name": "enjoymentFactors",
"value": "- High: Working with cattle/horses, outdoor physical work, ranch setting, seasonal variety\n- Medium: Mix of animal work and equipment operation, some fence/maintenance work\n- Low: Mostly equipment operation only, limited animal contact, indoor work"
}
]
},
"options": {}
},
"id": "45d83eee-b514-4660-bac6-9f3d20814139",
"name": "User Config",
"type": "n8n-nodes-base.set",
"typeVersion": 2,
"position": [
-752,
32
]
},
{
"parameters": {
"jsCode": "// ===================================================================\n// PARSE WEBHOOK JOBS - Extract job posting data from incoming webhook\n// ===================================================================\n// Input: Webhook body.message with (into) and (added) tagged lines\n// Output: Clean concatenated job posting text for AI analysis\n// ===================================================================\n\n// Get the message from the webhook (it comes through User Config node)\nconst message = $input.first().json.body?.message || '';\n\nif (!message) {\n return [{ json: { inputText: 'No message received' } }];\n}\n\n// Process each line (data already has real newlines, not literal \\n)\nconst result = [];\nconst allLines = message.split('\\n');\n\nfor (const line of allLines) {\n const trimmed = line.trim();\n \n // Skip empty lines and (changed) lines\n if (!trimmed || trimmed.startsWith('(changed)')) {\n continue;\n }\n \n // If line has (into) or (added), remove the tag and keep the content\n if (trimmed.startsWith('(into)')) {\n result.push(trimmed.replace(/^\\(into\\)\\s*/, ''));\n } else if (trimmed.startsWith('(added)')) {\n result.push(trimmed.replace(/^\\(added\\)\\s*/, ''));\n }\n}\n\nif (result.length === 0) {\n return [{ json: { inputText: 'No job content found' } }];\n}\n\n// Join with newlines and clean up HTML entities\nconst output = result.join('\\n')\n .replace(/&/g, '&')\n .replace(/</g, '<')\n .replace(/>/g, '>')\n .replace(/<\\/pre><\\/body><\\/html>/g, ''); // Remove trailing HTML\n\n// IMPORTANT: Pass through ALL the config data from User Config node\n// AND add the parsed inputText\nconst configData = $input.first().json;\n\nreturn [{ \n json: { \n ...configData, // Spread all config values\n inputText: output // Add parsed job text\n } \n}];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
-560,
32
],
"id": "4e0b044c-d844-401f-8bdb-8c06dc6388f9",
"name": "Parse Webhook Jobs"
},
{
"parameters": {
"modelId": {
"__rl": true,
"value": "qwen2.5-coder:7b",
"mode": "list",
"cachedResultName": "qwen2.5-coder:7b"
},
"messages": {
"values": [
{
"content": "=# Job Match Analysis\n\n## YOUR PROFILE (Quick Reference)\n**Role:** | **Age:** | **Location:** \n**Salary Target:** $-$\n\n**Technical Skills (Ranked by Expertise):**\n\n\n**Target Job Titles ( Market Focus):**\n\n\n**Target Companies (Prioritized):**\n- HIGH PRIORITY: \n- GOOD FIT: \n- CONSIDER: \n\n**Key Projects:**\n\n\n**Career Preferences:**\n✅ \n❌ \n\n---\n\n## INSTRUCTIONS FOR ANALYZING JOB POSTINGS\n\n**CRITICAL: Keep responses SHORT. User viewing on small mobile screen.**\n**CRITICAL: Output PLAIN TEXT ONLY. NO markdown formatting. NO code blocks. NO backticks.**\n**CRITICAL: DO NOT add explanatory notes, disclaimers, or additional commentary after your analysis.**\n**CRITICAL: STOP after the RECOMMENDATION line. DO NOT add \"NOTE:\" or any extra paragraphs.**\n\nThe job posting text will be provided below. Analyze it and respond with ONLY this format (use emoji, keep concise):\n\n\n\n\nOUTPUT FORMAT:\n🎯 Role: [Job Title]\n📋 Company: [Company Name]\n💰 Salary: [Salary Range or \"Not Listed\"]\n🛠 Skills: [List 3-5 relevant skills from posting]\nSummary:\n\n FIT SCORE: [X/10]\n✅ Match: [2-3 brief reasons why good fit, one line each]\n⚠️ Gaps: [1-2 brief concerns, one line each]\n Enjoyment: [High/Medium/Low] - [One sentence explaining why]\n\n RECOMMENDATION: [Apply/Maybe/Skip] - [One sentence reasoning]\n\n**RULES:**\n1. Use emoji at start of each section as shown above\n2. Keep each bullet point to ONE line maximum\n3. Be direct and honest about fit\n4. Focus on skill match first\n5. Consider work environment preferences\n6. Factor in location and remote work preference\n7. Total response should be readable on mobile in one glance\n8. Your final line must be the RECOMMENDATION line - nothing after it\n9. Do not add notes, disclaimers, summaries, or extra commentary\n10. Do not explain the scoring system or methodology\n\n**FIT SCORE FACTORS (calculate by adding points):**\n\n\n**SCORING GUIDE:**\n- 8-10/10 = Strong match, apply immediately\n- 6-7/10 = Good match, worth applying\n- 4-5/10 = Possible fit, apply if interested\n- 1-3/10 = Poor match, skip unless desperate\n\n**ENJOYMENT FACTORS:**\n\n\n---\n\n## JOB POSTING TO ANALYZE:\n\n"
}
]
},
"options": {}
},
"type": "@n8n/n8n-nodes-langchain.ollama",
"typeVersion": 1,
"position": [
-416,
32
],
"id": "5f4f8a29-3b38-4d95-b2c9-2538f3575e3f",
"name": "AI Analysis"
},
{
"parameters": {
"jsCode": "// Clean up AI response - remove markdown formatting\nconst content = $input.item.json.content || '';\n\n// Remove code block markers\nlet cleaned = content\n .replace(/```[a-z]*\\n?/gi, '') // Remove ```language\n .replace(/```/g, '') // Remove closing ```\n .replace(/`([^`]+)`/g, '$1') // Remove inline backticks\n .trim();\n\nreturn {\n json: {\n cleanedContent: cleaned\n }\n};"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
-144,
32
],
"id": "e28d17e6-2cb4-4672-9603-0813d421189b",
"name": "Clean Response"
},
{
"parameters": {
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": "",
"typeValidation": "loose",
"version": 1
},
"conditions": [
{
"id": "recommendation-check",
"leftValue": "=",
"rightValue": "RECOMMENDATION: Skip",
"operator": {
"type": "string",
"operation": "contains",
"singleValue": true
}
}
],
"combinator": "and"
},
"options": {
"looseTypeValidation": true
}
},
"type": "n8n-nodes-base.if",
"typeVersion": 2,
"position": [
64,
32
],
"id": "f0a7bba4-08c5-4624-8a01-d38cd16304b1",
"name": "Check Recommendation"
},
{
"parameters": {
"chatId": "YOUR_TELEGRAM_CHAT_ID",
"text": "=\n",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [
240,
144
],
"id": "d9994740-e86d-4bec-b069-f64de2e06e41",
"name": "Send to Telegram - Apply Jobs"
},
{
"parameters": {
"chatId": "YOUR_TELEGRAM_CHAT_ID_FOR_SKIPS",
"text": "=\n",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [
240,
-96
],
"id": "750a6ec4-ddd4-4784-8e7f-6db279edd3c1",
"name": "Send to Telegram - Rejected Jobs"
}
],
"pinData": {},
"connections": {
"Webhook": {
"main": [
[
{
"node": "User Config",
"type": "main",
"index": 0
}
]
]
},
"User Config": {
"main": [
[
{
"node": "Parse Webhook Jobs",
"type": "main",
"index": 0
}
]
]
},
"Parse Webhook Jobs": {
"main": [
[
{
"node": "AI Analysis",
"type": "main",
"index": 0
}
]
]
},
"AI Analysis": {
"main": [
[
{
"node": "Clean Response",
"type": "main",
"index": 0
}
]
]
},
"Clean Response": {
"main": [
[
{
"node": "Check Recommendation",
"type": "main",
"index": 0
}
]
]
},
"Check Recommendation": {
"main": [
[
{
"node": "Send to Telegram - Rejected Jobs",
"type": "main",
"index": 0
}
],
[
{
"node": "Send to Telegram - Apply Jobs",
"type": "main",
"index": 0
}
]
]
}
},
"active": false,
"settings": {
"executionOrder": "v1"
},
"tags": []
}
Congratulations - I Hope It Works
Playwright is rendering JavaScript, Ollama is scoring opportunities, and Telegram is queued up to buzz your phone.
None of the nonsense that clogs everyone’s inbox.
Your system might be on fire, but hopefully it’s monitoring, analyzing, and filtering.
Your infrastructure, your models, your rules. No subscriptions, no API limits, no external dependencies that disappear.
You earned them by creating and using this. Well done.
Correctly Customize the Constraints
Now you have actual matches that align with you.
If your matches dont align, you may have to ask an LLM to customize your User Config section.
I asked an LLM to review this website and post what I thought the answers to the User Config section should be:
answers
 about the massive queue of opportunities infront of you now.
