惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
G
GRAHAM CLULEY
P
Privacy & Cybersecurity Law Blog
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
宝玉的分享
宝玉的分享
P
Proofpoint News Feed
H
Help Net Security
V
Visual Studio Blog
阮一峰的网络日志
阮一峰的网络日志
C
Cisco Blogs
人人都是产品经理
人人都是产品经理
Know Your Adversary
Know Your Adversary
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
Recorded Future
Recorded Future
I
Intezer
罗磊的独立博客
T
The Exploit Database - CXSecurity.com
Blog — PlanetScale
Blog — PlanetScale
Malwarebytes
Malwarebytes
Spread Privacy
Spread Privacy
T
Tor Project blog
V
Vulnerabilities – Threatpost
云风的 BLOG
云风的 BLOG
腾讯CDC
B
Blog RSS Feed
Stack Overflow Blog
Stack Overflow Blog
F
Future of Privacy Forum
MyScale Blog
MyScale Blog
Latest news
Latest news
IT之家
IT之家
MongoDB | Blog
MongoDB | Blog
The Hacker News
The Hacker News
S
Securelist
博客园 - 【当耐特】
C
CXSECURITY Database RSS Feed - CXSecurity.com
T
Threat Research - Cisco Blogs
Jina AI
Jina AI
Cisco Talos Blog
Cisco Talos Blog
B
Blog
博客园 - 三生石上(FineUI控件)
Last Week in AI
Last Week in AI
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
M
MIT News - Artificial intelligence
V
V2EX
D
Darknet – Hacking Tools, Hacker News & Cyber Security
The Cloudflare Blog
The GitHub Blog
The GitHub Blog
博客园 - 聂微东
F
Full Disclosure
C
CERT Recently Published Vulnerability Notes

DEV Community

Stop Being Nice, Start Being Right": The Day My User Reconfigured My Reward Function Building a Database Performance Testing Tool With AI: The Honest Breakdown Hot To Run LLMs Locally Research blockchain with post-quantum Dilithium and custom zk-STARKs from scratch AI agents do not just need tool access. They need execution control. The CTO’s Blueprint for Governing Multi-Agent AI Systems in the Enterprise Upselling Explained Industry-Specific Tactics for EC Owners 2026 I Keep Hermes Agent's Self-Improvement OFF For the First 14 Days — Here's What Happens When I Don't I Built the Hermes + Claude Code Dual-Stack: Orchestrator Meets Coder — Here's the Full Architecture Stop Using .iterrows(). Here's What Actually Fast Looks Like I Built a SaaS to Stop the Awkward "Hey, Did You Get My Invoice?" Conversation I Renamed a Hot Postgres Table Without Dropping a Request How to Build a Self-Hosted AI Gateway With LiteLLM and Open WebUI What is a Webhook? A Complete Guide for Beginners Headless BI: How a Universal Semantic Layer Replaces Tool-Specific Models Beyond Translation: A Developer's Guide to App Localization (i18n & l10n) Aegis: Designing an Offline Ambient Co-Working Companion for High-Burnout Medical and STEM Grinds Local LLM Code Completion Showdown: Zed AI vs Continue vs Cursor (Honest 2026 Review) The Agentic Payment Protocol Wars Your No-Code AI Agent Has a Memory Problem The Agentic Payment Protocol Wars How to Bypass LinkedIn Commercial Use Limit in 2026 (Without Paying $150/mo) We built a statechart hosting platform where two actors in the same state can migrate to different versions — here's why that matters Playwright vs TWD: A Frontend Developer's Honest Comparison Claude Code's skillListingBudgetFraction: The Undocumented Setting Silently Killing Half Your Skills O GitHub pode mudar sua carreira mais do que você imagina Just redesigned and launched my developer portfolio 🚀 Would genuinely love some honest feedback from the dev community 👨‍💻 Data Virtualization and the Semantic Layer: Query Without Copying Launching opub: donated compute for open-source maintainers Four iteration rounds on a security scanner I run, all of them visible. Here is what the loop actually looks like. Why Good Abstractions Make Debugging Harder Found a Coordinated Inauthentic Network on GitHub: 24 Accounts, Fabricated History, and a Generator That Left Its PID in Three READMEs Cursor Just Released Composer 2.5. Here's What Actually Changed for AI Coding Agents. What Wrong Docs Cost Test Automation Teams Export Your DeepSeek Chats to Word, PDF, Google Docs, Markdown & Notion in One Click When the Docs Lie OpenShift Observability: Built-in vs. Bring-Your-Own If your AI initiative is pending for 6 months, the bottleneck is probably not technology Hermes Agent Under the Hood: The Open-Source Runtime for Autonomous AI Systems Expert Systems -The AI That Existed Before AI Was Cool AI-generated accessibility, an update — frontier models still fail, but skills change the game My HTML Learning Journey 🚀 The Day PayPal Failed and the Rust Rewrite Saved the Product Launch Google Sheets CRM: 4 Ways I've Actually Done It (with Apps Script Code) BrontoScope: AI-Powered Error Investigations The job of an AI engineer inside a 40-person company is not what most CEOs think it is Building a Clinical Speech-Therapy App With a Real SLP: 4 Lessons From PhoenixSteps 7 overlooked .Net features How Stripe Took 48 Hours and 3 API Calls to Break My Freelance Income Stream in Lagos Pretty normal Both Camps in the 'Left Behind' Argument Are Right About Each Other Flutter MCP Toolkit v3 Google Just Shipped Gemini 3.5 Flash. Here's What Developers Actually Need to Know. 🔐 Working with Private Symfony Recipes Rate limiting in web apps: what to protect before picking a library Rate limiting en aplicaciones web: qué proteger antes de elegir una librería What Are Lakehouse Catalogs? The Role of Catalogs in Apache Iceberg What It Really Takes to Become a Senior Software Engineer Microservices Were Never About Technology JS Crime Scene: The Misleading Array Project-as-code for a Directus v9 backend When the API literally burned your database after a typo COOKIES DPRK Hacking Trends 2026: AI‑Powered Supply Chain and Developer Environment Attacks Phone control for AI coding sessions is not a tiny terminal PayPal and Crypto Are Not Equals: How I Built a Gumroad Alternative for Restricted Countries Exploring Tech as a Content Writer I Raised Gemma 4's Token Cap. The Dense Model Stopped Refusing. React Server Components Don't Make Your App Fast by Default Multi-Stage Builds for a Next.js App — Reduce Image Size by 70% I Built a Chrome Extension That Teaches Vocabulary While You Browse Why I Walked Back from Next.js and RSC to a Plain SPA and a Separate Backend NeuralPocket: Private On-Device AI with Gemma 4 — Android & Web Github Speckit: Revolucionando o Desenvolvimento com SDD Cloud Cost Elasticity I Built a Payment System for Bangladesh—Heres Why Stripe Failed Us Polyglot Persistence in Microservices: Choosing the Right Database for Each Service Centralized Authentication for a Multi-Brand Laravel Ecosystem How I made a perfect recording button. Simple yet complex thing. Mumbli – my personal Wispr Flow Getting Paid Should Not Be a Geopolitical Nightmare: My NOWPayments Integration Story Four Layers of Validation in Kubernetes with Claude Code Prompt Flow — a visual side project for flow design, trace, and integration steps (looking for feedback) AI Citation Registry: Temporal Gaps in Government Publishing Cycles ShowDev: I built a 100% local, zero-upload PDF editor using WebAssembly JavaC Written by an AI Pipeline, Verified by Three Models. Is It Slop? Part1 Vulkan: Drawing Triangle 1 Why I Stopped Using useEffect to Sync State — and What I Use Instead Por qué dejé de usar useEffect para sincronizar estado y qué uso ahora Migrating a Long-Running WordPress Site to Payload CMS (And All The Chaos That Came With It) Hidden Partitioning: How Iceberg Eliminates Accidental Full Table Scans Azure DevOps Structure Explained: Organizations, Projects, and Repos Without the Mess A Simple React Hook for localStorage State, Expiry, and Sync I sold you on /scratchpad. Then I migrated to /note. Fixing WSL Errors on Windows 11 Your app is not Netflix. Stop building like it is. Resolving inter-service communication issue I built an email cleaner. CSV parsing took longer than the actual validators. How I Would Learn Full-Stack Development in 2026 If I Started From Zero
I audited our CMS and 86% of our articles were invisible. A Sanity gotcha.
sk8ordie84 · 2026-05-21 · via DEV Community

A week ago I ran a routine count on our Sanity dataset, expecting maybe a 5% gap between drafts and published articles. The result was 33 published, 253 drafts. 86% of the content I thought was on our site wasn't there. The bug had been silently shipping for the entire 9-day life of the project.

This post is the postmortem. It is specifically about Sanity, but the underlying gotcha (a CMS client default that disagrees with what you actually want at read time) applies to any headless setup.

The setup

I run Fax Office 1987, a small daily editorial publication. Next.js 15 App Router, Sanity for the CMS, Inngest for the dispatch pipeline. The editor (me) gets a review email for each AI-assisted draft and approves or rejects via a link. Approval was supposed to make a piece appear at /dispatch/<slug>.

The review route handler looked like this:

const next = action === 'approve' ? 'approved' : 'rejected'
await sanity
  .patch(id)
  .set({ reviewStatus: next, reviewedAt: new Date().toISOString() })
  .commit()

Enter fullscreen mode Exit fullscreen mode

Clean. Set the flag, return the HTML confirmation page. Done.

And separately, the public site read articles like this:

const sanity = createClient({
  projectId,
  dataset,
  apiVersion: '2024-10-01',
  useCdn: false,
  token: process.env.SANITY_WRITE_TOKEN, // we use a token so private-read works
})

export const ARTICLES_QUERY = `
  *[_type == "article"
      && defined(slug.current)
      && (reviewStatus == "approved" || !defined(reviewStatus))
    ]
    | order(publishedAt desc) { ... }
`

Enter fullscreen mode Exit fullscreen mode

For 9 days I thought this worked. Approve emails arrived, I clicked approve, the confirmation page said "the piece is now visible on the site." It wasn't.

The audit

I ran a per-status count:

const r = await client.fetch(`{
  "published":      count(*[_type=="article" && !(_id in path("drafts.**"))]),
  "draft_approved": count(*[_type=="article" && _id in path("drafts.**") && reviewStatus=="approved"]),
  "draft_pending":  count(*[_type=="article" && _id in path("drafts.**") && reviewStatus=="pending"]),
  "draft_rejected": count(*[_type=="article" && _id in path("drafts.**") && reviewStatus=="rejected"])
}`)

Enter fullscreen mode Exit fullscreen mode

Result:

{
  "published":       33,
  "draft_approved": 227,
  "draft_pending":    7,
  "draft_rejected":  19
}

Enter fullscreen mode Exit fullscreen mode

227 drafts marked approved. All of them had their reviewStatus flag set correctly. None of them were visible to readers.

Root cause #1: perspective default

Sanity documents have two layers. A draft sits at drafts.<id>, the published version sits at <id>. Both can coexist for the same logical document. When you fetch with a token, the default perspective overlays the draft on top of the published version and returns whichever exists. For an editorial site running with a token (because the dataset is in private-read mode for our use case), this is the wrong default. We always want to read the published version on the public site, never the draft.

Without perspective: 'published' set on the client, a draft document with reviewStatus == "approved" would pass our GROQ filter and get served to readers under the same slug as its published twin. We never noticed because the in-progress drafts were never published to begin with: the bug below kept them stuck.

Fix:

export const sanity = projectId
  ? createClient({
      projectId,
      dataset,
      apiVersion: '2024-10-01',
      useCdn: false,
      token: process.env.SANITY_WRITE_TOKEN,
      perspective: 'published', // <-- the one-line fix
    })
  : null

Enter fullscreen mode Exit fullscreen mode

Belt-and-suspenders, every public GROQ query also gained a filter:

const NO_DRAFTS = `!(_id in path("drafts.**"))`

Enter fullscreen mode Exit fullscreen mode

So even if a future caller built an ad-hoc client without the perspective set, the query itself would still hide drafts.

Root cause #2: the approve handler

After the perspective fix, the published count was still 33. The 227 approved drafts were still drafts, just with a flag set.

Reading the approve handler again with the perspective context in mind:

await sanity.patch(id).set({ reviewStatus: 'approved', ... }).commit()

Enter fullscreen mode Exit fullscreen mode

This patches the draft document. It does not promote it. The published version under the bare id never gets created. From the public site's point of view, nothing changed.

The standard Sanity draft-promotion idiom is:

if (next === 'approved' && id.startsWith('drafts.')) {
  const publishedId = id.replace(/^drafts\./, '')
  const draft = await sanity.getDocument(id)
  if (draft) {
    const { _id, _rev, _createdAt, _updatedAt, ...rest } = draft as any
    await sanity.createOrReplace({
      ...rest,
      _id: publishedId,
      _type: 'article',
      reviewStatus: 'approved',
      reviewedAt: new Date().toISOString(),
    })
    await sanity.delete(id)
  }
}

Enter fullscreen mode Exit fullscreen mode

Fetch the full draft, write it under the published id (strip the drafts. prefix), delete the draft. Sanity treats the result as published. The reject path still patches in place because rejected items stay as drafts on purpose: kept as a record, hidden from readers.

Backfill

That fixes new approvals. The 227 already in the backlog still needed promoting. A one-time script that walks every approved draft and applies the same promotion logic:

const drafts = await client.fetch(
  `*[_type == "article" && _id in path("drafts.**") && reviewStatus == "approved"]
    | order(_createdAt asc)`,
)

for (const draft of drafts) {
  const publishedId = draft._id.replace(/^drafts\./, '')
  // skip if a published twin already exists; don't clobber manual edits
  const existing = await client.fetch(
    `*[_id == $id][0]{ _id }`,
    { id: publishedId },
  )
  if (existing) continue

  const { _id, _rev, _createdAt, _updatedAt, ...rest } = draft
  await client.createOrReplace({ ...rest, _id: publishedId, _type: 'article' })
  await client.delete(draft._id)
}

Enter fullscreen mode Exit fullscreen mode

Ran it against production. 227 promoted, 0 errors. Published count moved from 33 to 260. Sitemap discovered URLs went from a couple dozen to 293.

Followed up with an IndexNow bulk ping so Bing, Yandex, and the consortium would crawl the new URLs without waiting for sitemap re-discovery. Single POST, 289 URLs, accepted in one shot.

The takeaway

The Sanity perspective default is not a bug. The docs are clear. The mistake was a blind spot: when you write code that uses a token for read operations (because your dataset is private-read), you have to actively pick a perspective. Otherwise you get whichever overlay Sanity decided to give you, which for a public website is rarely what you want.

The deeper lesson: I had two bugs that combined into invisible content. Either alone would have been visible. Together they hid the site from itself. A monthly audit catches this kind of compounding silently-fails-but-works-anyway state.

Code: the fix landed in two commits on the Fax Office 1987 repo. If you run Sanity and your dataset is private-read, the perspective: 'published' line might be the highest-leverage one-character change you ship this month.