AgentSkillsCN

caching-strategies

为 Flare Stack 博客的架构提供全面概览。当您需要深入了解项目结构、技术栈、路由层级(Hono/TanStack)、依赖注入机制,或目录组织方式时,此技能将为您提供清晰的指引。

SKILL.md
--- frontmatter
name: caching-strategies
description: Dual-layer caching strategies for the Flare Stack Blog. Use when implementing CDN cache headers, KV caching with versioned invalidation, or debugging cache-related issues.

Caching Strategies

The project employs a dual-layer caching architecture: CDN (HTTP headers) and KV (Cloudflare KV storage).

CDN Layer (HTTP Headers)

Control browser and CDN caching via response headers set by middlewares.

Setting Cache Headers

Use createCacheHeaderMiddleware(strategy) in Server Functions to set Cache-Control headers:

typescript
import { createServerFn } from "@tanstack/react-start";
import {
  createCacheHeaderMiddleware,
  createRateLimitMiddleware,
} from "@/lib/middlewares";

export const getPostsFn = createServerFn()
  .middleware([
    createRateLimitMiddleware({
      capacity: 60,
      interval: "1m",
      key: "posts:list",
    }),
    createCacheHeaderMiddleware("swr"), // Sets SWR Cache-Control headers
  ])
  .handler(({ context }) => PostService.getPosts(context));

Cache Header Strategies

StrategyHeaderUse Case
"swr"Stale-While-RevalidatePublic API responses
"immutable"Long-term immutableHashed static assets
"private"no-store, privateAuth/admin responses

Cache Control Constants (lib/constants.ts)

These constants are used by the middleware factory. Additional constants for error pages:

ConstantUse Case
CACHE_CONTROL.notFound404 pages
CACHE_CONTROL.serverError500 pages

Invalidation

Purge CDN cache using the Cloudflare API:

typescript
await purgePostCDNCache(context, post.slug);

KV Layer (Cloudflare KV)

Used for persistent caching of longer-lived data (post lists, details).

Cache Key Definition

The CacheKey type supports both strings and readonly arrays (tuples), allowing for type-safe key construction using as const.

typescript
// features/cache/types.ts
export type CacheKey =
  | string
  | readonly (string | number | boolean | null | undefined)[];

Cache Key Factory Pattern

Instead of hardcoding key arrays in services, define Cache Key Factories in the feature's schema.ts. This provides a single source of truth and ensures types match the requirements of the cache key.

1. Define Factory in schema.ts

typescript
// features/posts/posts.schema.ts
export const POSTS_CACHE_KEYS = {
  /** Post detail cache key (includes version) */
  detail: (version: string, slug: string) => [version, "post", slug] as const,
} as const;

2. Use in Service Layer

Pass the tuple directly to CacheService functions. No spread ([...]) is needed since CacheKey supports readonly arrays.

typescript
const version = await CacheService.getVersion(context, "posts:detail");
return await CacheService.get(
  context,
  POSTS_CACHE_KEYS.detail(version, data.slug),
  PostSchema,
  fetcher,
);

Versioned Key Invalidation Strategy

This pattern enables efficient bulk invalidation without iterating through keys:

1. Get Current Version

typescript
const version = await CacheService.getVersion(context, "posts:detail");
// Returns "v1", "v2", etc.

2. Bump Version to Invalidate

When data changes, increment the version number:

typescript
await CacheService.bumpVersion(context, "posts:detail");
// All old keys with the previous version become unreachable

3. Direct Key Deletion

For single-record invalidation, delete the specific key using the factory:

typescript
const version = await CacheService.getVersion(context, "posts:detail");
await CacheService.deleteKey(context, POSTS_CACHE_KEYS.detail(version, slug));

Complete Example

typescript
// posts.service.ts
import { POSTS_CACHE_KEYS } from "./posts.schema";

export async function updatePost(
  context: DbContext & { executionCtx: ExecutionContext },
  data: UpdatePostInput,
) {
  // 1. Update in database
  const post = await PostRepo.updatePost(context.db, data);

  // 2. Invalidate KV cache
  await CacheService.bumpVersion(context, "posts:list");
  const version = await CacheService.getVersion(context, "posts:detail");
  await CacheService.deleteKey(context, POSTS_CACHE_KEYS.detail(version, post.slug));

  // 3. Purge CDN cache
  await purgePostCDNCache(context.env, post.slug);

  return post;
}

Cache Namespace Conventions

NamespaceData TypeInvalidation Trigger
posts:listPost listingsPost create/update/delete
posts:detailIndividual postsPost update/delete
tags:listTag listingsTag create/update/delete
comments:listComment listingsComment create/approve/delete

When to Use Each Layer

ScenarioCDNKV
Public API responses✅ SWR✅ Version-keyed
Admin API responses❌ PrivateOptional
Static assets✅ Immutable
User-specific data❌ PrivateDepends

Debugging Cache Issues

  1. Stale data after update?

    • Check if bumpVersion() was called
    • Verify CDN purge completed
    • Check cache key construction
  2. Cache misses?

    • Verify version string consistency
    • Check TTL settings
    • Inspect key serialization
  3. Memory issues?

    • Review cached data size
    • Consider selective field caching