Skip to content

Commit 2727585

Browse files
authored
fix: don't fail parse if called again with identical input (#557)
* fix: don't fail parse if called again with identical input * test: add e2e test * style: run prettier * fix: use a symbol to avoid conflict * style: run prettier * fix: freeze searchParams
1 parent 2461d8d commit 2727585

File tree

4 files changed

+101
-13
lines changed

4 files changed

+101
-13
lines changed

packages/e2e/cypress/e2e/cache.cy.js

+1
Original file line numberDiff line numberDiff line change
@@ -18,5 +18,6 @@ describe('cache', () => {
1818
cy.get('#get-bool').should('have.text', 'true')
1919
cy.get('#get-def').should('have.text', 'default')
2020
cy.get('#get-nope').should('have.text', 'null')
21+
cy.title().should('eq', 'metadata-title-str:foo')
2122
})
2223
})

packages/e2e/src/app/app/cache/page.tsx

+12-4
Original file line numberDiff line numberDiff line change
@@ -4,11 +4,11 @@ import { Get } from './get'
44
import { cache } from './searchParams'
55
import { Set } from './set'
66

7-
export default function Page({
8-
searchParams
9-
}: {
7+
type Props = {
108
searchParams: Record<string, string | string[] | undefined>
11-
}) {
9+
}
10+
11+
export default function Page({ searchParams }: Props) {
1212
const { str, bool, num, def, nope } = cache.parse(searchParams)
1313
return (
1414
<>
@@ -29,3 +29,11 @@ export default function Page({
2929
</>
3030
)
3131
}
32+
33+
export function generateMetadata({ searchParams }: Props) {
34+
// parse here too to ensure we can idempotently parse the same search params as the page in the same request
35+
const { str } = cache.parse(searchParams)
36+
return {
37+
title: `metadata-title-str:${str}`
38+
}
39+
}

packages/nuqs/src/cache.test.ts

+54
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
import { describe, expect, it, vi } from 'vitest'
2+
import { createSearchParamsCache } from './cache'
3+
import { parseAsString } from './parsers'
4+
5+
// provide a simple mock for React cache
6+
vi.mock('react', () => {
7+
return {
8+
cache<T, CachedFunction extends () => T>(fn: CachedFunction) {
9+
let cache: T | undefined = undefined
10+
function cachedFn() {
11+
cache ??= fn()
12+
return cache
13+
}
14+
return cachedFn
15+
}
16+
}
17+
})
18+
19+
describe('cache', () => {
20+
describe('createSearchParamsCache', () => {
21+
const input = {
22+
string: "I'm a string"
23+
}
24+
25+
it('allows parsing same object multiple times in a request', () => {
26+
const cache = createSearchParamsCache({
27+
string: parseAsString
28+
})
29+
30+
// parse the input and perform some sanity checks
31+
expect(cache.parse(input).string).toBe(input.string)
32+
const all = cache.all()
33+
expect(all.string).toBe(input.string)
34+
35+
// second call should be successful and return the same result
36+
expect(cache.parse(input).string).toBe(input.string)
37+
expect(cache.all()).toBe(all)
38+
})
39+
40+
it('disallows parsing different objects in a request', () => {
41+
const cache = createSearchParamsCache({
42+
string: parseAsString
43+
})
44+
expect(cache.parse(input).string).toBe(input.string)
45+
const all = cache.all()
46+
47+
// second call with different object should fail
48+
expect(() => cache.parse({ string: 'I am a different string' })).toThrow()
49+
50+
// cache still works though
51+
expect(cache.all()).toBe(all)
52+
})
53+
})
54+
})

packages/nuqs/src/cache.ts

+34-9
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,8 @@ import type { ParserBuilder } from './parsers'
44

55
export type SearchParams = Record<string, string | string[] | undefined>
66

7+
const $input: unique symbol = Symbol('Input')
8+
79
type ExtractParserType<Parser> =
810
Parser extends ParserBuilder<any>
911
? ReturnType<Parser['parseServerSide']>
@@ -16,33 +18,56 @@ export function createSearchParamsCache<
1618
type ParsedSearchParams = {
1719
[K in Keys]: ExtractParserType<Parsers[K]>
1820
}
21+
22+
type Cache = {
23+
searchParams: Partial<ParsedSearchParams>
24+
[$input]?: SearchParams
25+
}
26+
1927
// Why not use a good old object here ?
2028
// React's `cache` is bound to the render lifecycle of a page,
2129
// whereas a simple object would be bound to the lifecycle of the process,
2230
// which may be reused between requests in a serverless environment
2331
// (warm lambdas on Vercel or AWS).
24-
const getCache = cache<() => Partial<ParsedSearchParams>>(() => ({}))
32+
const getCache = cache<() => Cache>(() => ({
33+
searchParams: {}
34+
}))
2535
function parse(searchParams: SearchParams) {
2636
const c = getCache()
27-
if (Object.isFrozen(c)) {
37+
38+
if (Object.isFrozen(c.searchParams)) {
39+
// parse has already been called
40+
if (searchParams === c[$input]) {
41+
// but we're being called with the identical object again, so we can safely return the same cached result
42+
// (an example of when this occurs would be if parse was called in generateMetadata as well as the page itself).
43+
// note that this simply checks for referential equality and will still fail if a different object with the
44+
// same contents is passed. fortunately next.js uses the same object for search params in the same request.
45+
return all()
46+
}
47+
48+
// different input in the same request - fail
2849
throw new Error(error(501))
2950
}
51+
3052
for (const key in parsers) {
3153
const parser = parsers[key]!
32-
c[key] = parser.parseServerSide(searchParams[key])
54+
c.searchParams[key] = parser.parseServerSide(searchParams[key])
3355
}
34-
return Object.freeze(c) as Readonly<ParsedSearchParams>
56+
57+
c[$input] = searchParams
58+
59+
return Object.freeze(c.searchParams) as Readonly<ParsedSearchParams>
3560
}
3661
function all() {
37-
const c = getCache()
38-
if (Object.keys(c).length === 0) {
62+
const { searchParams } = getCache()
63+
if (Object.keys(searchParams).length === 0) {
3964
throw new Error(error(500))
4065
}
41-
return c as Readonly<ParsedSearchParams>
66+
return searchParams as Readonly<ParsedSearchParams>
4267
}
4368
function get<Key extends Keys>(key: Key): ParsedSearchParams[Key] {
44-
const c = getCache()
45-
const entry = c[key]
69+
const { searchParams } = getCache()
70+
const entry = searchParams[key]
4671
if (typeof entry === 'undefined') {
4772
throw new Error(
4873
error(500) +

0 commit comments

Comments
 (0)