Skip to content

Commit 846e029

Browse files
committed
test: Throttle scheduling
Removed premature optimisation: skipping scheduling a flush returns a different Promise reference, breaking existing behaviour.
1 parent 8124b19 commit 846e029

File tree

2 files changed

+96
-11
lines changed

2 files changed

+96
-11
lines changed

packages/nuqs/src/lib/queues/throttle.test.ts

+96-7
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,19 @@
1-
import { describe, expect, it, vi } from 'vitest'
1+
import { afterEach, describe, expect, it, vi } from 'vitest'
22
import type { UpdateUrlFunction } from '../../adapters/lib/defs'
33
import { defaultRateLimit } from './rate-limiting'
44
import { ThrottledQueue, type UpdateQueueAdapterContext } from './throttle'
55

6+
const mockAdapter = {
7+
updateUrl: vi.fn<UpdateUrlFunction>(),
8+
getSearchParamsSnapshot() {
9+
return new URLSearchParams()
10+
}
11+
} satisfies UpdateQueueAdapterContext
12+
13+
afterEach(() => {
14+
mockAdapter.updateUrl.mockReset()
15+
})
16+
617
describe('throttle: ThrottleQueue value queueing', () => {
718
it('should enqueue key & values', () => {
819
const queue = new ThrottledQueue()
@@ -70,12 +81,6 @@ describe('throttle: ThrottleQueue option combination logic', () => {
7081
}
7182
const startTransitionA = vi.fn().mockImplementation(mockStartTransition)
7283
const startTransitionB = vi.fn().mockImplementation(mockStartTransition)
73-
const mockAdapter: UpdateQueueAdapterContext = {
74-
updateUrl: vi.fn<UpdateUrlFunction>(),
75-
getSearchParamsSnapshot() {
76-
return new URLSearchParams()
77-
}
78-
}
7984
const queue = new ThrottledQueue()
8085
queue.push({
8186
key: 'a',
@@ -113,3 +118,87 @@ describe('throttle: ThrottleQueue option combination logic', () => {
113118
expect(queue.timeMs).toBe(100)
114119
})
115120
})
121+
122+
describe('throttle: flush', () => {
123+
it('returns a Promise of updated URL search params', async () => {
124+
const throttle = new ThrottledQueue()
125+
throttle.push({ key: 'a', query: 'a', options: {} })
126+
const promise = throttle.flush(mockAdapter)
127+
await expect(promise).resolves.toEqual(new URLSearchParams('?a=a'))
128+
expect(mockAdapter.updateUrl).toHaveBeenCalledExactlyOnceWith(
129+
new URLSearchParams('?a=a'),
130+
{
131+
history: 'replace',
132+
scroll: false,
133+
shallow: true
134+
}
135+
)
136+
})
137+
it('combines updates in order of push', async () => {
138+
const throttle = new ThrottledQueue()
139+
throttle.push({ key: 'b', query: 'b', options: {} })
140+
throttle.push({ key: 'a', query: 'a', options: {} })
141+
const promise = throttle.flush(mockAdapter)
142+
await expect(promise).resolves.toEqual(new URLSearchParams('?b=b&a=a'))
143+
expect(mockAdapter.updateUrl).toHaveBeenCalledExactlyOnceWith(
144+
new URLSearchParams('?b=b&a=a'),
145+
{
146+
history: 'replace',
147+
scroll: false,
148+
shallow: true
149+
}
150+
)
151+
})
152+
it('returns the same Promise for multiple flushes in the same tick', () => {
153+
vi.useFakeTimers()
154+
const throttle = new ThrottledQueue()
155+
throttle.push({ key: 'b', query: 'b', options: {} })
156+
const p1 = throttle.flush(mockAdapter)
157+
throttle.push({ key: 'a', query: 'a', options: {} })
158+
const p2 = throttle.flush(mockAdapter)
159+
expect(p1).toBe(p2)
160+
vi.runAllTimers()
161+
expect(mockAdapter.updateUrl).toHaveBeenCalledExactlyOnceWith(
162+
new URLSearchParams('?b=b&a=a'),
163+
{
164+
history: 'replace',
165+
scroll: false,
166+
shallow: true
167+
}
168+
)
169+
})
170+
it('returns the same Promise if the initial flush has no updates', () => {
171+
vi.useFakeTimers()
172+
const throttle = new ThrottledQueue()
173+
const p1 = throttle.flush(mockAdapter)
174+
throttle.push({ key: 'a', query: 'a', options: {} })
175+
const p2 = throttle.flush(mockAdapter)
176+
expect(p1).toBe(p2)
177+
vi.runAllTimers()
178+
expect(mockAdapter.updateUrl).toHaveBeenCalledExactlyOnceWith(
179+
new URLSearchParams('?a=a'),
180+
{
181+
history: 'replace',
182+
scroll: false,
183+
shallow: true
184+
}
185+
)
186+
})
187+
it('returns the same Promise if the second flush has no updates', () => {
188+
vi.useFakeTimers()
189+
const throttle = new ThrottledQueue()
190+
throttle.push({ key: 'a', query: 'a', options: {} })
191+
const p1 = throttle.flush(mockAdapter)
192+
const p2 = throttle.flush(mockAdapter)
193+
expect(p1).toBe(p2)
194+
vi.runAllTimers()
195+
expect(mockAdapter.updateUrl).toHaveBeenCalledExactlyOnceWith(
196+
new URLSearchParams('?a=a'),
197+
{
198+
history: 'replace',
199+
scroll: false,
200+
shallow: true
201+
}
202+
)
203+
})
204+
})

packages/nuqs/src/lib/queues/throttle.ts

-4
Original file line numberDiff line numberDiff line change
@@ -80,10 +80,6 @@ export class ThrottledQueue {
8080
// Flush already scheduled
8181
return this.resolvers.promise
8282
}
83-
if (this.updateMap.size === 0) {
84-
// Nothing to flush
85-
return Promise.resolve(getSearchParamsSnapshot())
86-
}
8783
this.resolvers = withResolvers<URLSearchParams>()
8884
const flushNow = () => {
8985
this.lastFlushedAt = performance.now()

0 commit comments

Comments
 (0)