Cron = Cloudflare Workers cron (eo-backup-worker, scheduled handler). Déterministe, server-side, zero token LLM. Pour jobs mécaniques (count rows, push Notion API).
Routine = Claude Code scheduled task (~/.claude/scheduled-tasks/<id>/SKILL.md). Spawn une session Claude. Coûte des tokens. Pour jobs de raisonnement (audit, brief, review).
Event = Code applicatif fire'd sur événement utilisateur ou système (page load, form submit, file upload, webhook). Pas scheduled, pas LLM.
CF Worker code: ~/Projects/eo-backup-worker/.
Monthly Kit + D1 backup to private R2. Quarterly dry-run on preview DB. Email recap.
0 3 1 * * · R2 target: eo-backups bucket · doc: Notion ch. 13 (Monitoring/alerting) + sub-page Setup backup mensuel Kit + D1 — 2026-04-30
Daily check that Cal.com slots are bookable across the 3 brands. Email if any slot is dead.
cal_slot_health:v2) — stores state per (event × date) to detect transitions.0 7 * * * · code: eo-backup-worker/src/cal-monitor.js · manual trigger: POST /admin/cal-monitor-now?token=...
Weekly chaos magic: re-randomize Cal.com slot times + brand rotation across the 4 weeks ahead.
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.0 21 * * SUN · code: eo-backup-worker/src/cal-randomize.js · manual trigger: POST /admin/cal-randomize-now?token=...
Monday morning report: 7-day counters + delta vs prev 7d + top pages, posted to Notion + email.
alexandre.corne.ac@gmail.com (override via NOTIFY_TO_WEEKLY).eo-backup-worker/src/weekly-snapshot.js + Notion helper src/notion.js. Email subject: EO weekly — YYYY-WXX.0 8 * * 1 · code: eo-backup-worker/src/weekly-snapshot.js · manual trigger: POST /admin/weekly-snapshot-now?token=...
Monthly archive of prior-month events to R2 + Notion, then DELETE old rows from D1.
⚠ 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.
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>.renderNotionBlocks().wrangler r2 object get eo-backups/events-archive/2026-04/events-archive-2026-05-01.json0 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)
Monday crawl of eliteoutsiders.com sitemap, upsert page_inventory + button_inventory in D1.
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.0 4 * * 1 · code: eo-backup-worker/src/page-flow-crawl.js · manual trigger: POST /admin/page-flow-crawl-now?token=...
Monthly crawl of every known URL (open + me + admin + tools), email anomaly report.
0 3 1 * * pour ne pas charger le réseau pendant l'upload R2), ce robot crawl toutes les URLs connues du site (open + me + admin + tools), compare au baseline EXPECTED dans le code, et envoie un récap mail Resend à AL.EXPECTED dans eo-backup-worker/src/url-audit.js.0 4 1 * * · code: eo-backup-worker/src/url-audit.js · manual trigger: POST /admin/url-audit-now?token=...
Semi-annual snapshot of D1 row counts per analytics table, append to Notion source-of-truth.
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.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.0 5 1 1,7 * · code: eo-backup-worker/src/analytics-map-refresh.js · manual trigger: POST /admin/analytics-map-refresh-now?token=...
Every 30 min, rsync uncommitted changes from 3 working dirs to wip-autosave GitHub branch.
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.~/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.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.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
Every 30 min, scan Google Calendar for mentorship RDVs outside Cal.com, upsert bookings in D1.
bookings with is_mentorship=1. So the "Your Bookings" cell turns green in the client's /me without you touching the DB by hand.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.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.*/30 * * * * · status: SPEC · context: added 2026-05-01 at AL's request
Monthly Anthropic Cloud routine: scan 3 repos for new CRON / ROUTINE / EVENT, open PR if missing.
~/.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.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.<!-- monthly audit YYYY-MM-DD: no new crons / no infra changes --> and exit. No PR for nothing.data-last-update spans:
public/crons.html, right of the "Crons" H1public/infra-tree.html, right of the "Infra Tree" H1DD-MM-YYYY—UPDATE (e.g. 03-05-2026—UPDATE). Both date changes go into the PR (or the silent commit if no PR).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.Run now. The on-demand run executes the same prompt as the monthly cron and stays idempotent.0 9 1 * * · management: claude.ai/code/routines · out-of-scope (mentioned in PR): ~/.claude/scheduled-tasks/*/SKILL.md (local only)
| What | Value |
|---|---|
| Worker | eo-backup-worker · Cloudflare account 88ff6259a88cb5fb8e8b9e13e082d7c4 |
| URL | https://eo-backup-worker.alexandre-corne-ac.workers.dev |
| Repo | ~/Projects/eo-backup-worker/ |
| Crons | 0 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 |
| Secrets | RESEND_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) |
| Bindings | R2 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 |
When /alsboard loads, auto-purge weeks whose end date has passed.
/alsboard, it automatically removes from the display the weeks whose end date has passed — the page cleans itself.eliteoutsiders-tools/public/alsboard.html · code: parseWeekEndDate + purgePastUpcomingWeeks · persists via PUT /api/admin/state/eisenhower_matrix_al
When user submits Trauma Map quiz, send results email via Resend (fire-and-forget).
{ok:true} — the email goes out in background via waitUntil.functions/api/trauma-map/save.js calls buildTraumaMapEmail + sendEmail. Fire-and-forget. No retry yet.POST /api/trauma-map/save · file: eliteoutsiders-site/functions/api/trauma-map/save.js
When client uploads to /me/toolkit, store blob in R2 + index in D1 toolkit_resources.
/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.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./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.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