sanity-live-cache-components
Migrar aplicativos next-sanity para cacheComponents - modo estrito, padrão de componente de três camadas, perspective/stega/includeDrafts explícitos, convenções de prop-drilling
npx skills add https://github.com/sanity-io/next-sanity --skill sanity-live-cache-componentsSanity Live + Cache Components
Wires next-sanity into a Next.js 16+ app with cacheComponents: true. Data is fetched with sanityFetch (which calls cacheTag/cacheLife internally), and <SanityLive> in the root layout revalidates cached content over an EventSource connection to Sanity Content Lake. Visual Editing and Presentation Tool are fully supported when draft mode is enabled.
Read the relevant guide in node_modules/next/dist/docs/ (when available) before writing code. If a guide conflicts with this skill, follow this skill.
This skill assumes familiarity with the next-cache-components skill — it covers 'use cache', cacheLife, cacheTag, and the cookies/headers/params rule. The only Sanity-relevant exception: await draftMode() is allowed inside 'use cache' (Next.js bypasses caching when draft mode is enabled — see the use cache reference).
Prerequisites
- Next.js 16.2+ installed in the project (check
package.jsonor runpnpm list next/npm ls next— don't usepnpm view next version, that reports the registry's latest, not what's installed). AGENTS.mdexists, or follow the guide.- These environment variables are set:
NEXT_PUBLIC_SANITY_PROJECT_IDNEXT_PUBLIC_SANITY_DATASETSANITY_API_READ_TOKEN
- Embedded Sanity Studio configuration (
sanity.config.ts,sanity.cli.ts, anything undersanity/) needs no changes — this skill only touches the Next.js app surface.
Reference files
| File | When to read |
|---|---|
| reference/live-helpers.md | Full client.ts / live.ts, sanityFetch* and getDynamicFetchOptions details |
| reference/three-layer-pattern.md | The Page → Dynamic → Cached pattern for page.tsx, including the searchParams variant |
| reference/layouts.md | Non-blocking data fetching inside layout.tsx with a shared 'use cache' helper |
| reference/dynamic-segments.md | High-performance [slug] routes: loading.tsx + partial generateStaticParams, or non-blocking dynamic params in a layout |
1. Install next-sanity@^13
npm install next-sanity@^13 --save-exact
Migrating an existing Sanity Live setup
If the app is already using defineLive, this skill is a refactor, not a rewrite. The 5-step sequence below still applies, but watch for these specific differences:
- Don't overwrite
client.tsorlive.tsif they exist. Append missing options. Preserve any existingtokenandstega.*settings — see reference/live-helpers.md. - Search the codebase for hardcoded
perspective: 'published'andstega: falseinsanityFetchcallsites and refactor them to sourceperspective/stegaviagetDynamicFetchOptionsand the three-layer pattern. - Search for
sanityFetchcalls insidegenerateStaticParams→ swap forsanityFetchStaticParams. - Search for
sanityFetchcalls insidegenerateMetadata/sitemap.ts/opengraph-image.tsx/ etc. → swap forsanityFetchMetadata. - Search for
sanityFetchcalls directly inside a'use server'function → split into a separate'use cache'helper. - Verify there is exactly one
<SanityLive>and one<VisualEditing>in the tree. Multiple renders are undefined behavior.
The "Anti-patterns to grep for" section at the bottom of this file lists the search patterns.
2. Configure next.config.ts
Enable cacheComponents and set cacheLife.default to sanity so default revalidation is 1 year (instead of 15 minutes). sanityFetch is optimized for on-demand revalidation and doesn't need time-based revalidation.
// next.config.ts
import type {NextConfig} from 'next'
import {sanity} from 'next-sanity/live/cache-life'
const nextConfig: NextConfig = {
cacheComponents: true,
cacheLife: {default: sanity},
}
export default nextConfig
3. Configure defineLive and export helpers
Create src/sanity/lib/client.ts and src/sanity/lib/live.ts. The minimal defineLive call:
// src/sanity/lib/live.ts (excerpt)
export const {SanityLive, sanityFetch} = defineLive({
client,
serverToken: token,
browserToken: token,
strict: true,
})
Full file contents (including client.ts, getDynamicFetchOptions, sanityFetchMetadata, sanityFetchStaticParams) and per-helper guidance: reference/live-helpers.md.
The helpers exported from live.ts:
| Helper | Used in |
|---|---|
sanityFetch | 'use cache' components rendered from page.tsx / layout.tsx |
sanityFetchMetadata | generateMetadata, generateViewport, sitemap.ts, robots.ts, opengraph-image.tsx, etc. |
sanityFetchStaticParams | generateStaticParams only |
getDynamicFetchOptions | Resolving perspective/stega outside any 'use cache' boundary |
SanityLive | Rendered once in a root layout |
4. Render <SanityLive> in a root layout
<SanityLive> and <VisualEditing> both belong in a layout.tsx, never a page.tsx. Both must be rendered at most once across the whole tree — duplicate renders are undefined behavior.
includeDraftsis required whendefineLiveis configured withstrict: true(the recommended setup). TypeScript will surface the error if it's missing; passincludeDrafts={isDraftMode}so live revalidation includes drafts only in draft mode.- Preserve any existing optional callback props on
<SanityLive>when migrating:onError,onWelcome,onReconnect. They are commonly wired to a toast/notification helper and silently dropping them regresses UX.
// src/app/layout.tsx
import {SanityLive} from '@/sanity/lib/live'
import {VisualEditing} from 'next-sanity/visual-editing'
import {draftMode} from 'next/headers'
export default async function RootLayout({children}: LayoutProps<'/'>) {
const {isEnabled: isDraftMode} = await draftMode()
return (
<html lang="en">
<body>
{children}
<SanityLive includeDrafts={isDraftMode} />
{isDraftMode && <VisualEditing />}
</body>
</html>
)
}
With an embedded Sanity Studio
If a route mounts NextStudio from next-sanity/studio (e.g. app/studio/[[...index]]/page.tsx), <SanityLive> must live in a layout the embedded studio doesn't share. Use route groups: put <SanityLive> in src/app/(website)/layout.tsx and keep the rest of the app under src/app/(website).
5. Apply the three-layer pattern to pages and layouts
Every route that should be statically prerendered uses the same shape:
Page/Layout (Layer 1: draftMode branch)
├── NOT draft mode → <CachedX perspective="published" stega={false} /> (no Suspense)
└── draft mode → <Suspense fallback={...}>
<DynamicX params={params} /> (Layer 2: awaits dynamic APIs)
└── <CachedX perspective={p} stega={s} /> (Layer 3: 'use cache')
Critical rule: Only Layer 3 carries 'use cache'. The top-level Page / Layout must not have 'use cache' — it awaits params, searchParams, or cookies() (via getDynamicFetchOptions), and those dynamic APIs are forbidden inside 'use cache'. Layer 3 carrying 'use cache' is enough for the whole route to prerender into the static shell. Adding 'use cache' to the top-level function is the most common failure mode — TypeScript and the runtime will both complain.
Pick the right reference for the file you're editing:
page.tsxwith static orgenerateStaticParams-backed params → reference/three-layer-pattern.md.page.tsxthat usessearchParamsor other dynamic APIs → thesearchParamsvariant in reference/three-layer-pattern.md.layout.tsxthat fetches its own data → reference/layouts.md.- Dynamic
[slug]route that needs theloading.tsx+ partialgenerateStaticParamsoptimization, or a layout that needs non-blockingparams→ reference/dynamic-segments.md.
Anti-patterns to grep for
When auditing an app, search for these and refactor:
perspective: 'published'andstega: falsehardcoded together in asanityFetchcall → use the three-layer pattern, sourceperspective/stegaviagetDynamicFetchOptions.sanityFetch(directly inside a function whose body begins with'use server'→ split into a separate'use cache'helper.sanityFetch(insidegenerateStaticParams→ swap forsanityFetchStaticParams.sanityFetch(insidegenerateMetadata/generateViewport/sitemap.ts/robots.ts/opengraph-image.tsxetc. → swap forsanityFetchMetadataand resolveperspectiveviagetDynamicFetchOptions.await draftMode()immediately followed byawait getDynamicFetchOptions()at the top of apage.tsxorlayout.tsxwithout a siblingloading.tsx→ move those dynamic-API calls into a child component wrapped in<Suspense>so the static shell can prerender.- More than one
<SanityLive>or<VisualEditing>rendered in the tree → consolidate to a single render in the right layout.