Elite Outsiders ← Tools menu

Crons

05-05-2026—UPDATE
eo-backup-worker CF · maintained by Claude Code
Legend — categories

Cron = Cloudflare Workers cron (eo-backup-worker, scheduled handler). Deterministic, server-side, zero LLM tokens. For mechanical jobs (count rows, push Notion API).

Routine = Claude Code scheduled task (~/.claude/scheduled-tasks/<id>/SKILL.md). Spawns a Claude session. Costs tokens. For reasoning jobs (audit, brief, review).

Event = Application code fired on a user or system event (page load, form submit, file upload, webhook). Not scheduled, not LLM.

CONTENT tag = any cron / routine / event that touches the content pipeline (scripts, shorts, carousels, quotes, thumbnails, scheduling, n8n workflows, etc.). Visual flag so the content-related automation surface is auditable in 1 glance. Cards carrying this tag should mirror in /backdata/content-system.

CF Worker code: ~/Projects/eo-backup-worker/.

Conventions — non negotiable

Single AL inbox — every automated email produced by any cron, routine or event MUST land at al@eliteoutsiders.com. Never alexandre.corne.ac@gmail.com (personal inbox, off-limits to ops noise).

How it's enforced — every Resend send in eo-backup-worker reads env.NOTIFY_TO with hardcoded fallback al@eliteoutsiders.com. Audit grep: grep -rnE "to:.*[\"'].*@" src/ — every match must end with @eliteoutsiders.com in the fallback string.

Override — if a single cron ever needs a different recipient, set a per-cron secret (e.g. NOTIFY_TO_WEEKLY) via wrangler secret put on the worker. Do not change the fallback. Last audit: 2026-05-08 — 4 stale fallbacks fixed (weekly-snapshot, events-archive, url-audit, personal-ops).

CRONS11 cards · 10 CF Workers + 1 local launchd
CF Workers — 9 deployed

CRON1 — Security backup

CRON MONTHLY 01st @ 03h UTC
03:00 UTC (≈ 04-05h Paris)

Monthly Kit + D1 backup to private R2. Quarterly dry-run on preview DB. Email recap.

Technical details
  • Photocopies the Kit newsletter (all subscribers + tags) and the site DB (D1), stores 4 files in a private Cloudflare R2 drawer. Like a family photo kept safe.
  • 12-month retention. On Jan/Apr/Jul/Oct 1st, runs a restore dry-run on a preview DB to verify backups are readable (quarterly dry-run).
  • Each run sends a Resend recap email (✅ ok / ⚠️ partial / ❌ failed) listing what was backed up.
cron: 0 3 1 * * · R2 target: eo-backups bucket · doc: Notion ch. 13 (Monitoring/alerting) + sub-page Setup backup mensuel Kit + D1 — 2026-04-30

CRON2 — Cal.com slot watcher

CRON DAILY 07h UTC
07:00 UTC (≈ 08-09h Paris)

Daily check that Cal.com slots are bookable across the 3 brands. Email if any slot is dead.

Technical details
  • Watches the 3 public Cal.com pages (byCaliber / Forja / EO) and counts available slots over the next 4 weeks.
  • If a slot is dead (another booking blocks it), sends a "DEAD" email so you can decide what to do.
  • If you fix the conflict (or weekly chaos moves the slot), you get a "REVIVED" email at next check. No email when all is well — radio silence.
  • Memory in Cloudflare KV (key cal_slot_health:v2) — stores state per (event × date) to detect transitions.
cron: 0 7 * * * · code: eo-backup-worker/src/cal-monitor.js · manual trigger: POST /admin/cal-monitor-now?token=...

CRON3 — Weekly chaos magician

CRON WEEKLY SUN 21h UTC
21:00 UTC (≈ 22-23h Paris)

Weekly chaos magic: re-randomize Cal.com slot times + brand rotation across the 4 weeks ahead.

Technical details
  • Randomizes slot times for MON afternoon / WED morning / FRI evening over the next 4 weeks, and rotates byCaliber / Forja / EO across the 3 daily 30-min slots.
  • Goal: every visitor returning to cal.com a week later sees different times. Creates a "real busy life" feel rather than "well-oiled 3 fixed slots machine".
  • Time bounds in the STARTS constant in code (MON 13:00-16:00 / WED 10:00-11:00 / FRI 18:30-19:00). Edit + redeploy to change a bound. Recap email after each run.
cron: 0 21 * * SUN · code: eo-backup-worker/src/cal-randomize.js · manual trigger: POST /admin/cal-randomize-now?token=...

CRON4 — Weekly report

CRON WEEKLY MON 08h UTC
08:00 UTC (≈ 09-10h Paris) · LIVE since 2026-05-01

Monday morning report: 7-day counters + delta vs prev 7d + top pages, posted to Notion + email.

Technical details
  • Every Monday morning, reads the DB directly (D1 binding, not via dashboard — gated by CF Access), builds a Markdown with 7d counters + delta vs prior 7d + top 10 pages, and creates a Notion sub-page under the operational hub. Then sends a recap email to alexandre.corne.ac@gmail.com (override via NOTIFY_TO_WEEKLY).
  • Resilience: if Notion is down (token/parent_id missing or integration not connected), the email goes out anyway with the markdown embedded — AL never loses the info.
  • Code: eo-backup-worker/src/weekly-snapshot.js + Notion helper src/notion.js. Email subject: EO weekly — YYYY-WXX.
cron: 0 8 * * 1 · code: eo-backup-worker/src/weekly-snapshot.js · manual trigger: POST /admin/weekly-snapshot-now?token=...

CRON5 — Monthly mover

CRON MONTHLY 01st @ 02h UTC
02:00 UTC · LIVE since 2026-05-01

Monthly archive of prior-month events to R2 + Notion, then DELETE old rows from D1.

Technical details

⚠ Heads-up — this cron DELETEs rows from the live events D1 table after archiving them to R2. Safe in normal conditions (R2 backup-first, current-month events untouched, R2-fail-HARD prevents data loss), but it's the only cron on this page that touches data destructively. Yellow button = think before you click.

  • On the 1st of each month at 02h UTC (1h BEFORE the monthly backup 0 3 1 * * — by design so the R2 dump captures an already-cleaned table), takes prior-month events, dumps JSON to R2 (eo-backups/events-archive/YYYY-MM/), creates a Notion archive page with aggregations (event_type / top paths / activity per day), then only after that runs DELETE FROM events WHERE created_at < <month_start>.
  • Why: the D1 base grows fast (5,518 rows in 15 pre-launch days). Without cleanup we pay to store data we never look at live — aggregates land in the Notion page + the weekly report.
  • Manual precedent: Analytics Archive — Pre-2026-05-01 (5,518 rows archived 2026-04-30) — format mirrored by renderNotionBlocks().
  • Absolute safety (backup-first): R2 fail = HARD, the DELETE doesn't run. Notion fail = soft, the DELETE runs anyway (the JSON on R2 is the official safety net; Notion is read comfort). The recap email always shows the exact state of the 3 steps (R2 / Notion / DELETE).
  • Pull an archive: wrangler r2 object get eo-backups/events-archive/2026-04/events-archive-2026-05-01.json
cron: 0 2 1 * * · code: eo-backup-worker/src/events-archive.js · manual trigger: POST /admin/events-archive-now?token=... (idempotent on mid-month clicks; current-month events are never touched, R2-fail-HARD prevents data loss if backup fails)

CRON6 — Page Flow crawler

CRON WEEKLY MON 04h UTC
04:00 UTC (≈ 05-06h Paris) · LIVE since 2026-05-03

Monday crawl of eliteoutsiders.com sitemap, upsert page_inventory + button_inventory in D1.

Technical details
  • Every Monday at 04h UTC, this robot fetches eliteoutsiders.com/sitemap.xml, walks each public page, and extracts every clickable element (<a> / <button> / [role="button"] / [data-track-id]). Result: complete inventory of pages + buttons in D1 (page_inventory + button_inventory), even buttons that have never been clicked.
  • Runs 4h BEFORE the weekly-snapshot (08h UTC) so the weekly report benefits from a freshly updated inventory.
  • Automatic stale-cleanup: any page removed from the sitemap is removed from the inventory (with its buttons). Skipped if the sitemap fails (guard against accidental wipe).
  • Read by tools.eliteoutsiders.com/page-flow (User Journey / Traffic Flow). Also triggerable manually via the "Refresh now" button on that page.
cron: 0 4 * * 1 · code: eo-backup-worker/src/page-flow-crawl.js · manual trigger: POST /admin/page-flow-crawl-now?token=...

CRON7 — URL audit

CRON MONTHLY 01st @ 04h UTC
04:00 UTC · LIVE since 2026-05-03 (split off from the monthly backup that same day)

Monthly crawl of every known URL (open + me + admin + tools), email anomaly report.

Technical details
  • On the 1st of each month at 04h UTC (1h after the backup 0 3 1 * * to avoid loading the network during the R2 upload), this robot crawls every known URL of the site (open + me + admin + tools), compares against the EXPECTED baseline in the code, and sends a Resend email summary to AL.
  • Performs ZERO auto-fix. Pure audit: "here are the URLs that respond differently than expected, up to you whether you fix or update the baseline".
  • To extend the URL list: edit EXPECTED in eo-backup-worker/src/url-audit.js.
cron: 0 4 1 * * · code: eo-backup-worker/src/url-audit.js · manual trigger: POST /admin/url-audit-now?token=...

CRON8 — Analytics map refresh

CRON SEMI-ANNUAL 01/01 + 01/07 05h
05:00 UTC · LIVE since 2026-05-05

Semi-annual snapshot of D1 row counts per analytics table, append to Notion source-of-truth.

Technical details
  • Twice a year (1st of January + 1st of July, 05:00 UTC), this robot queries D1 for row counts + last-write timestamp on every analytics table, and the breakdown of event_type values in the events table (30d + total). It then appends a snapshot block to the Notion source-of-truth page 📊 EO Analytics — Source of truth so AL has a history of how the analytics stack has grown over time.
  • Companion live mirror: tools.eliteoutsiders.com/backdata/analytics-map queries D1 in real time on every page load (no scheduled refresh needed for the live numbers — they're always fresh).
  • To extend the inventory: edit the TABLES list in eo-backup-worker/src/analytics-map-refresh.js AND the mirror in eliteoutsiders-tools/functions/api/admin/analytics-map.js. Keep both in sync.
cron: 0 5 1 1,7 * · code: eo-backup-worker/src/analytics-map-refresh.js · manual trigger: POST /admin/analytics-map-refresh-now?token=...
Local launchd Mac — 1

CRON9 — WIP safety net

CRON EVERY 30 MIN LOCAL LAUNCHD
Local launchd · LIVE since 2026-05-02

Every 30 min, rsync uncommitted changes from 3 working dirs to wip-autosave GitHub branch.

Pulse: loading…
Technical details
  • Every 30 min, this robot looks in your 3 working dirs (eliteoutsiders-site, eliteoutsiders-tools, eo-backup-worker) and copies anything not yet in Git into a parallel GitHub box called wip-autosave. Like a photocopier passing through your office every 30 min, photographing what's lying on the table without touching anything. If your Mac dies or you close by mistake, max 30 min of lost work.
  • Architecture: 3 mirror clones live in ~/Library/Application Support/eo-wip-autosave/. Each run rsyncs your working tree (excluding node_modules, .wrangler, dist, logs) to its mirror, commits + pushes to the wip-autosave branch (never main). Your main working tree is NEVER touched. No main history pollution.
  • Recover a copy: git fetch origin wip-autosave && git checkout wip-autosave in any clone of the repo, or directly on GitHub. Commits are timestamped: WIP autosave 2026-05-02T01:39:59Z.
cron: 0/30 * * * * (StartInterval 1800s launchd) · script: eo-backup-worker/scripts/wip-autosave.sh · plist: ~/Library/LaunchAgents/com.eo.wip-autosave.plist · log: ~/Projects/eo-backup-worker/logs/wip-autosave.log

CRON10 — Google Calendar scanner

CRON EVERY 30 MIN GCAL POLL
Every 30 min · LIVE since 2026-05-05

Every 30 min, scan Google Calendar for mentorship RDVs outside Cal.com, upsert bookings in D1.

Technical details
  • The robot reads your Google Calendar, detects "client mentorship" RDVs (tagged or with participant email matching an EO user), and inserts/updates the row in bookings with is_mentorship=1. So the "Your Bookings" cell turns green in the client's /me without you touching the DB by hand.
  • Why: today bookings rows are created only by the Cal.com webhook. If you book with a client outside Cal.com (manual email, direct GCal event), the client sees nothing in their /me. This robot fills that gap.
  • Prereq: Google Calendar OAuth already set up (table oauth_tokens with service='google_calendar', see migration 0010). Code to write: eo-backup-worker/src/gcal-scan.js running on 30-min cron, calls Google Calendar API, normalizes events, INSERT/UPSERT in D1 bookings with idempotency on external_id.
planned cron: */30 * * * * · status: SPEC · context: added 2026-05-01 at AL's request

CRON11 — Error events watchdog

CRON DAILY 06:00 UTC D1 → MAIL IF ANY
Daily 06:00 UTC · LIVE since 2026-05-07

Every day, query D1 error_events for any rows in last 24h. If > 0, email AL via Resend with the breakdown (level / source / status_code / latest 20 rows). If 0, silent — no spam.

Technical details
  • Why: server-side errors logged via functions/_lib/log-error.js (e.g. 500 on /api/admin/trauma-maps/[id]) are silent until AL clicks the broken thing. This watchdog flags them within 24h of occurrence.
  • Sources currently wired: /api/events, /api/auth/magic-link (Resend fail), /api/trauma-map/save (DB error), /api/admin/trauma-maps/[id]. Any new endpoint can opt-in by importing logError and calling it in the catch block.
  • Email shows: total count + breakdown by level + breakdown by source + latest 20 errors with timestamp/level/source/status/path.
  • "Silent on zero" by design — if you go a week without an email, all good. If you want to test, hit the manual trigger.
cron: 0 6 * * * · code: eo-backup-worker/src/error-events-watchdog.js · manual trigger: POST /admin/error-events-watchdog-now?token=$BACKUP_HEALTH_TOKEN
ROUTINES2 cards · 1 Anthropic Cloud + 1 local Claude Code

ROUTINE1 — Cron auditor

ROUTINE MONTHLY 01st @ 09h UTC
09:00 UTC (≈ 11h Paris) · LIVE since 2026-05-02

Monthly Anthropic Cloud routine: scan 3 repos for new CRON / ROUTINE / EVENT, open PR if missing.

Technical details
  • Scope 1 — crons / routines / events audit. Once a month, this robot reads code from the 3 repos (eliteoutsiders-site, eliteoutsiders-tools, eo-backup-worker) and looks for any new CRON (CF Worker schedule or local launchd plist), ROUTINE (claude.ai/code/routines or ~/.claude/scheduled-tasks/), or EVENT-triggered code (page load handlers, form submit POSTs, webhook handlers) AL built without listing on /crons. For each new entry, it adds a card in the right outer dropdown (CRONS / ROUTINES / EVENTS) on a separate git branch and opens a Pull Request for AL to validate. Like a teacher checking each new drawing is taped to the wall, leaving a sticky note for any missing.
  • Scope 1bis — TECHNICAL BLOCK update (added 2026-05-05). When a new CRON / ROUTINE / EVENT is detected (Scope 1), the robot ALSO updates the "Technical block — for Claude Code" table at the bottom of /crons : adds the new cron expression to the Crons row, the new admin endpoint to the Admin endpoints row, the new secret to the Secrets row if applicable, and any new R2/D1/KV binding to the Bindings row. Same branch + PR as Scope 1.
  • Scope 2 — infra audit (added 2026-05-03). Same pass also scans the 3 repos for any new external tool / service / binding (CF Workers, R2 buckets, KV namespaces, D1 tables, Resend, Kit, Notion, Cal.com, GCal, Stripe, Lemon Squeezy, Anthropic, OAuth providers, etc.) and any new wiring between them. Updates /infra-tree if a tool or interconnection is missing — same branch + PR mechanic as Scope 1.
  • If nothing new in either scope: silent commit of an HTML comment <!-- monthly audit YYYY-MM-DD: no new crons / no infra changes --> and exit. No PR for nothing.
  • ON EVERY RUN (even when nothing new), the robot must update BOTH data-last-update spans:
    • public/crons.html, right of the "Crons" H1
    • public/infra-tree.html, right of the "Infra Tree" H1
    With today's date in the format DD-MM-YYYY—UPDATE (e.g. 03-05-2026—UPDATE). Both date changes go into the PR (or the silent commit if no PR).
  • Anthropic Cloud routine (CCR) — ID trig_01UU96BG1d6FXknfdYHDEY21. Runs in an isolated sandbox with HTTPS clones of the 3 repos via the GitHub account alexandrecorne. Never modifies main directly — always branch + PR.
  • Manual run (off-cycle): open the routine page on claude.ai/code/routines and click Run now. The on-demand run executes the same prompt as the monthly cron and stays idempotent.
cron: 0 9 1 * * · management: claude.ai/code/routines · out-of-scope (mentioned in PR): ~/.claude/scheduled-tasks/*/SKILL.md (local only)
Technical reference (for Claude Code)
WhatValue
Workereo-backup-worker · Cloudflare account 88ff6259a88cb5fb8e8b9e13e082d7c4
URLhttps://eo-backup-worker.alexandre-corne-ac.workers.dev
Repo~/Projects/eo-backup-worker/
Crons0 2 1 * * (events-archive) · 0 3 1 * * (backup) · 0 4 1 * * (url-audit) · 0 4 * * 1 (page-flow-crawl) · 0 5 1 1,7 * (analytics-map-refresh) · 0 7 * * * (cal-monitor) · 0 8 * * 1 (weekly-snapshot) · 0 21 * * SUN (cal-randomize) — 8 deployed
SecretsRESEND_API_KEY · KIT_API_KEY · D1_API_TOKEN · D1_PREVIEW_API_TOKEN · BACKUP_HEALTH_TOKEN · CAL_API_KEY · NOTIFY_FROM · NOTIFY_TO · NOTION_API_KEY · NOTION_WEEKLY_PARENT_ID · NOTION_ARCHIVE_PARENT_ID · NOTIFY_TO_WEEKLY (optional)
BindingsR2 eo-backups (+ prefix events-archive/YYYY-MM/ from CRON5 LIVE) · D1 eliteoutsiders + eliteoutsiders-preview · KV BACKUP_STATE
Admin endpoints (POST + ?token=BACKUP_HEALTH_TOKEN)/admin/run-now · /admin/cal-monitor-now · /admin/cal-randomize-now · /admin/weekly-snapshot-now · /admin/events-archive-now · /admin/page-flow-crawl-now · /admin/url-audit-now · /admin/analytics-map-refresh-now
Healthcheck (GET + ?token=BACKUP_HEALTH_TOKEN)/backup/last-run
State KV (cal-monitor)namespace id 09082b5063a34c238de4b117efaf7d93 · key cal_slot_health:v2

ROUTINE2 — EO Quote weekly pipeline CONTENT

ROUTINE WEEKLY Fri @ 18h Paris
Friday 18:00 Europe/Paris · LIVE since 2026-05-23

Weekly local Claude Code routine: produce next Saturday quote (Canva visual + PostFast upload + Notion satellites X/Threads) unattended through Phase 8. AL gets a final report + manual Phase 9 PostFast approval.

Technical details
  • Scope. Every Friday evening, fetch the Quote Content Planner (Notion data source collection://9078c89a-5334-43fa-b501-7ac535d31c0d), pick the row with Posting Date = upcoming Saturday + Validation=Yes + Status="Not started", and run the full Quote pipeline single-row per Quote Grammar v1.5 and Grammar/Prompt doctrine V1.
  • Phases executed unattended. Phase 0 invariants → Phase 1 Notion query → Phase 2 live schedule fetch (content-system) → Phase 5 Canva production (single OR multi based on chars/height) → Phase 5.8 pixel check P3 → Phase 6 PostFast signed-URL upload + write keys → Phase 7 X + Threads satellites (NO LinkedIn per Grammar E.4 / AL Option A 2026-05-23) → Phase 8 Notion Status "In progress".
  • Phase 9 manual. Routine completion notifies AL (notification + open Finder folder). AL approves the 5 PostFast posts (IG/TT/FB/X/Threads) either manually in PostFast UI OR by saying "go approve Qxx" in any Claude conversation (Chrome MCP auto-approve).
  • Anti-patterns guarded. Em-dash interdiction in captions/filenames · signature_image element never edited (preserve or delete only) · single-page KEEPS signature + alexandre_corne · multi-page P1 cover strips both · multi PN final keeps both · all safe-zone/slot values fetched live from content-system.
  • Local Claude Code routine. Storage: ~/.claude/scheduled-tasks/eo-quote-weekly/SKILL.md. Runs only while a Claude Code app is open on AL's Mac; if closed, fires on next launch.
  • Manual trigger (off-cycle). Open Claude Code → Scheduled section sidebar → eo-quote-weekly → Run now. OR in any Claude conversation, say "schedule la quote de la semaine" and the same logic kicks off in-conversation.
cron: 0 18 * * 5 (Paris local time) · management: ~/.claude/scheduled-tasks/eo-quote-weekly/ · out-of-scope crons (CF Workers): /crons CRONS section above
Technical reference (for Claude Code)
WhatValue
TriggerLocal Claude Code scheduled-task eo-quote-weekly · cron 0 18 * * 5 Paris
Storage~/.claude/scheduled-tasks/eo-quote-weekly/SKILL.md
Notion DBs touchedQuote Content Planner 9078c89a-5334-43fa-b501-7ac535d31c0d (read+update) · X 81b3d7ba-d64e-40af-bd99-03627abfb9f0 (create) · Threads 90a1b900-bdc9-4691-aab9-45e6cacf0ed7 (create)
CanvaBrand template EAHIywfFr_k · folder FAHKGTx8peY · 5 slots per Grammar D.5
PostFastAPI api.postfa.st · header pf-api-key · endpoint POST /file/get-signed-upload-urls
Secrets read~/.config/eo/secrets.env: CF_ACCESS_CLIENT_ID, CF_ACCESS_CLIENT_SECRET, POSTFAST_API_KEY
Output path~/Library/Mobile Documents/com~apple~CloudDocs/iCloud Documents/DD/Elite Outsiders/content/quotes/2026/ kebab-case Qxx-title.png or -p1/-p2/...png
n8n downstreamEO Quotes Weekly workflow (reads Notion publication rows Status="In progress" + PostFast Image Key, creates SCHEDULED+PENDING_APPROVAL post in PostFast)
EVENTS3 cards · user/system action triggers

EVENT1 — Past weeks cleaner

EVENT ON EVENT /alsboard load
Client-side onload · LIVE since 2026-04-30

When /alsboard loads, auto-purge weeks whose end date has passed.

Technical details
  • When you open /alsboard, it automatically removes from the display the weeks whose end date has passed — the page cleans itself.
trigger: client-side onload · file: eliteoutsiders-tools/public/alsboard.html · code: parseWeekEndDate + purgePastUpcomingWeeks · persists via PUT /api/admin/state/eisenhower_matrix_al

EVENT2 — Post-quiz mailer

EVENT ON EVENT quiz submit
On Trauma Map submit · LIVE since 2026-04-26

When user submits Trauma Map quiz, send results email via Resend (fire-and-forget).

Technical details
  • When someone finishes the Trauma Map quiz and submits their email, this robot immediately sends a Resend email with full results (Past/Current scores + Inner/Outer Journey + 2 doors). Like a postman delivering the letter within the minute. If Resend crashes, the user still got their {ok:true} — the email goes out in background via waitUntil.
  • Code: functions/api/trauma-map/save.js calls buildTraumaMapEmail + sendEmail. Fire-and-forget. No retry yet.
trigger: POST /api/trauma-map/save · file: eliteoutsiders-site/functions/api/trauma-map/save.js

EVENT3 — Coaching library

EVENT ON EVENT toolkit upload
On /me/toolkit upload · LIVE since 2026-05-01

When client uploads to /me/toolkit, store blob in R2 + index in D1 toolkit_resources.

Technical details
  • When a coaching client uploads a document from /me/toolkit (PDF, DOCX, image, audio, video — up to 100 MB), this robot stores the file in the private R2 vault eo-user-toolkit and writes the index card in D1 toolkit_resources (with columns r2_key/mime/bytes added via migration 0018). Like a librarian shelving the book on the client's private shelf and noting the location. Nobody else (besides the client and the admin) can access the file — every read re-verifies identity before serving the blob.
  • Read side: GET /api/me/toolkit/file/<id> streams the R2 blob with ownership check (WHERE user_email = <jwt-email>) and Range header support so videos can scrub. Annotation side: toolkit_annotations table (migration 0019) — 3-color highlights + margin comments, same mechanics as the manifesto reader. Annotations active only on text types (TXT, MD, DOCX rendered to HTML via mammoth.js); PDF/image/audio/video read-only for V1.
  • Pricing link: the coaching cell on /workwithalex turns green and redirects to /me/toolkit as soon as a purchases row with product='coaching' and status='paid' exists for the logged-in email. Single source of truth: /api/me.
trigger: POST /api/me/toolkit/upload · files: eliteoutsiders-site/functions/api/me/toolkit/*.js + public/me/toolkit.html · binding: R2 USER_TOOLKIT → bucket eo-user-toolkit