|
1 | | -import { describe, test, expect, beforeEach, vi, afterEach } from 'vitest'; |
2 | | -import { sdkNavigator } from '../../src/shared/navigator'; |
3 | | - |
4 | | -describe('sdkNavigator', () => { |
5 | | - let originalNavigator: Navigator; |
6 | | - |
7 | | - beforeEach(() => { |
8 | | - originalNavigator = global.navigator; |
9 | | - vi.stubGlobal('navigator', { |
10 | | - ...originalNavigator, |
11 | | - locks: { |
12 | | - request: vi.fn(), |
13 | | - }, |
14 | | - }); |
15 | | - }); |
| 1 | +import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest'; |
| 2 | +import { getNavigationLocks } from '../../src/shared/navigator'; |
16 | 3 |
|
| 4 | +describe('getNavigationLocks', () => { |
17 | 5 | afterEach(() => { |
18 | | - vi.stubGlobal('navigator', originalNavigator); |
| 6 | + vi.restoreAllMocks(); |
19 | 7 | }); |
20 | 8 |
|
21 | | - test('should inherit properties from Navigator', () => { |
22 | | - expect(sdkNavigator.userAgent).toBe(navigator.userAgent); |
| 9 | + it('should return native navigator.locks if available', () => { |
| 10 | + const mockLocks = { |
| 11 | + request: vi.fn(), |
| 12 | + query: vi.fn(), |
| 13 | + }; |
| 14 | + |
| 15 | + vi.spyOn(navigator, 'locks', 'get').mockReturnValue(mockLocks); |
| 16 | + |
| 17 | + const result = getNavigationLocks(); |
| 18 | + expect(result).toBe(mockLocks); |
23 | 19 | }); |
24 | 20 |
|
25 | | - test('should have locks property', () => { |
26 | | - expect(sdkNavigator.locks).toBeDefined(); |
27 | | - expect(typeof sdkNavigator.locks.request).toBe('function'); |
| 21 | + it('should return fallback implementation if navigator.locks is not available', () => { |
| 22 | + // @ts-ignore |
| 23 | + vi.spyOn(navigator, 'locks', 'get').mockReturnValue(undefined); |
| 24 | + |
| 25 | + const result = getNavigationLocks(); |
| 26 | + expect(result).toHaveProperty('request'); |
| 27 | + expect(result).toHaveProperty('query'); |
| 28 | + expect(result).not.toBe(navigator.locks); |
28 | 29 | }); |
29 | 30 |
|
30 | | - test('should throw error when locks are not available', () => { |
31 | | - vi.stubGlobal('navigator', { ...originalNavigator, locks: undefined }); |
32 | | - expect(() => sdkNavigator.locks).toThrowError('Navigator locks are not available in this context.'); |
| 31 | + it('fallback request should acquire and release a lock', async () => { |
| 32 | + // @ts-ignore |
| 33 | + vi.spyOn(navigator, 'locks', 'get').mockReturnValue(undefined); |
| 34 | + const locks = getNavigationLocks(); |
| 35 | + |
| 36 | + const mockCallback = vi.fn().mockResolvedValue('result'); |
| 37 | + const result = await locks.request('test-lock', mockCallback); |
| 38 | + |
| 39 | + expect(mockCallback).toHaveBeenCalledWith(expect.objectContaining({ |
| 40 | + name: 'test-lock', |
| 41 | + mode: 'exclusive' |
| 42 | + })); |
| 43 | + expect(result).toBe('result'); |
33 | 44 | }); |
34 | 45 |
|
35 | | - test('locks proxy should pass through method calls when locks are available', () => { |
36 | | - const mockCallback = vi.fn(); |
37 | | - sdkNavigator.locks.request('test', mockCallback); |
38 | | - expect(navigator.locks.request).toHaveBeenCalledWith('test', mockCallback); |
| 46 | + it('fallback query should return held locks', async () => { |
| 47 | + // @ts-ignore |
| 48 | + vi.spyOn(navigator, 'locks', 'get').mockReturnValue(undefined); |
| 49 | + const locks = getNavigationLocks(); |
| 50 | + |
| 51 | + // Acquire a lock first |
| 52 | + await locks.request('test-lock', async () => { |
| 53 | + const queryResult = await locks.query(); |
| 54 | + expect(queryResult.held).toHaveLength(1); |
| 55 | + expect(queryResult.held![0]).toEqual(expect.objectContaining({ |
| 56 | + name: 'test-lock', |
| 57 | + mode: 'exclusive' |
| 58 | + })); |
| 59 | + expect(queryResult.pending).toHaveLength(0); |
| 60 | + }); |
| 61 | + |
| 62 | + const finalQueryResult = await locks.query(); |
| 63 | + expect(finalQueryResult.held).toHaveLength(0); |
39 | 64 | }); |
40 | 65 |
|
41 | | - test('should only expose expected Navigator properties', () => { |
42 | | - const sdkNavigatorKeys = Object.keys(sdkNavigator); |
43 | | - const navigatorKeys = Object.keys(navigator); |
44 | | - sdkNavigatorKeys.forEach(key => { |
45 | | - expect(navigatorKeys).toContain(key); |
| 66 | + it('fallback implementation should handle concurrent requests', async () => { |
| 67 | + // @ts-ignore |
| 68 | + vi.spyOn(navigator, 'locks', 'get').mockReturnValue(undefined); |
| 69 | + const locks = getNavigationLocks(); |
| 70 | + |
| 71 | + const delay = (ms: number) => new Promise(resolve => setTimeout(resolve, ms)); |
| 72 | + |
| 73 | + const request1 = locks.request('test-lock', async () => { |
| 74 | + await delay(200); |
| 75 | + return 'first'; |
46 | 76 | }); |
| 77 | + |
| 78 | + const request2 = locks.request('test-lock', async () => { |
| 79 | + return 'second'; |
| 80 | + }); |
| 81 | + |
| 82 | + const [result1, result2] = await Promise.all([request1, request2]); |
| 83 | + |
| 84 | + expect(result1).toBe('first'); |
| 85 | + expect(result2).toBe('second'); |
47 | 86 | }); |
48 | 87 | }); |
0 commit comments