<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: szp2005</title>
    <description>The latest articles on DEV Community by szp2005 (@szp2005).</description>
    <link>https://dev.to/szp2005</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.us-east-2.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3954570%2Fc95d8429-edc0-4742-aef7-a8a16b1e0bdb.png</url>
      <title>DEV Community: szp2005</title>
      <link>https://dev.to/szp2005</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/szp2005"/>
    <language>en</language>
    <item>
      <title>Reconciling 8 IP-reputation feeds into one verdict: averaging is the wrong default</title>
      <dc:creator>szp2005</dc:creator>
      <pubDate>Fri, 19 Jun 2026 08:42:18 +0000</pubDate>
      <link>https://dev.to/szp2005/reconciling-8-ip-reputation-feeds-into-one-verdict-averaging-is-the-wrong-default-1258</link>
      <guid>https://dev.to/szp2005/reconciling-8-ip-reputation-feeds-into-one-verdict-averaging-is-the-wrong-default-1258</guid>
      <description>&lt;p&gt;Wire more than one IP-reputation source into a risk check and sooner or later they disagree. One feed says the IP is a residential ISP address. Another calls it a datacenter VPN. A blocklist says it relayed spam last week. A geolocation provider says it's clean and unremarkable.&lt;/p&gt;

&lt;p&gt;The naive move is to normalize everything to 0–100 and average it. I did that first. It produces a number that's wrong in specific, reproducible ways, and on top of that a number nobody can act on. The moment a verdict matters, someone asks "&lt;em&gt;why&lt;/em&gt; is this 0.62?" and the average has no answer.&lt;/p&gt;

&lt;p&gt;The version I landed on after the averaging one kept embarrassing me reads as a decision log. Every rule below is there because some real IP broke the version before it.&lt;/p&gt;

&lt;h2&gt;
  
  
  Why averaging fails: three concrete failure modes
&lt;/h2&gt;

&lt;p&gt;&lt;strong&gt;1. Low-precision sources dominate the consensus.&lt;/strong&gt; Some feeds label entire datacenter /16 blocks as "proxy" or "VPN" wholesale. They're cheap and high-recall, so they're noisy. Average them in and a plain Hetzner or Linode box that two of these feeds tagged as "proxy" gets dragged up into mid-risk territory, even when every higher-precision source says it's just hosting. You've shipped a scorer that cries wolf on half of AWS.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;2. A single low-confidence report flips a binary feed.&lt;/strong&gt; Abuse-report databases are community-fed. If your rule is &lt;code&gt;flagged = (totalReports &amp;gt; 0)&lt;/code&gt;, one retaliatory or mistaken report marks an address as a known abuser. I watched &lt;code&gt;8.8.8.8&lt;/code&gt;, Google Public DNS, come back as "abuser" because somebody somewhere reported it once. Averaging doesn't save you. It buries the bad signal under the good ones for &lt;em&gt;most&lt;/em&gt; IPs and then surfaces it on the unlucky ones.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;3. Averaging dilutes the one source that matters most.&lt;/strong&gt; A live spam-relay listing, or membership in a Tor exit-node list, sits close to ground truth. Seven geolocation feeds saying "nothing unusual" should not be allowed to wash that out. Risk signals aren't symmetric, and an average pretends they are.&lt;/p&gt;

&lt;h2&gt;
  
  
  The model: visible per-source verdicts, asymmetric floors
&lt;/h2&gt;

&lt;p&gt;Two ideas did most of the work.&lt;/p&gt;

&lt;p&gt;The first: don't collapse to one opaque number. Keep every source's verdict and show it as its own line item. Which feed, what it claimed, what signal category it falls under (datacenter, residential proxy, Tor exit, active abuser, spam-list hit). Then whoever consumes the score decides whether a given flag matters for &lt;em&gt;their&lt;/em&gt; case. A Tor-exit listing is disqualifying for a signup flow and irrelevant for a geo-IP cache.&lt;/p&gt;

&lt;p&gt;The second: keep a weighted baseline, but let signal &lt;em&gt;type&lt;/em&gt; set a hard floor. The aggregate starts as a precision-weighted average, and then certain confirmed signals impose a minimum the average can't pull below.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;Tor exit node (confirmed)      → floor 90
Dedicated proxy/VPN (consensus)→ floor 65
Confirmed abuser               → floor 55
Datacenter / hosting           → floor 35
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;A floor says: if this signal is present, the score can't drop below X no matter how many geo feeds call the address clean. Swapping type-driven floors in for the pure average is the one change that got the output to line up with what an analyst would actually conclude.&lt;/p&gt;

&lt;h2&gt;
  
  
  The rules that keep the floors honest
&lt;/h2&gt;

&lt;p&gt;A floor is only as trustworthy as the boolean that trips it. Each of these earned its place by killing a specific false positive.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Proxy/VPN needs consensus from &lt;em&gt;dedicated&lt;/em&gt; sources.&lt;/strong&gt; The low-precision general feed never gets to establish a proxy verdict on its own. On datacenter ranges I require ≥2 dedicated (purpose-built proxy/VPN) sources to agree. On residential ranges ≥1 is enough, since a residential proxy is rarer and so means more when a specialized feed flags it. Hetzner and Linode fall back to "hosting 35" instead of a phantom "proxy 65," and a real consumer-ISP proxy still trips.&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Tighten the noisy binary feed.&lt;/strong&gt; An abuse listing now requires &lt;code&gt;score ≥ 25 AND reports ≥ 3&lt;/code&gt; (or ≥2 distinct reporters), and the address can't be on the provider's own allowlist. &lt;code&gt;8.8.8.8&lt;/code&gt; stops being an abuser.&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Whitelist known infrastructure ASNs.&lt;/strong&gt; Google, Cloudflare, and the like suppress the abuser and hosting floors. A CDN edge node isn't a threat, and you don't want your scorer picking fights with the backbone of the internet.&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Treat ASN reputation as standalone evidence.&lt;/strong&gt; A small set of autonomous systems are VPN/proxy-only businesses: M247, Mullvad, Proton, a handful of others. For these, membership alone settles it, with no cross-source consensus needed, because the network operator's identity &lt;em&gt;is&lt;/em&gt; the signal. This recovers the case where one feed alone recognizes a niche VPN that the consensus rule above would otherwise suppress.&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Add a hard, independent signal: DNSBL over DoH.&lt;/strong&gt; I query a handful of DNS blocklists, reversing the octets against each zone and going over DNS-over-HTTPS so it runs from an edge runtime. A hit there is close to ground truth and leans on nobody's opaque vendor score.&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Short-circuit reserved and CGNAT ranges before scoring.&lt;/strong&gt; CGNAT (100.64.0.0/10), TEST-NET, benchmark, multicast, and the IPv6 equivalents get an explicit "reserved, here's the category" response rather than going through the pipeline to be mislabeled. It also keeps thousands of carrier-NAT users behind one exit from being scored as a shared proxy.&lt;/p&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Make the verdict auditable, not just displayable
&lt;/h2&gt;

&lt;p&gt;If I had to press one point on anyone building this, it's this: emit the &lt;em&gt;breakdown&lt;/em&gt; as structured data, not just the final number. Every lookup returns each source's contribution, the weighted average before floors, which floors fired and why, and the final value. You get debuggability out of it. When a verdict looks wrong, the breakdown tells you at a glance whether it was a bad weight, a floor that shouldn't have fired, or thin data. You also let the user overrule you: the person reading the score can tell whether it rests on one thin signal or a five-way consensus, and judge for their own case. A black-box number forces all-or-nothing, trust it blind or throw it out.&lt;/p&gt;

&lt;h2&gt;
  
  
  What I still haven't solved well
&lt;/h2&gt;

&lt;p&gt;A few open problems, since anyone who's done this for real will have opinions.&lt;/p&gt;

&lt;p&gt;CGNAT and mobile carriers are the worst of them. Shared-exit NAT and a residential proxy pool throw off the same surface signal: many users, one IP. Short-circuiting the reserved CGNAT block helps, but carriers also use public ranges that look identical to a proxy from the outside. I flag uncertainty rather than guess, and I still don't have a clean discriminator.&lt;/p&gt;

&lt;p&gt;Then there's absence of evidence versus evidence of absence. For smaller regional ISPs the databases run thin. "No source flagged it" reads as "clean" when it often just means "nobody has data." Right now I surface coverage, the count of how many sources had any opinion at all, next to the verdict. I'm not convinced that's enough.&lt;/p&gt;

&lt;p&gt;Last, the residential-versus-datacenter split. When two classifiers disagree on the same IP I show both labels and leave it unresolved. Whether a confidence-weighted merge beats preserving the raw disagreement, I genuinely don't know.&lt;/p&gt;

&lt;p&gt;If you've run reputation scoring at scale, I'd value your take on the /24 neighbor signal (contamination ratio weighted by flag recency?) and on the residential/datacenter conflict above.&lt;/p&gt;




&lt;p&gt;&lt;em&gt;The scorer described here runs behind &lt;a href="https://ipok.io" rel="noopener noreferrer"&gt;ipok.io&lt;/a&gt;, a free, no-login IP reputation checker that shows the per-source breakdown instead of a single number. The CLI is MIT on &lt;a href="https://github.com/szp2005/ipok-cli" rel="noopener noreferrer"&gt;GitHub&lt;/a&gt;. Happy to go deeper on any of the data-source quirks in the comments.&lt;/em&gt;&lt;/p&gt;

</description>
      <category>networking</category>
      <category>security</category>
      <category>devops</category>
      <category>showdev</category>
    </item>
    <item>
      <title>Making "files never leave your browser" verifiable with DevTools and CSP</title>
      <dc:creator>szp2005</dc:creator>
      <pubDate>Mon, 15 Jun 2026 03:23:04 +0000</pubDate>
      <link>https://dev.to/szp2005/making-files-never-leave-your-browser-verifiable-with-devtools-and-csp-4n99</link>
      <guid>https://dev.to/szp2005/making-files-never-leave-your-browser-verifiable-with-devtools-and-csp-4n99</guid>
      <description>&lt;p&gt;"Files never leave your browser" is becoming standard copy for PDF tools, image editors, and document converters. But a trust claim and a verifiable fact are different things. Here's how to turn "zero upload" into something any user can audit in about two minutes, and how to enforce it at the browser level so it isn't just a promise.&lt;/p&gt;

&lt;h2&gt;
  
  
  Step 1: Read the Network panel
&lt;/h2&gt;

&lt;p&gt;Open DevTools → Network, enable "Disable cache", reload. While processing a file, filter by "Fetch/XHR" and "Doc". A genuinely client-side tool should show only HTML/CSS/JS/WASM asset loads — no POST requests, no GETs carrying file content in query parameters.&lt;/p&gt;

&lt;p&gt;The non-obvious trap: third-party analytics, Google Fonts, and CDNs all show up as outbound requests. If you claim zero uploads, those count too. The honest move is to self-host fonts and scripts and drop analytics entirely, so the request list is genuinely short enough to eyeball.&lt;/p&gt;

&lt;p&gt;The Network panel is the human-readable check. The next part is what actually makes it hold.&lt;/p&gt;

&lt;h2&gt;
  
  
  Step 2: Enforce egress with CSP &lt;code&gt;connect-src&lt;/code&gt;
&lt;/h2&gt;

&lt;p&gt;This is the piece people get backwards, so it's worth stating precisely.&lt;/p&gt;

&lt;p&gt;CSP's &lt;code&gt;connect-src&lt;/code&gt; is an egress allowlist the browser enforces &lt;em&gt;before the request is sent&lt;/em&gt;. A &lt;code&gt;fetch&lt;/code&gt;/XHR to an origin that isn't on the list is blocked by the browser and never leaves the machine. You'll see it fail in the console as a CSP violation, with no entry in the Network tab going out to that origin.&lt;/p&gt;

&lt;p&gt;This includes &lt;code&gt;no-cors&lt;/code&gt; requests. &lt;code&gt;no-cors&lt;/code&gt; is sometimes assumed to be an escape hatch, but it isn't one for this purpose. All &lt;code&gt;no-cors&lt;/code&gt; does is let you &lt;em&gt;issue&lt;/em&gt; a cross-origin request while making the response opaque (you can't read the body). It does not bypass &lt;code&gt;connect-src&lt;/code&gt;: if the target origin isn't in your &lt;code&gt;connect-src&lt;/code&gt; allowlist, the &lt;code&gt;no-cors&lt;/code&gt; request is blocked exactly the same way — it never goes out. So you can't smuggle a file out to a third party with &lt;code&gt;no-cors&lt;/code&gt; under a tight CSP.&lt;/p&gt;

&lt;p&gt;That's what makes CSP the actual proof, not just documentation. Tighten &lt;code&gt;connect-src&lt;/code&gt; to &lt;code&gt;'self'&lt;/code&gt; (or an explicit list of the few endpoints you genuinely need), and any code path that tries to ship data to another origin — yours, a third party's, an injected script's — is stopped by the browser. A realistic policy:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight conf"&gt;&lt;code&gt;&lt;span class="n"&gt;connect&lt;/span&gt;-&lt;span class="n"&gt;src&lt;/span&gt; &lt;span class="s1"&gt;'self'&lt;/span&gt;;
&lt;span class="n"&gt;font&lt;/span&gt;-&lt;span class="n"&gt;src&lt;/span&gt; &lt;span class="s1"&gt;'self'&lt;/span&gt;;
&lt;span class="n"&gt;script&lt;/span&gt;-&lt;span class="n"&gt;src&lt;/span&gt; &lt;span class="s1"&gt;'self'&lt;/span&gt; &lt;span class="s1"&gt;'wasm-unsafe-eval'&lt;/span&gt;;
&lt;span class="n"&gt;img&lt;/span&gt;-&lt;span class="n"&gt;src&lt;/span&gt; &lt;span class="s1"&gt;'self'&lt;/span&gt; &lt;span class="n"&gt;data&lt;/span&gt;:;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Note &lt;code&gt;'wasm-unsafe-eval'&lt;/code&gt; rather than the broader &lt;code&gt;'unsafe-eval'&lt;/code&gt; — modern browsers support the narrower directive for instantiating WASM, so there's no reason to grant full &lt;code&gt;eval&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;With that in place, the Network panel check from Step 1 stops being "trust me, the list is short" and becomes "the browser will refuse to send anything I didn't whitelist, and here's the empty list to confirm it."&lt;/p&gt;

&lt;h2&gt;
  
  
  Step 3 (optional): Content-Length as a sanity check
&lt;/h2&gt;

&lt;p&gt;If you want a quick gut-check rather than reasoning about the allowlist, clear the Network panel before triggering processing, then sum the Size column afterward. If the total is nowhere near the original file size, no file content went out. This also catches chunked-transfer or WebSocket approaches that a naive "look for a POST" scan might miss. It's a weaker check than the CSP guarantee, but it's fast and visual.&lt;/p&gt;

&lt;h2&gt;
  
  
  A Service Worker doesn't replace CSP
&lt;/h2&gt;

&lt;p&gt;A Service Worker can intercept fetches and is useful for offline caching, but it's not the egress boundary — it's first-party code that can be bypassed or simply not cover a code path, and it does nothing about requests that don't route through it. CSP &lt;code&gt;connect-src&lt;/code&gt; is enforced by the browser regardless of your application code. Use a Service Worker for caching if you want; rely on CSP for the "can't exfiltrate" guarantee.&lt;/p&gt;

&lt;h2&gt;
  
  
  What this looks like in practice
&lt;/h2&gt;

&lt;p&gt;I built a PDF tool this way (moguanpdf.com, my own project — mentioning it only because it's a live example you can poke at). The classic tools (compress, merge, split, OCR, watermark, encrypt/decrypt, etc.) run entirely in-browser via WASM + pdf.js. Open DevTools → Network while processing a file and you'll see only &lt;code&gt;.wasm&lt;/code&gt;, &lt;code&gt;.js&lt;/code&gt;, and &lt;code&gt;.css&lt;/code&gt; loads, no POST, no analytics. The one server-side exception is the AI features (summarize/translate/Q&amp;amp;A), which send extracted text rather than the file, and the UI says so. I'd encourage auditing it the same way you'd audit anyone else's — that's the whole point.&lt;/p&gt;

&lt;h2&gt;
  
  
  The broader point
&lt;/h2&gt;

&lt;p&gt;If your users handle contracts, medical records, or financial documents, "open DevTools and follow these steps, and here's the CSP that guarantees it" is a stronger statement than any privacy policy. The Network panel shows users an empty list; &lt;code&gt;connect-src 'self'&lt;/code&gt; is the reason the list stays empty. A tool that can't survive that audit probably shouldn't be making the claim.&lt;/p&gt;

</description>
      <category>privacy</category>
      <category>security</category>
      <category>tutorial</category>
      <category>webdev</category>
    </item>
    <item>
      <title>643 articles, 11 Google clicks: my 4-month AI SEO experiment</title>
      <dc:creator>szp2005</dc:creator>
      <pubDate>Wed, 27 May 2026 15:29:12 +0000</pubDate>
      <link>https://dev.to/szp2005/643-articles-11-google-clicks-my-4-month-ai-seo-experiment-4ami</link>
      <guid>https://dev.to/szp2005/643-articles-11-google-clicks-my-4-month-ai-seo-experiment-4ami</guid>
      <description>&lt;h1&gt;
  
  
  I spent 4 months and 643 articles to test AI-generated SEO. Here's what 11 clicks taught me.
&lt;/h1&gt;

&lt;p&gt;&lt;strong&gt;TL;DR&lt;/strong&gt;: I built a 4-site, 643-article AI-content portfolio over 4 months. Google gave me 11 clicks. AdSense rejected one site for "low-value content". Here's the experiment, the numbers, and what I think I got wrong.&lt;/p&gt;




&lt;h2&gt;
  
  
  The setup
&lt;/h2&gt;

&lt;p&gt;January 2026, I left a salaried job to build a "one-person company." The thesis was simple and very 2025: a single operator + Claude + a content pipeline can produce SEO-friendly articles at a scale that used to require an agency. Pick 4 niches, pick 4 domains, point the pipeline at each, wait for Google traffic, monetize with AdSense + affiliate.&lt;/p&gt;

&lt;p&gt;This is the kind of plan that sounds reasonable in a YouTube video and obvious in a Twitter thread.&lt;/p&gt;

&lt;p&gt;The 4 sites:&lt;/p&gt;

&lt;div class="table-wrapper-paragraph"&gt;&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Site&lt;/th&gt;
&lt;th&gt;Niche&lt;/th&gt;
&lt;th&gt;Articles&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;ai.toolrouteai.com&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;AI tool reviews and comparisons&lt;/td&gt;
&lt;td&gt;172&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;gear.toolrouteai.com&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Home office equipment&lt;/td&gt;
&lt;td&gt;142&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;
&lt;code&gt;notes-automate.com&lt;/code&gt; (EN + zh-cn)&lt;/td&gt;
&lt;td&gt;Obsidian / PKM workflows&lt;/td&gt;
&lt;td&gt;174 + 174&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;pkm-insights.com&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Personal knowledge management theory&lt;/td&gt;
&lt;td&gt;191&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;&lt;/div&gt;

&lt;p&gt;Total: &lt;strong&gt;679 English articles + 174 Chinese translations = 853 published URLs&lt;/strong&gt;. Across 4 months. That's ~7 articles per day, every day, including weekends. No human writer can sustain that rate. I didn't try — Claude wrote every word.&lt;/p&gt;

&lt;p&gt;The stack:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Astro 5 static sites, deployed on Cloudflare Pages (free tier)&lt;/li&gt;
&lt;li&gt;Content pipeline: n8n workflows on a Docker host, calling Claude for drafts, Gemini for research, MinIO for asset storage&lt;/li&gt;
&lt;li&gt;A custom Telegram dispatcher (19 slash commands) for monitoring, manual triggers, daily briefs&lt;/li&gt;
&lt;li&gt;A cross-platform "viral content" scraper hitting Reddit, Hacker News, Substack RSS, YouTube transcripts — designed to source title angles from what was already winning elsewhere&lt;/li&gt;
&lt;li&gt;6 Cloudflare Workers running cron jobs for various pipelines&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Infra cost: &lt;strong&gt;$0/month&lt;/strong&gt;. The whole thing runs on free tiers, hardware I already owned, and the fact that I write code faster than I write prose.&lt;/p&gt;

&lt;h2&gt;
  
  
  The result
&lt;/h2&gt;

&lt;p&gt;After 4 months and 643 indexed articles:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;&lt;strong&gt;Google Search Console clicks: 11&lt;/strong&gt;&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Email subscribers: 0&lt;/strong&gt; (the newsletter form sat at the footer of every page)&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;AdSense status: notes-automate.com rejected for "low-value content"; ai.toolrouteai.com still under review; the other two never submitted&lt;/strong&gt;&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Affiliate revenue: $0&lt;/strong&gt;&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Direct traffic / brand searches: 0&lt;/strong&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;11 clicks. Across 4 months. Across 643 articles. Across 4 domains.&lt;/p&gt;

&lt;p&gt;That's a click-through rate that rounds to zero. If I'd just posted one comment per day on Hacker News for 4 months, I would have gotten more traffic.&lt;/p&gt;

&lt;h2&gt;
  
  
  What actually happened
&lt;/h2&gt;

&lt;p&gt;I want to be precise about this part, because the obvious narrative ("AI content slop doesn't work") is too simple and partly wrong.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;The content was not visibly garbage.&lt;/strong&gt; I read through several articles last week. Most of them are coherent, technically accurate, structured for SEO (H2/H3, intro paragraph, FAQ, related links), and would pass a casual human reader's smell test. They are not Mad-Libs blogspam. They are something more interesting: technically competent, contextually empty.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Google indexed almost everything.&lt;/strong&gt; Out of 853 URLs, ~580 are indexed. The "已抓取-尚未编入索引" (crawled, not indexed) bucket is real but not dominant. So this is not a "Google never saw my site" problem.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;The 11 clicks were spread across long-tail queries with single-digit monthly volume.&lt;/strong&gt; Things like "obsidian dataview snippets for book trackers." Niche enough that there was little competition. Common enough that Google ranked me on page 1 for the query. But also: small enough that ranking #1 means 1-3 clicks per month, total.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;AdSense's "low-value content" rejection is the most informative signal.&lt;/strong&gt; It didn't say "too thin" or "duplicate." It said "low value." That's a different judgment — the reviewer (or model) decided my articles, despite being long and structured, weren't adding anything a reader couldn't get faster from the next 10 search results.&lt;/p&gt;

&lt;p&gt;I'd been telling myself for 4 months that I was building a scalable content business. What I had actually built was a scalable irrelevance machine.&lt;/p&gt;

&lt;h2&gt;
  
  
  Where I think the mistake was
&lt;/h2&gt;

&lt;p&gt;A few candidates, in order of how much I now believe them:&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;1. I optimized the production loop, not the distribution loop.&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;I spent ~80% of my time on the pipeline: making the n8n workflow more reliable, building the dispatcher, adding the Telegram alerts, the cron jobs, the viral scraper, the auto-translation system, the markdown linter, the OG image generator, the AdSense injector, the sitemap builder. All of this is "make production faster."&lt;/p&gt;

&lt;p&gt;I spent ~5% of my time on distribution: submitting sitemaps, building 4 random backlinks. There is no version of this experiment that works without distribution.&lt;/p&gt;

&lt;p&gt;If I had spent the 4 months differently — say, &lt;strong&gt;80% on distribution and 20% on writing 50 articles by hand&lt;/strong&gt; — I'm now fairly sure the outcome would have been better. Not great, but better than 11 clicks.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;2. SEO doesn't work like it did in 2021.&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;The old playbook: write 500 articles, build a few backlinks, wait 6 months, get Google traffic. People still teach this. It worked. It does not work now, at this scale, with this content type, for one operator.&lt;/p&gt;

&lt;p&gt;Google's quality threshold has moved. Helpful Content Update + the December 2024 spam updates + whatever they're doing internally with their own AI classifiers — the bar is higher than the bar AI-generated articles can clear, even when the articles are coherent.&lt;/p&gt;

&lt;p&gt;I should have known this from reading Google's documentation. I read the documentation. I rationalized it as "applies to other people."&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;3. "Niche selection" became "niche of niches" without me noticing.&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;&lt;code&gt;gear.toolrouteai.com&lt;/code&gt; is about home office gear. Fine niche. But the articles I wrote weren't "best monitor 2026" — they were "best portable monitor for dual-screen laptop setup." That second query has maybe 30 searches per month. Globally. After Google takes its cut for shopping results and YouTube widgets, you're competing for ~5 organic clicks.&lt;/p&gt;

&lt;p&gt;I had been told that "long-tail = less competition = good." This is technically true and operationally useless. Less competition for a query with 5 clicks per month is still 5 clicks per month.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;4. I confused "publishable" with "valuable."&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;The pipeline produced articles that were publishable. They cleared the bar of "not embarrassing to have on the internet." That bar is not the bar that gets traffic. The bar that gets traffic is "this is the best result a reader will find for this query today." I was nowhere near that bar.&lt;/p&gt;

&lt;h2&gt;
  
  
  What I think the experiment actually proved
&lt;/h2&gt;

&lt;p&gt;Not "AI content doesn't work." That's the lazy take and it's not what my data shows.&lt;/p&gt;

&lt;p&gt;What it proved, for me:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;AI content at scale, deployed by a solo operator with no distribution, does not produce a business in 4 months.&lt;/strong&gt; (Maybe in 12 months. Probably not.)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;The infrastructure is the easy part.&lt;/strong&gt; I built a sophisticated pipeline. So can you. So can 10,000 other people. None of us are going to make money from the infrastructure.&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;The bottleneck is distribution, not production. Always was. Will be more so as production gets cheaper.&lt;/strong&gt;&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;"Programmatic SEO" is mostly a 2021-era pattern that smart operators are still riding the tail of, but the entrance is closed.&lt;/strong&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  What I'm doing now
&lt;/h2&gt;

&lt;p&gt;I'm not deleting the sites. They cost $0/month to run. The 11 clicks might be 1,000 clicks in 12 months — Google indexing curves are long. I'm just not going to write any more articles for them.&lt;/p&gt;

&lt;p&gt;Instead, I extracted the genuinely useful parts of my own workflow into 5 free tools and shipped them as &lt;a href="https://tools.toolrouteai.com" rel="noopener noreferrer"&gt;tools.toolrouteai.com&lt;/a&gt;:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Prompt Optimizer&lt;/strong&gt; — turns "write me a blog post about X" into a structured prompt with role, constraints, output format&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Comparison Builder&lt;/strong&gt; — pulls from a JSON index of ~50 AI tools, lets you pick 2-5, exports Markdown or PDF&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Obsidian Template Generator&lt;/strong&gt; — browser-side, generates a .zip of Markdown + Dataview + Templater files&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Price Tracker&lt;/strong&gt; — scheduled scraper of 50+ AI tool pricing pages, exposes RSS + JSON + a UI for change signals&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Side Hustle Ideas&lt;/strong&gt; — give it your skill + weekly hours + budget, returns 3 realistic ideas with first-week action plans&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;No signup, no API key, no paywall. $0/month to run. They're useful to me; they might be useful to you. If they're not, please tell me what's missing.&lt;/p&gt;

&lt;p&gt;And I'm writing this post — which is, finally, distribution.&lt;/p&gt;

&lt;h2&gt;
  
  
  If you're thinking about doing the same thing
&lt;/h2&gt;

&lt;p&gt;Don't, if your plan is "643 articles, ranking, AdSense." That door is closed.&lt;/p&gt;

&lt;p&gt;Do, if your plan is:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;5-20 articles, each one targeting something you have a genuine and verifiable edge on&lt;/li&gt;
&lt;li&gt;80% distribution (HN, Reddit, niche newsletters, podcast appearances, paid placement on niche sites)&lt;/li&gt;
&lt;li&gt;Treat the articles as proof of competence, not as traffic-generation devices&lt;/li&gt;
&lt;li&gt;Monetize with consulting, products, or paid newsletters — not display ads&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Or do, if your plan is "I want to learn the infrastructure." You'll learn a lot. Just don't expect a business at the end.&lt;/p&gt;




&lt;p&gt;I'm a solo maker based in China. I'll respond to every comment in this thread, including the ones telling me I missed something obvious — those are the most useful.&lt;/p&gt;

&lt;p&gt;If you want to follow what I do next, my email is on &lt;code&gt;tools.toolrouteai.com&lt;/code&gt;. No automated newsletter; I'll write to you when I have something worth saying.&lt;/p&gt;

&lt;p&gt;— Alex&lt;/p&gt;

</description>
      <category>ai</category>
      <category>seo</category>
      <category>indiehackers</category>
      <category>webdev</category>
    </item>
  </channel>
</rss>
