Skip to content

Commit 4d4fc9e

Browse files
authored
fix: Shallow equality check on cache parsing input (#580)
1 parent bf8ab26 commit 4d4fc9e

File tree

4 files changed

+61
-16
lines changed

4 files changed

+61
-16
lines changed

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

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22

33
describe('cache', () => {
44
it('works in app router', () => {
5-
cy.visit('/app/cache?str=foo&num=42&bool=true')
5+
cy.visit('/app/cache?str=foo&num=42&bool=true&multi=foo&multi=bar')
66
cy.get('#parse-str').should('have.text', 'foo')
77
cy.get('#parse-num').should('have.text', '42')
88
cy.get('#parse-bool').should('have.text', 'true')

packages/nuqs/src/cache.test.ts

+37-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import { describe, expect, it, vi } from 'vitest'
2-
import { createSearchParamsCache } from './cache'
2+
import { compareSearchParams, createSearchParamsCache } from './cache'
33
import { parseAsString } from './parsers'
44

55
// provide a simple mock for React cache
@@ -22,7 +22,7 @@ describe('cache', () => {
2222
string: "I'm a string"
2323
}
2424

25-
it('allows parsing same object multiple times in a request', () => {
25+
it('allows parsing the same object multiple times in a request', () => {
2626
const cache = createSearchParamsCache({
2727
string: parseAsString
2828
})
@@ -37,6 +37,15 @@ describe('cache', () => {
3737
expect(cache.all()).toBe(all)
3838
})
3939

40+
it('allows parsing the same content with different references', () => {
41+
const cache = createSearchParamsCache({
42+
string: parseAsString
43+
})
44+
const copy = { ...input }
45+
expect(cache.parse(input).string).toBe(input.string)
46+
expect(cache.parse(copy).string).toBe(input.string)
47+
})
48+
4049
it('disallows parsing different objects in a request', () => {
4150
const cache = createSearchParamsCache({
4251
string: parseAsString
@@ -51,4 +60,30 @@ describe('cache', () => {
5160
expect(cache.all()).toBe(all)
5261
})
5362
})
63+
64+
describe('compareSearchParams', () => {
65+
it('works on empty search params', () => {
66+
expect(compareSearchParams({}, {})).toBe(true)
67+
})
68+
it('rejects different lengths', () => {
69+
expect(compareSearchParams({ a: 'a' }, { a: 'a', b: 'b' })).toBe(false)
70+
})
71+
it('rejects different values', () => {
72+
expect(compareSearchParams({ x: 'a' }, { x: 'b' })).toBe(false)
73+
})
74+
it('does not care about order', () => {
75+
expect(compareSearchParams({ x: 'a', y: 'b' }, { y: 'b', x: 'a' })).toBe(
76+
true
77+
)
78+
})
79+
it('supports array values (referentially stable)', () => {
80+
const array = ['a', 'b']
81+
expect(compareSearchParams({ x: array }, { x: array })).toBe(true)
82+
})
83+
it('does not do deep comparison', () => {
84+
expect(compareSearchParams({ x: ['a', 'b'] }, { x: ['a', 'b'] })).toBe(
85+
false
86+
)
87+
})
88+
})
5489
})

packages/nuqs/src/cache.ts

+22-12
Original file line numberDiff line numberDiff line change
@@ -34,28 +34,23 @@ export function createSearchParamsCache<
3434
}))
3535
function parse(searchParams: SearchParams) {
3636
const c = getCache()
37-
3837
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.
38+
// Parse has already been called...
39+
if (c[$input] && compareSearchParams(searchParams, c[$input])) {
40+
// ...but we're being called with the same contents again,
41+
// so we can safely return the same cached result (an example of when
42+
// this occurs would be if parse was called in generateMetadata as well
43+
// as the page itself).
4544
return all()
4645
}
47-
48-
// different input in the same request - fail
46+
// Different inputs in the same request - fail
4947
throw new Error(error(501))
5048
}
51-
5249
for (const key in parsers) {
5350
const parser = parsers[key]!
5451
c.searchParams[key] = parser.parseServerSide(searchParams[key])
5552
}
56-
5753
c[$input] = searchParams
58-
5954
return Object.freeze(c.searchParams) as Readonly<ParsedSearchParams>
6055
}
6156
function all() {
@@ -80,3 +75,18 @@ export function createSearchParamsCache<
8075
}
8176
return { parse, get, all }
8277
}
78+
79+
export function compareSearchParams(a: SearchParams, b: SearchParams) {
80+
if (a === b) {
81+
return true
82+
}
83+
if (Object.keys(a).length !== Object.keys(b).length) {
84+
return false
85+
}
86+
for (const key in a) {
87+
if (a[key] !== b[key]) {
88+
return false
89+
}
90+
}
91+
return true
92+
}

packages/nuqs/src/index.server.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
1-
export * from './cache'
1+
export { createSearchParamsCache, type SearchParams } from './cache'
22
export * from './parsers'
33
export { createSerializer } from './serializer'

0 commit comments

Comments
 (0)