Skip to content

Commit ddb8f83

Browse files
committed
chore: Shallow-compare objects rather than stringifying
While the searchParams object is not referentially stable, its values are, so we can do a simple shallow comparison. Adding a test case in the e2e test to make sure this holds with array of values (repeated keys in the query string).
1 parent 9000440 commit ddb8f83

File tree

3 files changed

+39
-19
lines changed

3 files changed

+39
-19
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

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

55
// provide a simple mock for React cache
@@ -61,15 +61,29 @@ describe('cache', () => {
6161
})
6262
})
6363

64-
describe('stringify', () => {
65-
it('works on string values', () => {
66-
expect(stringify({ foo: 'bar' })).toEqual('foo=bar')
64+
describe('compareSearchParams', () => {
65+
it('works on empty search params', () => {
66+
expect(compareSearchParams({}, {})).toBe(true)
6767
})
68-
it('works on array values', () => {
69-
expect(stringify({ foo: ['bar', 'baz'] })).toEqual('foo=bar%2Cbaz')
68+
it('rejects different lengths', () => {
69+
expect(compareSearchParams({ a: 'a' }, { a: 'a', b: 'b' })).toBe(false)
7070
})
71-
it('works on undefined values', () => {
72-
expect(stringify({ foo: undefined })).toEqual('foo=undefined')
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+
)
7387
})
7488
})
7589
})

packages/nuqs/src/cache.ts

+16-10
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@ export function createSearchParamsCache<
2121

2222
type Cache = {
2323
searchParams: Partial<ParsedSearchParams>
24-
[$input]?: string
24+
[$input]?: SearchParams
2525
}
2626

2727
// Why not use a good old object here ?
@@ -36,7 +36,7 @@ export function createSearchParamsCache<
3636
const c = getCache()
3737
if (Object.isFrozen(c.searchParams)) {
3838
// Parse has already been called...
39-
if (stringify(searchParams) === c[$input]) {
39+
if (c[$input] && compareSearchParams(searchParams, c[$input])) {
4040
// ...but we're being called with the same contents again,
4141
// so we can safely return the same cached result (an example of when
4242
// this occurs would be if parse was called in generateMetadata as well
@@ -50,7 +50,7 @@ export function createSearchParamsCache<
5050
const parser = parsers[key]!
5151
c.searchParams[key] = parser.parseServerSide(searchParams[key])
5252
}
53-
c[$input] = stringify(searchParams)
53+
c[$input] = searchParams
5454
return Object.freeze(c.searchParams) as Readonly<ParsedSearchParams>
5555
}
5656
function all() {
@@ -76,11 +76,17 @@ export function createSearchParamsCache<
7676
return { parse, get, all }
7777
}
7878

79-
/**
80-
* Internal function: reduce a SearchParams object to a string
81-
* for internal comparison of inputs passed to the cache's `parse` function.
82-
*/
83-
export function stringify(searchParams: SearchParams) {
84-
// @ts-expect-error - URLSearchParams is actually OK with array values
85-
return new URLSearchParams(searchParams).toString()
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
8692
}

0 commit comments

Comments
 (0)