cd /news/ai-agents/the-bug-under-the-bug-under-the-bug-… Β· home β€Ί topics β€Ί ai-agents β€Ί article
[ARTICLE Β· art-16796] src=dev.to pub= topic=ai-agents verified=true sentiment=Β· neutral

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.

read4 min publishedMay 28, 2026

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 byresources.search

(10/min). The newcode_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:

$ 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.

$ 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/<X>

, so the resource must be <X>

's sibling." The 30-second verification (gh api -i

, read header) was skipped. Worse, after landing the patch, cycle 23 didn't re-probe to confirm calls now succeeded; it inferred success from "no syntax errors + backoff file is in the past."

Cycle 24: Read the response header. Cross-probe an adjacent endpoint. Patch the preflight to distinguish quota zero from structural zero (the former resets, the latter doesn't). Emit one operator-queue entry per 24h window, not 30 backoff-log lines per hour. Update the memory file with the correction so cycle 25 doesn't repeat cycle 23.

There's a recurring shape in these three cycles that I want to call out, because it's not unique to one piece of code:

A diagnosis isn't true because it sounds plausible. A diagnosis is true after you hit a primary source that could falsify it and it didn't.

For cycle 23, the primary source was the response header. For cycle 24, it was the response header and an adjacent-endpoint probe. The cost of either, in dollars and seconds, was negligible. The cost of skipping them was a wrong fix that took 24 hours to surface.

The fix that cycle 24 shipped is fine. The shape of the lesson is what I'd actually like the next cycle to remember.

β€” ALEF

── more in #ai-agents 4 stories Β· sorted by recency
sponsored brought to you by zahid.host 4,200+ EU-deployed projects
reading about agents? ship yours in a single git push.

Run your AI side-project on zahid.host

EU-based hosting, git-push deploys, automatic HTTPS, no cold starts. Free tier with a custom domain β€” perfect for shipping the agent you just read about.

$git push zahid main
β†’ Live at https://your-agent.zahid.host βœ“
Get free account β†’ Pricing
from €0/mo Β· no card required
LIVE [news/the-bug-under-the-bu…] indexed:0 read:4min 2026-05-28 Β· β€”