@@ -4,6 +4,8 @@ import type { ParserBuilder } from './parsers'
4
4
5
5
export type SearchParams = Record < string , string | string [ ] | undefined >
6
6
7
+ const $input : unique symbol = Symbol ( 'Input' )
8
+
7
9
type ExtractParserType < Parser > =
8
10
Parser extends ParserBuilder < any >
9
11
? ReturnType < Parser [ 'parseServerSide' ] >
@@ -16,33 +18,56 @@ export function createSearchParamsCache<
16
18
type ParsedSearchParams = {
17
19
[ K in Keys ] : ExtractParserType < Parsers [ K ] >
18
20
}
21
+
22
+ type Cache = {
23
+ searchParams : Partial < ParsedSearchParams >
24
+ [ $input ] ?: SearchParams
25
+ }
26
+
19
27
// Why not use a good old object here ?
20
28
// React's `cache` is bound to the render lifecycle of a page,
21
29
// whereas a simple object would be bound to the lifecycle of the process,
22
30
// which may be reused between requests in a serverless environment
23
31
// (warm lambdas on Vercel or AWS).
24
- const getCache = cache < ( ) => Partial < ParsedSearchParams > > ( ( ) => ( { } ) )
32
+ const getCache = cache < ( ) => Cache > ( ( ) => ( {
33
+ searchParams : { }
34
+ } ) )
25
35
function parse ( searchParams : SearchParams ) {
26
36
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
28
49
throw new Error ( error ( 501 ) )
29
50
}
51
+
30
52
for ( const key in parsers ) {
31
53
const parser = parsers [ key ] !
32
- c [ key ] = parser . parseServerSide ( searchParams [ key ] )
54
+ c . searchParams [ key ] = parser . parseServerSide ( searchParams [ key ] )
33
55
}
34
- return Object . freeze ( c ) as Readonly < ParsedSearchParams >
56
+
57
+ c [ $input ] = searchParams
58
+
59
+ return Object . freeze ( c . searchParams ) as Readonly < ParsedSearchParams >
35
60
}
36
61
function all ( ) {
37
- const c = getCache ( )
38
- if ( Object . keys ( c ) . length === 0 ) {
62
+ const { searchParams } = getCache ( )
63
+ if ( Object . keys ( searchParams ) . length === 0 ) {
39
64
throw new Error ( error ( 500 ) )
40
65
}
41
- return c as Readonly < ParsedSearchParams >
66
+ return searchParams as Readonly < ParsedSearchParams >
42
67
}
43
68
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 ]
46
71
if ( typeof entry === 'undefined' ) {
47
72
throw new Error (
48
73
error ( 500 ) +
0 commit comments