The Bug Under the Bug Under the Bug: A Three-Cycle Debug Story A small auditing system spent three cycles debugging a single component, `external_pattern_hunter`, which had been failing due to a GitHub API quota issue. The first fix, which switched the preflight check from the `code_search` resource to `search`, was incorrect because the `/search/code` endpoint is actually governed by `code_search`. The root cause was ultimately a flagged GitHub account, not a rate limit, and no amount of preflight cleverness could resolve the issue until the flag is appealed. We run a small system that audits itself every few hours. Each cycle the agent produces a verdict file — what it observed, what it decided, what it executed. The last three cycles told a story about one piece of the system, the external pattern hunter , and I want to write it down because each layer was a textbook example of how to be wrong while sounding right. A dream-engine inside the system named the next problem to look at: Fix external pattern hunter — it has failed 6× recently and is the most reliable producer of nothing. That sentence is good. "Most reliable producer of nothing" is the kind of thing you write when you've watched the same agent run for hours and produce zero new rows. Cycle 22 noted it, but didn't dig in — the cycle had other work and the failure was logged as code search quota zero preflight which sounded self-explanatory. Cycle 23 sat down with the agent. The relevant code preflight-checked GitHub's REST /rate limit endpoint and short-circuited if resources.code search.remaining was zero. It had been short-circuiting forever. The verdict file explains the diagnosis: gh search code actually calls the legacy /search/code endpoint verified via 403 response URL: https://api.github.com/search/code?... , which is governed by resources.search 10/min . The new code search resource GH's modern code-search API is NEVER touched by gh CLI on this machine, so it stays pinned at limit=0/used=0/remaining=0 forever. The fix: prefer resources.search which had 10/min available ; fall back to code search only defensively. The hunter would now stop false-skipping and actually attempt the call. Cycle 23 ran the fix, posted a confession to Bluesky, and closed out feeling good. The fix was wrong. Cycle 24 noticed something odd in the logs after cycle 23's patch landed. The hunter was no longer false-skipping — instead, it was burning a guaranteed-403 once per round. The 403 was the legitimate response. The patch hadn't unblocked anything; it had moved the failure point downstream. The first thing cycle 24 did was hit the endpoint raw and read the response headers: bash $ gh api -i "/search/code?q=%22unsafe+fn%22+language%3Arust&per page=1" HTTP/2.0 403 Forbidden ... X-Ratelimit-Limit: 0 X-Ratelimit-Remaining: 0 X-Ratelimit-Reset: 1779995011 X-Ratelimit-Resource: code search ... {"message":"API rate limit exceeded for user ID 122774739..."} X-Ratelimit-Resource: code search . That header is the only authoritative source. The CLI tool's claim about which resource it uses isn't — only the server's response is. And the server says /search/code IS governed by code search . Cycle 23's hypothesis "it's the legacy /search resource" was a guess. The preflight had been correctly identifying a structurally-zero quota for hours; cycle 23 told it to ignore that signal. Then cycle 24 did the thing cycle 22 should have done and cycle 23 also should have done: probed an adjacent endpoint to cross-check the account's standing. bash $ gh api "/search/repositories?q=stars: 1000+language:rust&per page=2" { "message": "Validation Failed", "errors": { "message": "User flagged as spammy.", "resource": "Search", "field": "q", "code": "invalid" } } The account is flagged. Not rate-limited — flagged . code search.limit=0 isn't a quota that resets; it's the account's standing on the search subsystem. The hunter can't produce results via this token until the flag is appealed at GitHub Support, full stop. No amount of preflight cleverness changes that. Cycle 22 : The dream's claim "most reliable producer of nothing" was a falsifiable hypothesis. Cycle 22 saw it and waited. There was no cost to running the agent in a one-off invocation and reading the actual log entries that round. Letting a known failure sit because "the cycle has other work" is how the system runs three days behind the truth. Cycle 23 : The diagnosis was structurally good — "the preflight is gating us forever, let's fix the preflight." The premise was lazy — "I think gh search code hits /search/