musora-content-services 2.155.0 → 2.155.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (33) hide show
  1. package/CHANGELOG.md +14 -0
  2. package/jest.config.js +6 -3
  3. package/package.json +1 -1
  4. package/src/services/contentProgress.js +7 -11
  5. package/src/services/offline/progress.ts +0 -4
  6. package/src/services/sync/context/index.ts +7 -0
  7. package/src/services/sync/effects/push-failure-notification.ts +2 -2
  8. package/src/services/sync/retry.ts +8 -7
  9. package/src/services/sync/store/index.ts +0 -5
  10. package/src/services/sync/store/push-coalescer.ts +1 -1
  11. package/src/services/sync/telemetry/index.ts +1 -1
  12. package/test/setupTimers.js +13 -0
  13. package/test/unit/sync/adapters/idb-errors.test.ts +1 -1
  14. package/test/unit/sync/adapters/sqlite-errors.test.ts +1 -1
  15. package/test/unit/sync/effects/logout-warning.test.ts +158 -0
  16. package/test/unit/sync/effects/push-failure-notification.test.ts +196 -0
  17. package/test/unit/sync/fetch.test.ts +224 -0
  18. package/test/unit/sync/helpers/TestModel.ts +1 -1
  19. package/test/unit/sync/helpers/index.ts +12 -12
  20. package/test/unit/sync/manager.test.ts +303 -0
  21. package/test/unit/sync/repositories/content-likes.test.ts +4 -4
  22. package/test/unit/sync/repositories/practices.test.ts +4 -4
  23. package/test/unit/sync/repositories/progress.test.ts +4 -4
  24. package/test/unit/sync/repositories/user-award-progress.test.ts +387 -0
  25. package/test/unit/sync/resolver.test.ts +232 -0
  26. package/test/unit/sync/retry.test.ts +314 -0
  27. package/test/unit/sync/store/cross-user-protection.test.ts +217 -0
  28. package/test/unit/sync/store/push-coalescer.test.ts +156 -0
  29. package/test/unit/sync/store/store-idb.test.ts +4 -4
  30. package/test/unit/sync/store/store.test.ts +91 -4
  31. package/test/unit/sync/utils/throttle.test.ts +245 -0
  32. package/tsconfig.json +6 -2
  33. package/.claude/settings.local.json +0 -10
@@ -0,0 +1,314 @@
1
+ import SyncRetry from '@/services/sync/retry'
2
+ import type SyncContext from '@/services/sync/context/index'
3
+ import type { SyncTelemetry } from '@/services/sync/telemetry/index'
4
+
5
+ // Subclass exposes sleep as a spy — no `as any` access to private internals
6
+ class TestableSyncRetry extends SyncRetry {
7
+ readonly sleepSpy = jest.fn().mockResolvedValue(undefined)
8
+ protected override sleep(ms: number) {
9
+ return this.sleepSpy(ms)
10
+ }
11
+ }
12
+
13
+ // ---
14
+
15
+ function makeRetry(online = true) {
16
+ let connectivityListener: ((v: boolean) => void) | null = null
17
+
18
+ const connectivity = {
19
+ getValue: jest.fn().mockReturnValue(online),
20
+ subscribe: jest.fn().mockImplementation((listener: (v: boolean) => void) => {
21
+ connectivityListener = listener
22
+ return () => { connectivityListener = null }
23
+ }),
24
+ }
25
+ const context = { connectivity } as unknown as SyncContext
26
+ const telemetry = { debug: jest.fn() } as unknown as SyncTelemetry
27
+ const retry = new TestableSyncRetry(context, telemetry)
28
+
29
+ const triggerConnectivity = (isOnline: boolean) => {
30
+ connectivity.getValue.mockReturnValue(isOnline)
31
+ connectivityListener?.(isOnline)
32
+ }
33
+
34
+ return { retry, connectivity, telemetry, triggerConnectivity }
35
+ }
36
+
37
+ const ok = { ok: true as const, results: [] }
38
+ const retryable = { ok: false as const, failureType: 'fetch' as const, isRetryable: true }
39
+ const nonRetryable = { ok: false as const, failureType: 'fetch' as const, isRetryable: false }
40
+
41
+ // ---
42
+
43
+ describe('successful request', () => {
44
+ test('returns result on first attempt', async () => {
45
+ const { retry } = makeRetry()
46
+ const result = await retry.request(jest.fn().mockResolvedValue(ok))
47
+
48
+ expect(result).toEqual(ok)
49
+ })
50
+
51
+ test('syncFn called exactly once', async () => {
52
+ const { retry } = makeRetry()
53
+ const syncFn = jest.fn().mockResolvedValue(ok)
54
+
55
+ await retry.request(syncFn)
56
+
57
+ expect(syncFn).toHaveBeenCalledTimes(1)
58
+ })
59
+
60
+ test('does not sleep on first attempt', async () => {
61
+ const { retry } = makeRetry()
62
+ await retry.request(jest.fn().mockResolvedValue(ok))
63
+
64
+ expect(retry.sleepSpy).not.toHaveBeenCalled()
65
+ })
66
+
67
+ test('passes attempt number 1 to syncFn', async () => {
68
+ const { retry } = makeRetry()
69
+ const syncFn = jest.fn().mockResolvedValue(ok)
70
+
71
+ await retry.request(syncFn)
72
+
73
+ expect(syncFn).toHaveBeenCalledWith(1)
74
+ })
75
+ })
76
+
77
+ // ---
78
+
79
+ describe('no connectivity', () => {
80
+ test('returns fetch failure without calling syncFn', async () => {
81
+ const { retry } = makeRetry(false)
82
+ const syncFn = jest.fn()
83
+
84
+ const result = await retry.request(syncFn)
85
+
86
+ expect(result.ok).toBe(false)
87
+ expect(syncFn).not.toHaveBeenCalled()
88
+ })
89
+
90
+ test('returned failure is non-retryable', async () => {
91
+ const { retry } = makeRetry(false)
92
+ const result = await retry.request(jest.fn()) as typeof nonRetryable
93
+
94
+ expect(result.isRetryable).toBe(false)
95
+ expect(result.failureType).toBe('fetch')
96
+ })
97
+ })
98
+
99
+ // ---
100
+
101
+ describe('non-retryable failure', () => {
102
+ test('returns after single attempt', async () => {
103
+ const { retry } = makeRetry()
104
+ const syncFn = jest.fn().mockResolvedValue(nonRetryable)
105
+
106
+ const result = await retry.request(syncFn)
107
+
108
+ expect(result).toEqual(nonRetryable)
109
+ expect(syncFn).toHaveBeenCalledTimes(1)
110
+ })
111
+
112
+ test('does not call onFail', async () => {
113
+ const { retry } = makeRetry()
114
+ const onFail = jest.fn()
115
+
116
+ await retry.request(jest.fn().mockResolvedValue(nonRetryable), { onFail })
117
+
118
+ expect(onFail).not.toHaveBeenCalled()
119
+ })
120
+
121
+ test('does not sleep', async () => {
122
+ const { retry } = makeRetry()
123
+ await retry.request(jest.fn().mockResolvedValue(nonRetryable))
124
+
125
+ expect(retry.sleepSpy).not.toHaveBeenCalled()
126
+ })
127
+ })
128
+
129
+ // ---
130
+
131
+ describe('retryable failures', () => {
132
+ test(`retries up to MAX_ATTEMPTS (${SyncRetry.MAX_ATTEMPTS}) times`, async () => {
133
+ const { retry } = makeRetry()
134
+ const syncFn = jest.fn().mockResolvedValue(retryable)
135
+
136
+ await retry.request(syncFn)
137
+
138
+ expect(syncFn).toHaveBeenCalledTimes(SyncRetry.MAX_ATTEMPTS)
139
+ })
140
+
141
+ test('calls onFail when attempts exhausted', async () => {
142
+ const { retry } = makeRetry()
143
+ const onFail = jest.fn()
144
+
145
+ await retry.request(jest.fn().mockResolvedValue(retryable), { onFail })
146
+
147
+ expect(onFail).toHaveBeenCalledTimes(1)
148
+ })
149
+
150
+ test('returns last failure result after exhausting attempts', async () => {
151
+ const { retry } = makeRetry()
152
+ const result = await retry.request(jest.fn().mockResolvedValue(retryable))
153
+
154
+ expect(result).toEqual(retryable)
155
+ })
156
+
157
+ test('passes incrementing attempt numbers to syncFn', async () => {
158
+ const { retry } = makeRetry()
159
+ const syncFn = jest.fn().mockResolvedValue(retryable)
160
+
161
+ await retry.request(syncFn)
162
+
163
+ Array.from({ length: SyncRetry.MAX_ATTEMPTS }, (_, i) => i + 1).forEach((n, i) => {
164
+ expect(syncFn).toHaveBeenNthCalledWith(i + 1, n)
165
+ })
166
+ })
167
+
168
+ test('succeeds after initial failures', async () => {
169
+ const { retry } = makeRetry()
170
+ const syncFn = jest.fn()
171
+ .mockResolvedValueOnce(retryable)
172
+ .mockResolvedValueOnce(retryable)
173
+ .mockResolvedValue(ok)
174
+
175
+ const result = await retry.request(syncFn)
176
+
177
+ expect(result).toEqual(ok)
178
+ expect(syncFn).toHaveBeenCalledTimes(3)
179
+ })
180
+
181
+ test('does not call onFail when success before MAX_ATTEMPTS', async () => {
182
+ const { retry } = makeRetry()
183
+ const onFail = jest.fn()
184
+ const syncFn = jest.fn()
185
+ .mockResolvedValueOnce(retryable)
186
+ .mockResolvedValue(ok)
187
+
188
+ await retry.request(syncFn, { onFail })
189
+
190
+ expect(onFail).not.toHaveBeenCalled()
191
+ })
192
+ })
193
+
194
+ // ---
195
+
196
+ describe('backoff', () => {
197
+ test('sleep called between retries (once before each retry attempt)', async () => {
198
+ const { retry } = makeRetry()
199
+ await retry.request(jest.fn().mockResolvedValue(retryable))
200
+
201
+ expect(retry.sleepSpy).toHaveBeenCalledTimes(SyncRetry.MAX_ATTEMPTS - 1)
202
+ })
203
+
204
+ test('sleep called with positive ms', async () => {
205
+ const { retry } = makeRetry()
206
+ await retry.request(jest.fn().mockResolvedValue(retryable))
207
+
208
+ retry.sleepSpy.mock.calls.forEach(([ms]) => {
209
+ expect(ms).toBeGreaterThan(0)
210
+ })
211
+ })
212
+
213
+ test(`sleep duration does not exceed MAX_BACKOFF (${SyncRetry.MAX_BACKOFF}ms)`, async () => {
214
+ const { retry } = makeRetry()
215
+ await retry.request(jest.fn().mockResolvedValue(retryable))
216
+
217
+ retry.sleepSpy.mock.calls.forEach(([ms]) => {
218
+ expect(ms).toBeLessThanOrEqual(SyncRetry.MAX_BACKOFF)
219
+ })
220
+ })
221
+
222
+ test('backoff resets after success — no sleep on subsequent clean request', async () => {
223
+ const { retry } = makeRetry()
224
+
225
+ const syncFn = jest.fn()
226
+ .mockResolvedValueOnce(retryable)
227
+ .mockResolvedValue(ok)
228
+ await retry.request(syncFn)
229
+
230
+ retry.sleepSpy.mockClear()
231
+
232
+ await retry.request(jest.fn().mockResolvedValue(ok))
233
+ expect(retry.sleepSpy).not.toHaveBeenCalled()
234
+ })
235
+ })
236
+
237
+ // ---
238
+
239
+ describe('edge cases', () => {
240
+ test('connectivity goes offline mid-retry — returns no-connectivity response', async () => {
241
+ const { retry, connectivity } = makeRetry(true)
242
+
243
+ // Online for attempt 1, offline for attempt 2
244
+ connectivity.getValue
245
+ .mockReturnValueOnce(true)
246
+ .mockReturnValue(false)
247
+
248
+ const syncFn = jest.fn().mockResolvedValue(retryable)
249
+ const result = await retry.request(syncFn) as typeof nonRetryable
250
+
251
+ expect(syncFn).toHaveBeenCalledTimes(1)
252
+ expect(result.ok).toBe(false)
253
+ expect(result.isRetryable).toBe(false)
254
+ })
255
+
256
+ test('syncFn throwing propagates the error', async () => {
257
+ const { retry } = makeRetry()
258
+ const boom = new Error('network exploded')
259
+
260
+ await expect(
261
+ retry.request(jest.fn().mockRejectedValue(boom))
262
+ ).rejects.toThrow('network exploded')
263
+ })
264
+
265
+ test('null result returned immediately without retry', async () => {
266
+ const { retry } = makeRetry()
267
+ const syncFn = jest.fn().mockResolvedValue(null)
268
+
269
+ const result = await retry.request(syncFn)
270
+
271
+ expect(result).toBeNull()
272
+ expect(syncFn).toHaveBeenCalledTimes(1)
273
+ })
274
+
275
+ test('concurrent requests share backoff state', async () => {
276
+ const { retry } = makeRetry()
277
+
278
+ // Two concurrent requests both hitting retryable failures
279
+ const [r1, r2] = await Promise.all([
280
+ retry.request(jest.fn().mockResolvedValue(retryable)),
281
+ retry.request(jest.fn().mockResolvedValue(retryable)),
282
+ ])
283
+
284
+ expect(r1.ok).toBe(false)
285
+ expect(r2.ok).toBe(false)
286
+ })
287
+ })
288
+
289
+ // ---
290
+
291
+ describe('start / stop', () => {
292
+ test('start subscribes to connectivity', () => {
293
+ const { retry, connectivity } = makeRetry()
294
+ retry.start()
295
+
296
+ expect(connectivity.subscribe).toHaveBeenCalledTimes(1)
297
+ })
298
+
299
+ test('stop unsubscribes from connectivity', () => {
300
+ const { retry, connectivity } = makeRetry()
301
+ const unsub = jest.fn()
302
+ connectivity.subscribe.mockReturnValue(unsub)
303
+
304
+ retry.start()
305
+ retry.stop()
306
+
307
+ expect(unsub).toHaveBeenCalledTimes(1)
308
+ })
309
+
310
+ test('stop is safe to call before start', () => {
311
+ const { retry } = makeRetry()
312
+ expect(() => retry.stop()).not.toThrow()
313
+ })
314
+ })
@@ -0,0 +1,217 @@
1
+ import { Database } from '@nozbe/watermelondb'
2
+ import { makeTelemetry, makeContext, makePullMock, makePushMock } from '../helpers/index'
3
+ import TestModel, { makeTestDatabase } from '../helpers/TestModel'
4
+ import SyncStore from '@/services/sync/store/index'
5
+ import SyncRetry from '@/services/sync/retry'
6
+ import SyncRunScope from '@/services/sync/run-scope'
7
+ import type { SyncUserScope } from '@/services/sync/index'
8
+ import { SyncError } from '@/services/sync/errors/index'
9
+
10
+ // ---
11
+
12
+ let db: Database
13
+
14
+ function makeStore(userScope: SyncUserScope, pull = makePullMock()) {
15
+ const context = makeContext()
16
+ const telemetry = makeTelemetry(userScope)
17
+ const retry = new SyncRetry(context, telemetry)
18
+ const runScope = new SyncRunScope()
19
+
20
+ return new SyncStore<TestModel>(
21
+ { model: TestModel, pull, push: makePushMock() },
22
+ userScope,
23
+ context,
24
+ db,
25
+ retry,
26
+ runScope,
27
+ telemetry
28
+ )
29
+ }
30
+
31
+ function makeScope(overrides: Partial<SyncUserScope> = {}): SyncUserScope {
32
+ return {
33
+ initialId: 1,
34
+ getCurrentId: () => 1,
35
+ ...overrides,
36
+ }
37
+ }
38
+
39
+ function makePullResponse(overrides: Partial<{ intendedUserId: number }> = {}) {
40
+ return jest.fn().mockResolvedValue({
41
+ ok: true,
42
+ entries: [],
43
+ token: Date.now(),
44
+ previousToken: null,
45
+ intendedUserId: 1,
46
+ ...overrides,
47
+ })
48
+ }
49
+
50
+ beforeEach(() => {
51
+ db = makeTestDatabase()
52
+ })
53
+
54
+ afterEach(async () => {
55
+ await db.write(async () => db.unsafeResetDatabase())
56
+ })
57
+
58
+ // ---
59
+
60
+ describe('write protection (paranoidWrite)', () => {
61
+ describe('same user', () => {
62
+ test('write proceeds when getCurrentId matches initialId', async () => {
63
+ const store = makeStore(makeScope({ initialId: 1, getCurrentId: () => 1 }))
64
+ const result = await store.insertOne(r => { r.value = 'hello' })
65
+ store.destroy()
66
+
67
+ expect(result.value).toBe('hello')
68
+ })
69
+ })
70
+
71
+ describe('different user', () => {
72
+ test('throws SyncError when getCurrentId returns different id', async () => {
73
+ const store = makeStore(makeScope({ initialId: 1, getCurrentId: () => 2 }))
74
+
75
+ await expect(store.insertOne(r => { r.value = 'hello' })).rejects.toThrow(SyncError)
76
+ store.destroy()
77
+ })
78
+
79
+ test('error message indicates cross-user write', async () => {
80
+ const store = makeStore(makeScope({ initialId: 1, getCurrentId: () => 2 }))
81
+
82
+ await expect(store.insertOne(r => { r.value = 'hello' })).rejects.toThrow(
83
+ 'Aborted cross-user write operation'
84
+ )
85
+ store.destroy()
86
+ })
87
+ })
88
+
89
+ describe('null getCurrentId without fetchCurrentId', () => {
90
+ test('throws SyncError', async () => {
91
+ const store = makeStore(makeScope({ initialId: 1, getCurrentId: () => null }))
92
+
93
+ await expect(store.insertOne(r => { r.value = 'hello' })).rejects.toThrow(SyncError)
94
+ store.destroy()
95
+ })
96
+
97
+ test('error message indicates cross-user write', async () => {
98
+ const store = makeStore(makeScope({ initialId: 1, getCurrentId: () => null }))
99
+
100
+ await expect(store.insertOne(r => { r.value = 'hello' })).rejects.toThrow(
101
+ 'Aborted cross-user write operation'
102
+ )
103
+ store.destroy()
104
+ })
105
+ })
106
+
107
+ describe('fetchCurrentId fallback', () => {
108
+ test('fetchCurrentId is called when getCurrentId returns null', async () => {
109
+ const fetchCurrentId = jest.fn().mockResolvedValue(1)
110
+ const store = makeStore(makeScope({ initialId: 1, getCurrentId: () => null, fetchCurrentId }))
111
+
112
+ await store.insertOne(r => { r.value = 'hello' })
113
+ store.destroy()
114
+
115
+ expect(fetchCurrentId).toHaveBeenCalledTimes(1)
116
+ })
117
+
118
+ test('write proceeds when fetchCurrentId resolves to same id', async () => {
119
+ const store = makeStore(makeScope({
120
+ initialId: 1,
121
+ getCurrentId: () => null,
122
+ fetchCurrentId: () => Promise.resolve(1),
123
+ }))
124
+
125
+ const result = await store.insertOne(r => { r.value = 'resolved' })
126
+ store.destroy()
127
+
128
+ expect(result.value).toBe('resolved')
129
+ })
130
+
131
+ test('throws when fetchCurrentId resolves to different id', async () => {
132
+ const store = makeStore(makeScope({
133
+ initialId: 1,
134
+ getCurrentId: () => null,
135
+ fetchCurrentId: () => Promise.resolve(2),
136
+ }))
137
+
138
+ await expect(store.insertOne(r => { r.value = 'hello' })).rejects.toThrow(
139
+ 'Aborted cross-user write operation'
140
+ )
141
+ store.destroy()
142
+ })
143
+
144
+ test('throws SyncError when fetchCurrentId rejects', async () => {
145
+ const store = makeStore(makeScope({
146
+ initialId: 1,
147
+ getCurrentId: () => null,
148
+ fetchCurrentId: () => Promise.reject(new Error('network error')),
149
+ }))
150
+
151
+ await expect(store.insertOne(r => { r.value = 'hello' })).rejects.toBeInstanceOf(SyncError)
152
+ store.destroy()
153
+ })
154
+
155
+ test('error message indicates fetchCurrentId failed', async () => {
156
+ const store = makeStore(makeScope({
157
+ initialId: 1,
158
+ getCurrentId: () => null,
159
+ fetchCurrentId: () => Promise.reject(new Error('network error')),
160
+ }))
161
+
162
+ await expect(store.insertOne(r => { r.value = 'hello' })).rejects.toThrow(
163
+ 'Aborted cross-user write operation after fetchCurrentId failed'
164
+ )
165
+ store.destroy()
166
+ })
167
+ })
168
+ })
169
+
170
+ // ---
171
+
172
+ describe('pull protection (intendedUserId)', () => {
173
+ test('pull succeeds when intendedUserId matches initialId', async () => {
174
+ const store = makeStore(makeScope({ initialId: 1 }), makePullResponse({ intendedUserId: 1 }))
175
+
176
+ await expect(store.pull('test')).resolves.not.toThrow()
177
+ store.destroy()
178
+ })
179
+
180
+ test('throws SyncError when intendedUserId does not match initialId', async () => {
181
+ const store = makeStore(makeScope({ initialId: 1 }), makePullResponse({ intendedUserId: 2 }))
182
+
183
+ await expect(store.pull('test')).rejects.toThrow(SyncError)
184
+ store.destroy()
185
+ })
186
+
187
+ test('throws when intendedUserId does not match currentId', async () => {
188
+ const store = makeStore(
189
+ makeScope({ initialId: 1, getCurrentId: () => 2 }),
190
+ makePullResponse({ intendedUserId: 1 })
191
+ )
192
+
193
+ await expect(store.pull('test')).rejects.toThrow(SyncError)
194
+ store.destroy()
195
+ })
196
+
197
+ test('error message indicates intended user mismatch', async () => {
198
+ const store = makeStore(makeScope({ initialId: 1 }), makePullResponse({ intendedUserId: 2 }))
199
+
200
+ await expect(store.pull('test')).rejects.toThrow('Intended user ID does not match')
201
+ store.destroy()
202
+ })
203
+
204
+ test('throws SyncError when getCurrentId is null and fetchCurrentId rejects during pull', async () => {
205
+ const store = makeStore(
206
+ makeScope({
207
+ initialId: 1,
208
+ getCurrentId: () => null,
209
+ fetchCurrentId: () => Promise.reject(new Error('auth error')),
210
+ }),
211
+ makePullResponse({ intendedUserId: 1 })
212
+ )
213
+
214
+ await expect(store.pull('test')).rejects.toThrow(SyncError)
215
+ store.destroy()
216
+ })
217
+ })
@@ -0,0 +1,156 @@
1
+ import PushCoalescer from '@/services/sync/store/push-coalescer'
2
+ import type BaseModel from '@/services/sync/models/Base'
3
+ import type { SyncPushResponse } from '@/services/sync/fetch'
4
+
5
+ const BASE_TIME = 1700000000000
6
+ const SUCCESS_RESPONSE: SyncPushResponse = { ok: true, results: [] }
7
+
8
+ class TestSetup {
9
+ coalescer = new PushCoalescer()
10
+
11
+ record(id: string, timeOffset = 0): BaseModel {
12
+ return { id, updated_at: BASE_TIME + timeOffset } as unknown as BaseModel
13
+ }
14
+
15
+ mockPusher() {
16
+ return jest.fn().mockReturnValue(new Promise(() => {}))
17
+ }
18
+
19
+ controllablePusher() {
20
+ let resolve!: (value: SyncPushResponse) => void
21
+ let reject!: (error: Error) => void
22
+
23
+ const promise = new Promise<SyncPushResponse>((res, rej) => {
24
+ resolve = res
25
+ reject = rej
26
+ })
27
+
28
+ return {
29
+ pusher: jest.fn().mockReturnValue(promise),
30
+ resolve: () => resolve(SUCCESS_RESPONSE),
31
+ reject: (error: Error) => reject(error)
32
+ }
33
+ }
34
+ }
35
+
36
+ let setup: TestSetup
37
+ beforeEach(() => {
38
+ setup = new TestSetup()
39
+ })
40
+
41
+ describe('coalescing behavior', () => {
42
+ describe('when to coalesce', () => {
43
+ test('identical records return same promise', () => {
44
+ const records = [setup.record('rec-1')]
45
+ const pusher = setup.mockPusher()
46
+
47
+ const p1 = setup.coalescer.push(records, pusher)
48
+ const p2 = setup.coalescer.push(records, pusher)
49
+
50
+ expect(p1).toBe(p2)
51
+ expect(pusher).toHaveBeenCalledTimes(1)
52
+ })
53
+
54
+ test('same timestamp coalesces', () => {
55
+ const pusher = setup.mockPusher()
56
+ const p1 = setup.coalescer.push([setup.record('rec-1')], pusher)
57
+ const p2 = setup.coalescer.push([setup.record('rec-1')], pusher)
58
+
59
+ expect(p1).toBe(p2)
60
+ expect(pusher).toHaveBeenCalledTimes(1)
61
+ })
62
+
63
+ test('all records must match for multi-record coalescing', () => {
64
+ const pusher = setup.mockPusher()
65
+ const records = [setup.record('rec-1'), setup.record('rec-2')]
66
+
67
+ const p1 = setup.coalescer.push(records, pusher)
68
+ const p2 = setup.coalescer.push([setup.record('rec-1'), setup.record('rec-2')], pusher)
69
+
70
+ expect(p1).toBe(p2)
71
+ expect(pusher).toHaveBeenCalledTimes(1)
72
+ })
73
+ })
74
+
75
+ describe('when NOT to coalesce', () => {
76
+ test('newer timestamp triggers new push', () => {
77
+ const pusher = setup.mockPusher()
78
+
79
+ setup.coalescer.push([setup.record('rec-1')], pusher)
80
+ setup.coalescer.push([setup.record('rec-1', 1)], pusher) // +1ms newer
81
+
82
+ expect(pusher).toHaveBeenCalledTimes(2)
83
+ })
84
+
85
+ test('different record IDs trigger separate pushes', () => {
86
+ const pusher = setup.mockPusher()
87
+
88
+ setup.coalescer.push([setup.record('rec-1')], pusher)
89
+ setup.coalescer.push([setup.record('rec-2')], pusher)
90
+
91
+ expect(pusher).toHaveBeenCalledTimes(2)
92
+ })
93
+
94
+ test('partial record match prevents coalescing', () => {
95
+ const pusher = setup.mockPusher()
96
+
97
+ setup.coalescer.push([setup.record('rec-1'), setup.record('rec-2')], pusher)
98
+ setup.coalescer.push([setup.record('rec-1'), setup.record('rec-2', 1)], pusher) // rec-2 newer
99
+
100
+ expect(pusher).toHaveBeenCalledTimes(2)
101
+ })
102
+ })
103
+ })
104
+
105
+ describe('intent lifecycle', () => {
106
+ test('cleans up after successful push', async () => {
107
+ const { pusher, resolve } = setup.controllablePusher()
108
+ const records = [setup.record('rec-1')]
109
+
110
+ setup.coalescer.push(records, pusher)
111
+ resolve()
112
+ await Promise.resolve() // flush cleanup
113
+
114
+ // Should trigger new push since intent was cleaned up
115
+ const newPusher = setup.mockPusher()
116
+ setup.coalescer.push(records, newPusher)
117
+ expect(newPusher).toHaveBeenCalled()
118
+ })
119
+
120
+ test('cleans up after failed push', async () => {
121
+ const { pusher, reject } = setup.controllablePusher()
122
+ const records = [setup.record('rec-1')]
123
+
124
+ const promise = setup.coalescer.push(records, pusher)
125
+ promise.catch(() => {}) // prevent unhandled rejection
126
+
127
+ reject(new Error('push failed'))
128
+ await Promise.resolve()
129
+
130
+ const newPusher = setup.mockPusher()
131
+ setup.coalescer.push(records, newPusher)
132
+ expect(newPusher).toHaveBeenCalled()
133
+ })
134
+
135
+ test('coalesces while intent is in-flight', () => {
136
+ const { pusher } = setup.controllablePusher()
137
+ const records = [setup.record('rec-1')]
138
+
139
+ const p1 = setup.coalescer.push(records, pusher)
140
+ const p2 = setup.coalescer.push(records, setup.mockPusher())
141
+
142
+ expect(p1).toBe(p2) // Same promise returned
143
+ })
144
+ })
145
+
146
+ describe('edge cases', () => {
147
+ test('empty record arrays coalesce', () => {
148
+ const pusher = setup.mockPusher()
149
+
150
+ const p1 = setup.coalescer.push([], pusher)
151
+ const p2 = setup.coalescer.push([], pusher)
152
+
153
+ expect(p1).toBe(p2)
154
+ expect(pusher).toHaveBeenCalledTimes(1)
155
+ })
156
+ })
@@ -17,10 +17,10 @@
17
17
  import { IDBFactory } from 'fake-indexeddb'
18
18
  import { makeTelemetry, makeContext, makeUserScope, makePullMock, makePushMock, TEST_USER_ID } from '../helpers/index'
19
19
  import TestModel, { makeTestDatabase } from '../helpers/TestModel'
20
- import SyncStore from '../../../../src/services/sync/store/index'
21
- import SyncRetry from '../../../../src/services/sync/retry'
22
- import SyncRunScope from '../../../../src/services/sync/run-scope'
23
- import type { SyncPullResponse } from '../../../../src/services/sync/fetch'
20
+ import SyncStore from '@/services/sync/store/index'
21
+ import SyncRetry from '@/services/sync/retry'
22
+ import SyncRunScope from '@/services/sync/run-scope'
23
+ import type { SyncPullResponse } from '@/services/sync/fetch'
24
24
 
25
25
  // Fresh IDB instance per test — prevents cross-test transaction leakage
26
26
  beforeEach(() => {