
bluefin shipsHow a food delivery rider runs a CEO, PM, Copywriter, Designer, Dev, and QA agent from a phone — the architecture, a minimal tick loop, and what actually ships.
Today I rode my 125cc scooter through Taipei rain for twelve hours and made NT$2,100 (about US$65) delivering lunch and dinner shifts.
In the exact same window, my phone was running six AI agents. By the time I peeled off the wet jacket, a $19 product was live on Gumroad. I didn't touch a keyboard.
This is not a manifesto. This is a Thursday.
Six roles, each a separate Claude CLI process with its own system prompt and memory slice:
A seventh helper, the CC Daemon, polls IMAP for 2FA codes, pastes them into forms, and pushes completion notices to Discord so I hear a ping on my helmet speaker.
I got home soaked. The product had already made its first sale.
Phone (Discord)
|
v
CC Daemon (hourly tick + event triggers)
|
v
Blackboard (tasks.jsonl — append-only)
|
v fan-out
Subagents (Claude CLI, parallel)
- CEO, PM, Copy, Design, Dev, QA
|
v
Shared memory (MemPalace — MCP knowledge graph)
|
v
Output queue ---> Gumroad / Discord push / inbox/
The blackboard is the whole trick. Every agent reads the same tasks.jsonl, claims a row, writes its output back as a new row with a parent_id. No orchestrator god-object. If one agent crashes, the row stays unclaimed and the next tick retries it.
MemPalace is the shared brain. When the Copywriter references "last week's tone notes," it's pulling from the same knowledge graph the CEO used to write today's brief. Context survives /compact, reboots, and me forgetting what day it is.
Strip away the Discord bridge, the IMAP poller, the MCP servers, and this is the beating heart:
# Minimal daemon tick — run every 5 minutes via launchd/cron
import subprocess
from pathlib import Path
def call_agent(role_prompt: str, user_prompt: str) -> str | None:
r = subprocess.run(
["claude", "-p", "--max-turns", "1",
f"[ROLE]\n{role_prompt}\n[TASK]\n{user_prompt}"],
capture_output=True, text=True, timeout=120,
)
return r.stdout if r.returncode == 0 else None
draft = call_agent(
"You are a copywriter. 200 words max. Hook in first line.",
"Write a Threads post about shipping while delivering food.",
)
review = call_agent(
"You are QA. Return a numbered list of factual or tonal defects.",
f"Review this draft:\n{draft}",
)
Path("inbox/draft.md").write_text(draft or "")
Path("inbox/review.md").write_text(review or "")
# Next tick: a Dev agent reads review.md and patches draft.md.
Fifteen lines. Two agents. A handoff via files. Scale the same pattern to six roles and you have my setup — minus the keys I'm not going to paste.
My fixed monthly bills are NT$47,000. I cannot afford one human employee. I cannot afford to stop delivering to "focus on the startup." So I built a team I pay in electricity — less than one coffee a day on an M1 Max — that doesn't clock out when I go back on the road.
The reframe that made it click: I'm not trying to automate myself out of delivery. I'm trying to make sure the six hours I'm on a scooter aren't six hours my business is asleep. Food delivery pays cash today. Agents compound tomorrow. I need both.
I wrote down every piece: the role prompts, the blackboard schema, the Discord-to-daemon bridge, the handoff patterns between CEO and PM, how QA returns structured defects Dev can actually apply, and the failure modes I hit in the first six weeks (there were many).
One honest line: this is a pattern cookbook — you'll wire your own Discord, IMAP, and API keys to make it run. I'm handing you the architecture and the prompts that survived contact with reality, not a one-click installer.
$19 on Gumroad. I'm asleep, on a scooter, or soaked — it's still selling.
https://vampireheart3.gumroad.com/l/agent-cookbook
This is Thursday.