@sanity/sdk 2.8.0 → 2.10.0

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 (111) hide show
  1. package/dist/_chunks-dts/utils.d.ts +2450 -0
  2. package/dist/_chunks-es/_internal.js +129 -0
  3. package/dist/_chunks-es/_internal.js.map +1 -0
  4. package/dist/_chunks-es/createGroqSearchFilter.js +1537 -0
  5. package/dist/_chunks-es/createGroqSearchFilter.js.map +1 -0
  6. package/dist/_chunks-es/telemetryManager.js +87 -0
  7. package/dist/_chunks-es/telemetryManager.js.map +1 -0
  8. package/dist/_chunks-es/version.js +7 -0
  9. package/dist/_chunks-es/version.js.map +1 -0
  10. package/dist/_exports/_internal.d.ts +64 -0
  11. package/dist/_exports/_internal.js +20 -0
  12. package/dist/_exports/_internal.js.map +1 -0
  13. package/dist/index.d.ts +2 -2343
  14. package/dist/index.js +465 -1813
  15. package/dist/index.js.map +1 -1
  16. package/package.json +17 -12
  17. package/src/_exports/_internal.ts +14 -0
  18. package/src/_exports/index.ts +18 -1
  19. package/src/auth/authStore.test.ts +150 -1
  20. package/src/auth/authStore.ts +11 -11
  21. package/src/auth/dashboardAuth.ts +2 -2
  22. package/src/auth/handleAuthCallback.ts +9 -3
  23. package/src/auth/logout.test.ts +1 -1
  24. package/src/auth/logout.ts +1 -1
  25. package/src/auth/refreshStampedToken.test.ts +118 -1
  26. package/src/auth/refreshStampedToken.ts +3 -2
  27. package/src/auth/standaloneAuth.ts +9 -3
  28. package/src/auth/studioAuth.ts +34 -7
  29. package/src/auth/studioModeAuth.ts +2 -1
  30. package/src/auth/subscribeToStateAndFetchCurrentUser.test.ts +10 -2
  31. package/src/auth/subscribeToStateAndFetchCurrentUser.ts +5 -1
  32. package/src/auth/subscribeToStorageEventsAndSetToken.ts +2 -2
  33. package/src/auth/utils.ts +33 -0
  34. package/src/client/clientStore.test.ts +44 -30
  35. package/src/client/clientStore.ts +49 -48
  36. package/src/comlink/controller/actions/getOrCreateChannel.ts +2 -2
  37. package/src/comlink/node/actions/getOrCreateNode.ts +2 -2
  38. package/src/comlink/node/getNodeState.ts +2 -1
  39. package/src/config/sanityConfig.ts +78 -12
  40. package/src/document/actions.ts +18 -11
  41. package/src/document/applyDocumentActions.test.ts +7 -6
  42. package/src/document/applyDocumentActions.ts +10 -4
  43. package/src/document/documentStore.test.ts +542 -188
  44. package/src/document/documentStore.ts +142 -76
  45. package/src/document/events.ts +7 -2
  46. package/src/document/permissions.test.ts +18 -16
  47. package/src/document/permissions.ts +35 -11
  48. package/src/document/processActions.test.ts +359 -32
  49. package/src/document/processActions.ts +106 -78
  50. package/src/document/reducers.test.ts +117 -29
  51. package/src/document/reducers.ts +47 -40
  52. package/src/document/sharedListener.ts +16 -6
  53. package/src/document/util.ts +14 -0
  54. package/src/favorites/favorites.test.ts +9 -2
  55. package/src/presence/bifurTransport.test.ts +46 -6
  56. package/src/presence/bifurTransport.ts +19 -2
  57. package/src/presence/presenceStore.test.ts +96 -0
  58. package/src/presence/presenceStore.ts +96 -24
  59. package/src/preview/getPreviewState.test.ts +115 -98
  60. package/src/preview/getPreviewState.ts +38 -60
  61. package/src/preview/previewProjectionUtils.test.ts +179 -0
  62. package/src/preview/previewProjectionUtils.ts +93 -0
  63. package/src/preview/resolvePreview.test.ts +42 -25
  64. package/src/preview/resolvePreview.ts +33 -10
  65. package/src/preview/{previewStore.ts → types.ts} +8 -17
  66. package/src/projection/getProjectionState.test.ts +16 -16
  67. package/src/projection/getProjectionState.ts +6 -5
  68. package/src/projection/projectionQuery.ts +2 -3
  69. package/src/projection/projectionStore.test.ts +2 -2
  70. package/src/projection/resolveProjection.ts +2 -2
  71. package/src/projection/subscribeToStateAndFetchBatches.test.ts +1 -1
  72. package/src/projection/subscribeToStateAndFetchBatches.ts +12 -11
  73. package/src/projection/types.ts +1 -1
  74. package/src/query/queryStore.test.ts +12 -12
  75. package/src/query/queryStore.ts +12 -11
  76. package/src/query/reducers.ts +3 -3
  77. package/src/releases/getPerspectiveState.ts +7 -6
  78. package/src/releases/releasesStore.test.ts +20 -5
  79. package/src/releases/releasesStore.ts +20 -8
  80. package/src/store/createActionBinder.test.ts +31 -31
  81. package/src/store/createActionBinder.ts +43 -38
  82. package/src/store/createSanityInstance.ts +2 -3
  83. package/src/store/createStateSourceAction.test.ts +62 -0
  84. package/src/store/createStateSourceAction.ts +34 -39
  85. package/src/telemetry/__telemetry__/sdk.telemetry.ts +42 -0
  86. package/src/telemetry/devMode.test.ts +52 -0
  87. package/src/telemetry/devMode.ts +40 -0
  88. package/src/telemetry/initTelemetry.test.ts +225 -0
  89. package/src/telemetry/initTelemetry.ts +205 -0
  90. package/src/telemetry/telemetryManager.test.ts +263 -0
  91. package/src/telemetry/telemetryManager.ts +187 -0
  92. package/src/users/reducers.ts +3 -4
  93. package/src/users/usersStore.test.ts +1 -0
  94. package/src/users/usersStore.ts +5 -1
  95. package/src/utils/createFetcherStore.test.ts +6 -4
  96. package/src/utils/createFetcherStore.ts +8 -5
  97. package/src/utils/getStagingApiHost.test.ts +21 -0
  98. package/src/utils/getStagingApiHost.ts +14 -0
  99. package/src/utils/ids.test.ts +1 -29
  100. package/src/utils/ids.ts +0 -10
  101. package/src/utils/isImportError.test.ts +72 -0
  102. package/src/utils/isImportError.ts +34 -0
  103. package/src/utils/object.test.ts +95 -0
  104. package/src/utils/object.ts +142 -0
  105. package/src/utils/setCleanupTimeout.ts +24 -0
  106. package/src/preview/previewQuery.test.ts +0 -236
  107. package/src/preview/previewQuery.ts +0 -153
  108. package/src/preview/previewStore.test.ts +0 -36
  109. package/src/preview/subscribeToStateAndFetchBatches.test.ts +0 -221
  110. package/src/preview/subscribeToStateAndFetchBatches.ts +0 -112
  111. package/src/preview/util.ts +0 -13
@@ -0,0 +1,225 @@
1
+ import {afterEach, beforeEach, describe, expect, it, vi} from 'vitest'
2
+
3
+ import {createSanityInstance} from '../store/createSanityInstance'
4
+ import {isDevMode} from './devMode'
5
+ import {getTelemetryManager, initTelemetry, trackHookMounted} from './initTelemetry'
6
+ import {createTelemetryManager} from './telemetryManager'
7
+
8
+ vi.mock('./devMode', () => ({
9
+ isDevMode: vi.fn(() => false),
10
+ }))
11
+
12
+ vi.mock('./telemetryManager', () => ({
13
+ createTelemetryManager: vi.fn(() => ({
14
+ checkConsent: vi.fn(() => Promise.resolve(true)),
15
+ logSessionStarted: vi.fn(),
16
+ logHookFirstUsed: vi.fn(),
17
+ logDevError: vi.fn(),
18
+ endSession: vi.fn(),
19
+ dispose: vi.fn(),
20
+ hooksUsed: new Set(),
21
+ })),
22
+ }))
23
+
24
+ vi.mock('../client/clientStore', () => ({
25
+ getClient: vi.fn(() => ({})),
26
+ }))
27
+
28
+ vi.mock('../auth/authStore', () => ({
29
+ getTokenState: vi.fn(() => ({
30
+ getCurrent: vi.fn(() => 'mock-token'),
31
+ observable: {subscribe: vi.fn()},
32
+ })),
33
+ }))
34
+
35
+ /**
36
+ * Flush the microtask queue so the dynamic imports in initTelemetry
37
+ * have time to resolve before assertions run.
38
+ */
39
+ const flushPromises = () => new Promise<void>((r) => setTimeout(r, 0))
40
+
41
+ describe('initTelemetry', () => {
42
+ beforeEach(() => {
43
+ vi.clearAllMocks()
44
+ })
45
+
46
+ afterEach(() => {
47
+ vi.restoreAllMocks()
48
+ })
49
+
50
+ it('does nothing when dev mode is disabled', async () => {
51
+ vi.mocked(isDevMode).mockReturnValue(false)
52
+
53
+ const instance = createSanityInstance()
54
+
55
+ initTelemetry(instance, 'abc123')
56
+ await flushPromises()
57
+
58
+ expect(createTelemetryManager).not.toHaveBeenCalled()
59
+ instance.dispose()
60
+ })
61
+
62
+ it('does nothing when no projectId is provided', async () => {
63
+ vi.mocked(isDevMode).mockReturnValue(true)
64
+
65
+ const instance = createSanityInstance()
66
+ initTelemetry(instance, '')
67
+ await flushPromises()
68
+
69
+ expect(createTelemetryManager).not.toHaveBeenCalled()
70
+ instance.dispose()
71
+ })
72
+
73
+ it('initializes telemetry in dev mode with a projectId', async () => {
74
+ vi.mocked(isDevMode).mockReturnValue(true)
75
+
76
+ const instance = createSanityInstance()
77
+
78
+ initTelemetry(instance, 'abc123')
79
+ await flushPromises()
80
+
81
+ expect(createTelemetryManager).toHaveBeenCalledWith(
82
+ expect.objectContaining({
83
+ sessionId: instance.instanceId,
84
+ projectId: 'abc123',
85
+ }),
86
+ )
87
+
88
+ const manager = vi.mocked(createTelemetryManager).mock.results[0].value
89
+ expect(manager.logSessionStarted).toHaveBeenCalledWith(
90
+ expect.objectContaining({
91
+ projectId: 'abc123',
92
+ perspective: 'published',
93
+ }),
94
+ )
95
+
96
+ instance.dispose()
97
+ })
98
+
99
+ it('registers manager in the WeakMap', async () => {
100
+ vi.mocked(isDevMode).mockReturnValue(true)
101
+
102
+ const instance = createSanityInstance()
103
+
104
+ initTelemetry(instance, 'abc123')
105
+ await flushPromises()
106
+
107
+ expect(getTelemetryManager(instance)).toBeDefined()
108
+
109
+ instance.dispose()
110
+ })
111
+
112
+ it('does not initialize if instance is already disposed', async () => {
113
+ vi.mocked(isDevMode).mockReturnValue(true)
114
+
115
+ const instance = createSanityInstance()
116
+
117
+ instance.dispose()
118
+ initTelemetry(instance, 'abc123')
119
+ await flushPromises()
120
+
121
+ const manager = vi.mocked(createTelemetryManager).mock.results[0]?.value
122
+ if (manager) {
123
+ expect(manager.logSessionStarted).not.toHaveBeenCalled()
124
+ }
125
+ })
126
+
127
+ it('calls endSession and removes manager on instance dispose', async () => {
128
+ vi.mocked(isDevMode).mockReturnValue(true)
129
+
130
+ const instance = createSanityInstance()
131
+
132
+ initTelemetry(instance, 'abc123')
133
+ await flushPromises()
134
+
135
+ const manager = vi.mocked(createTelemetryManager).mock.results[0].value
136
+ expect(getTelemetryManager(instance)).toBeDefined()
137
+
138
+ instance.dispose()
139
+
140
+ expect(manager.endSession).toHaveBeenCalled()
141
+ expect(getTelemetryManager(instance)).toBeUndefined()
142
+ })
143
+
144
+ it('skips telemetry entirely when user has not opted in', async () => {
145
+ vi.mocked(isDevMode).mockReturnValue(true)
146
+
147
+ const instance = createSanityInstance()
148
+
149
+ vi.mocked(createTelemetryManager).mockReturnValueOnce({
150
+ checkConsent: vi.fn(() => Promise.resolve(false)),
151
+ logSessionStarted: vi.fn(),
152
+ logHookFirstUsed: vi.fn(),
153
+ logDevError: vi.fn(),
154
+ endSession: vi.fn(),
155
+ dispose: vi.fn(),
156
+ hooksUsed: new Set(),
157
+ })
158
+
159
+ initTelemetry(instance, 'abc123')
160
+ await flushPromises()
161
+
162
+ expect(createTelemetryManager).toHaveBeenCalled()
163
+ const manager = vi.mocked(createTelemetryManager).mock.results[0].value
164
+
165
+ expect(manager.logSessionStarted).not.toHaveBeenCalled()
166
+ expect(manager.dispose).toHaveBeenCalled()
167
+ expect(manager.endSession).not.toHaveBeenCalled()
168
+ expect(getTelemetryManager(instance)).toBeUndefined()
169
+
170
+ instance.dispose()
171
+ })
172
+
173
+ it('uses perspective from config when available', async () => {
174
+ vi.mocked(isDevMode).mockReturnValue(true)
175
+
176
+ const instance = createSanityInstance({perspective: 'previewDrafts'})
177
+
178
+ initTelemetry(instance, 'abc123')
179
+ await flushPromises()
180
+
181
+ const manager = vi.mocked(createTelemetryManager).mock.results[0].value
182
+ expect(manager.logSessionStarted).toHaveBeenCalledWith(
183
+ expect.objectContaining({
184
+ perspective: 'previewDrafts',
185
+ }),
186
+ )
187
+
188
+ instance.dispose()
189
+ })
190
+
191
+ it('flushes hooks buffered before manager is ready', async () => {
192
+ vi.mocked(isDevMode).mockReturnValue(true)
193
+
194
+ const instance = createSanityInstance()
195
+
196
+ trackHookMounted(instance, 'useQuery')
197
+ trackHookMounted(instance, 'useDocument')
198
+
199
+ initTelemetry(instance, 'abc123')
200
+ await flushPromises()
201
+
202
+ const manager = vi.mocked(createTelemetryManager).mock.results[0].value
203
+ expect(manager.logHookFirstUsed).toHaveBeenCalledWith('useQuery')
204
+ expect(manager.logHookFirstUsed).toHaveBeenCalledWith('useDocument')
205
+
206
+ instance.dispose()
207
+ })
208
+
209
+ it('finds manager through parent-child instance chain', async () => {
210
+ vi.mocked(isDevMode).mockReturnValue(true)
211
+
212
+ const root = createSanityInstance()
213
+ const child = root.createChild({})
214
+
215
+ initTelemetry(root, 'abc123')
216
+ await flushPromises()
217
+
218
+ trackHookMounted(child, 'useUsers')
219
+
220
+ const manager = vi.mocked(createTelemetryManager).mock.results[0].value
221
+ expect(manager.logHookFirstUsed).toHaveBeenCalledWith('useUsers')
222
+
223
+ root.dispose()
224
+ })
225
+ })
@@ -0,0 +1,205 @@
1
+ import {type SanityInstance} from '../store/createSanityInstance'
2
+ import {createLogger} from '../utils/logger'
3
+ import {isDevMode} from './devMode'
4
+ import {type TelemetryManager} from './telemetryManager'
5
+
6
+ const DEFAULT_TELEMETRY_API_VERSION = '2024-11-12'
7
+
8
+ const logger = createLogger('telemetry')
9
+
10
+ /**
11
+ * Per-instance map of active telemetry managers. Allows the React
12
+ * package (and other consumers) to look up the manager for a given
13
+ * instance and log hook usage / errors without importing the full
14
+ * telemetry module themselves.
15
+ *
16
+ * @internal
17
+ */
18
+ const telemetryManagers = new WeakMap<SanityInstance, TelemetryManager>()
19
+ const pendingHooks = new WeakMap<SanityInstance, Set<string>>()
20
+ const initInFlight = new WeakSet<SanityInstance>()
21
+
22
+ /**
23
+ * Initializes dev-mode telemetry for a SDK instance if the environment
24
+ * qualifies. Both `telemetryManager` and `clientStore` are dynamically
25
+ * imported to avoid circular dependencies and to keep telemetry code
26
+ * out of production bundles via code splitting.
27
+ *
28
+ * The `projectId` must be passed explicitly because the resource
29
+ * configuration is typically set by the React layer after the
30
+ * instance has already been created.
31
+ *
32
+ * @internal
33
+ */
34
+ export function initTelemetry(instance: SanityInstance, projectId: string): void {
35
+ if (!isDevMode()) {
36
+ logger.trace('initTelemetry skipped: not dev mode', {internal: true})
37
+ return
38
+ }
39
+ if (!projectId) {
40
+ logger.trace('initTelemetry skipped: no projectId', {internal: true})
41
+ return
42
+ }
43
+ if (telemetryManagers.has(instance) || initInFlight.has(instance)) {
44
+ return
45
+ }
46
+ initInFlight.add(instance)
47
+
48
+ logger.debug('initializing telemetry', {projectId})
49
+
50
+ Promise.all([
51
+ import('./telemetryManager'),
52
+ import('../client/clientStore'),
53
+ import('../auth/authStore'),
54
+ ])
55
+ .then(async ([{createTelemetryManager}, {getClient}, {getTokenState}]) => {
56
+ if (instance.isDisposed()) {
57
+ initInFlight.delete(instance)
58
+ logger.debug('telemetry skipped: instance disposed before imports resolved')
59
+ return
60
+ }
61
+
62
+ // Wait for the auth store to resolve a token. The client needs a
63
+ // Bearer token for the consent check and batch POSTs. If the token
64
+ // is already available (e.g. static token config), this resolves
65
+ // immediately. For OAuth/localStorage discovery it waits for the
66
+ // first emission. For unauthenticated apps the promise never
67
+ // resolves, which is fine since telemetry requires auth anyway.
68
+ const token = getTokenState(instance).getCurrent()
69
+ logger.trace('auth token check', {tokenPresent: !!token, internal: true})
70
+
71
+ if (!token) {
72
+ logger.debug('waiting for auth token')
73
+ const hasToken = await new Promise<boolean>((resolve) => {
74
+ if (instance.isDisposed()) return resolve(false)
75
+ const cleanup = {unsubscribe: () => {}}
76
+ const unsub = instance.onDispose(() => {
77
+ cleanup.unsubscribe()
78
+ resolve(false)
79
+ })
80
+ const sub = getTokenState(instance).observable.subscribe((t) => {
81
+ if (t) {
82
+ logger.debug('auth token received')
83
+ sub.unsubscribe()
84
+ unsub()
85
+ resolve(true)
86
+ }
87
+ })
88
+ cleanup.unsubscribe = () => sub.unsubscribe()
89
+ })
90
+ if (!hasToken || instance.isDisposed()) {
91
+ initInFlight.delete(instance)
92
+ logger.debug('telemetry skipped: no token resolved or instance disposed')
93
+ return
94
+ }
95
+ }
96
+
97
+ const manager = createTelemetryManager({
98
+ sessionId: instance.instanceId,
99
+ getClient: () => getClient(instance, {apiVersion: DEFAULT_TELEMETRY_API_VERSION}),
100
+ projectId,
101
+ })
102
+
103
+ const consented = await manager.checkConsent()
104
+ logger.debug('consent check complete', {consented})
105
+ if (!consented || instance.isDisposed()) {
106
+ initInFlight.delete(instance)
107
+ manager.dispose()
108
+ return
109
+ }
110
+
111
+ initInFlight.delete(instance)
112
+ telemetryManagers.set(instance, manager)
113
+
114
+ const buffered = pendingHooks.get(instance)
115
+ if (buffered) {
116
+ logger.debug('flushing buffered hooks', {hooks: Array.from(buffered)})
117
+ for (const hookName of buffered) {
118
+ manager.logHookFirstUsed(hookName)
119
+ }
120
+ pendingHooks.delete(instance)
121
+ }
122
+
123
+ const config = instance.config
124
+ const perspective = typeof config.perspective === 'string' ? config.perspective : 'published'
125
+ const authMethod = config.auth?.token
126
+ ? 'token'
127
+ : config.studio?.auth?.token
128
+ ? 'studio'
129
+ : 'default'
130
+
131
+ logger.info('telemetry session started', {projectId, perspective, authMethod})
132
+ manager.logSessionStarted({
133
+ projectId,
134
+ perspective,
135
+ authMethod,
136
+ })
137
+
138
+ instance.onDispose(() => {
139
+ manager.endSession()
140
+ telemetryManagers.delete(instance)
141
+ logger.debug('telemetry session ended')
142
+ })
143
+ })
144
+ .catch((err) => {
145
+ initInFlight.delete(instance)
146
+ logger.warn('telemetry init failed', {error: err})
147
+ })
148
+ }
149
+
150
+ /**
151
+ * Retrieves the telemetry manager for an instance, if one exists.
152
+ * Returns undefined when telemetry is disabled or not yet initialized.
153
+ *
154
+ * @internal
155
+ */
156
+ export function getTelemetryManager(instance: SanityInstance): TelemetryManager | undefined {
157
+ return telemetryManagers.get(instance)
158
+ }
159
+
160
+ /**
161
+ * Record a hook name for an instance. If the telemetry manager is
162
+ * already initialized the event is logged immediately. Otherwise
163
+ * the name is buffered and flushed when init completes.
164
+ *
165
+ * @internal
166
+ */
167
+ export function trackHookMounted(instance: SanityInstance, hookName: string): void {
168
+ if (!isDevMode()) return
169
+
170
+ const manager = findManager(instance)
171
+ if (manager) {
172
+ logger.trace('hook mounted (logged)', {hookName, internal: true})
173
+ manager.logHookFirstUsed(hookName)
174
+ return
175
+ }
176
+
177
+ const root = getRootInstance(instance)
178
+ let hooks = pendingHooks.get(root)
179
+ if (!hooks) {
180
+ hooks = new Set()
181
+ pendingHooks.set(root, hooks)
182
+ }
183
+ if (!hooks.has(hookName)) {
184
+ logger.trace('hook mounted (buffered)', {hookName, internal: true})
185
+ }
186
+ hooks.add(hookName)
187
+ }
188
+
189
+ function findManager(instance: SanityInstance): TelemetryManager | undefined {
190
+ let current: SanityInstance | undefined = instance
191
+ while (current) {
192
+ const manager = telemetryManagers.get(current)
193
+ if (manager) return manager
194
+ current = typeof current.getParent === 'function' ? current.getParent() : undefined
195
+ }
196
+ return undefined
197
+ }
198
+
199
+ function getRootInstance(instance: SanityInstance): SanityInstance {
200
+ let current = instance
201
+ while (typeof current.getParent === 'function' && current.getParent()) {
202
+ current = current.getParent()!
203
+ }
204
+ return current
205
+ }
@@ -0,0 +1,263 @@
1
+ import {createBatchedStore} from '@sanity/telemetry'
2
+ import {afterEach, beforeEach, describe, expect, it, vi} from 'vitest'
3
+
4
+ import {createTelemetryManager} from './telemetryManager'
5
+
6
+ vi.mock('@sanity/telemetry', () => {
7
+ const mockLogger = {
8
+ log: vi.fn(),
9
+ updateUserProperties: vi.fn(),
10
+ resume: vi.fn(),
11
+ trace: vi.fn(),
12
+ }
13
+ return {
14
+ createBatchedStore: vi.fn(() => ({
15
+ logger: mockLogger,
16
+ end: vi.fn(),
17
+ flush: vi.fn(() => Promise.resolve()),
18
+ })),
19
+ defineEvent: vi.fn((opts) => ({
20
+ type: 'log' as const,
21
+ name: opts.name,
22
+ version: opts.version,
23
+ description: opts.description,
24
+ schema: undefined as never,
25
+ })),
26
+ }
27
+ })
28
+
29
+ vi.mock('../version', () => ({
30
+ CORE_SDK_VERSION: '2.8.0-test',
31
+ }))
32
+
33
+ describe('createTelemetryManager', () => {
34
+ const mockClient = {
35
+ request: vi.fn((): Promise<unknown> => Promise.resolve()),
36
+ getUrl: vi.fn((path: string) => `https://abc123.api.sanity.io/v2024-11-12${path}`),
37
+ }
38
+
39
+ const getClient = () => mockClient as never
40
+
41
+ beforeEach(() => {
42
+ vi.clearAllMocks()
43
+ })
44
+
45
+ afterEach(() => {
46
+ vi.restoreAllMocks()
47
+ })
48
+
49
+ it('creates a batched store with the given session ID', () => {
50
+ createTelemetryManager({
51
+ sessionId: 'test-session-id',
52
+ getClient,
53
+ projectId: 'abc123',
54
+ })
55
+
56
+ expect(createBatchedStore).toHaveBeenCalledWith(
57
+ 'test-session-id',
58
+ expect.objectContaining({
59
+ flushInterval: 30_000,
60
+ }),
61
+ )
62
+ })
63
+
64
+ it('logs session started with SDK version and provided data', () => {
65
+ const manager = createTelemetryManager({
66
+ sessionId: 'test-session-id',
67
+ getClient,
68
+ projectId: 'abc123',
69
+ })
70
+
71
+ const storeInstance = vi.mocked(createBatchedStore).mock.results[0].value
72
+ const logger = storeInstance.logger
73
+
74
+ manager.logSessionStarted({
75
+ projectId: 'abc123',
76
+ perspective: 'published',
77
+ authMethod: 'token',
78
+ })
79
+
80
+ expect(logger.log).toHaveBeenCalledWith(
81
+ expect.objectContaining({name: 'SDK Dev Session Started'}),
82
+ expect.objectContaining({
83
+ version: '2.8.0-test',
84
+ projectId: 'abc123',
85
+ perspective: 'published',
86
+ authMethod: 'token',
87
+ }),
88
+ )
89
+ })
90
+
91
+ it('deduplicates hook first-used events by name', () => {
92
+ const manager = createTelemetryManager({
93
+ sessionId: 'test-session-id',
94
+ getClient,
95
+ projectId: 'abc123',
96
+ })
97
+
98
+ const storeInstance = vi.mocked(createBatchedStore).mock.results[0].value
99
+ const logger = storeInstance.logger
100
+
101
+ manager.logHookFirstUsed('useQuery')
102
+ manager.logHookFirstUsed('useQuery')
103
+ manager.logHookFirstUsed('useDocument')
104
+
105
+ const hookCalls = vi
106
+ .mocked(logger.log)
107
+ .mock.calls.filter(([event]: [{name: string}]) => event.name === 'SDK Hook Mounted')
108
+ expect(hookCalls).toHaveLength(2)
109
+ expect(hookCalls[0][1]).toEqual({hookName: 'useQuery'})
110
+ expect(hookCalls[1][1]).toEqual({hookName: 'useDocument'})
111
+ })
112
+
113
+ it('tracks hooksUsed set', () => {
114
+ const manager = createTelemetryManager({
115
+ sessionId: 'test-session-id',
116
+ getClient,
117
+ projectId: 'abc123',
118
+ })
119
+
120
+ manager.logHookFirstUsed('useQuery')
121
+ manager.logHookFirstUsed('useDocument')
122
+
123
+ expect(manager.hooksUsed).toEqual(new Set(['useQuery', 'useDocument']))
124
+ })
125
+
126
+ it('logs dev error events', () => {
127
+ const manager = createTelemetryManager({
128
+ sessionId: 'test-session-id',
129
+ getClient,
130
+ projectId: 'abc123',
131
+ })
132
+
133
+ const storeInstance = vi.mocked(createBatchedStore).mock.results[0].value
134
+ const logger = storeInstance.logger
135
+
136
+ manager.logDevError('TypeError', 'documentStore')
137
+
138
+ expect(logger.log).toHaveBeenCalledWith(expect.objectContaining({name: 'SDK Dev Error'}), {
139
+ errorType: 'TypeError',
140
+ hookName: 'documentStore',
141
+ })
142
+ })
143
+
144
+ it('logs session ended with duration and hooksUsed on endSession', () => {
145
+ vi.useFakeTimers()
146
+
147
+ const manager = createTelemetryManager({
148
+ sessionId: 'test-session-id',
149
+ getClient,
150
+ projectId: 'abc123',
151
+ })
152
+
153
+ const storeInstance = vi.mocked(createBatchedStore).mock.results[0].value
154
+ const logger = storeInstance.logger
155
+
156
+ manager.logHookFirstUsed('useQuery')
157
+
158
+ vi.advanceTimersByTime(5000)
159
+
160
+ manager.endSession()
161
+
162
+ expect(logger.log).toHaveBeenCalledWith(
163
+ expect.objectContaining({name: 'SDK Dev Session Ended'}),
164
+ expect.objectContaining({
165
+ durationSeconds: 5,
166
+ hooksUsed: ['useQuery'],
167
+ }),
168
+ )
169
+
170
+ vi.useRealTimers()
171
+ })
172
+
173
+ describe('endSession teardown', () => {
174
+ it('always uses flush + end (no sendBeacon due to auth header limitation)', () => {
175
+ const manager = createTelemetryManager({
176
+ sessionId: 'test-session-id',
177
+ getClient,
178
+ projectId: 'abc123',
179
+ })
180
+
181
+ const storeInstance = vi.mocked(createBatchedStore).mock.results[0].value
182
+
183
+ manager.endSession()
184
+
185
+ expect(storeInstance.flush).toHaveBeenCalled()
186
+ expect(storeInstance.end).toHaveBeenCalled()
187
+ })
188
+ })
189
+
190
+ describe('consent resolution', () => {
191
+ it('checkConsent returns true when user has opted in', async () => {
192
+ mockClient.request.mockResolvedValue({status: 'granted'})
193
+
194
+ const manager = createTelemetryManager({
195
+ sessionId: 'test-session-id',
196
+ getClient,
197
+ projectId: 'abc123',
198
+ })
199
+
200
+ const result = await manager.checkConsent()
201
+ expect(result).toBe(true)
202
+ expect(mockClient.request).toHaveBeenCalledWith(
203
+ expect.objectContaining({uri: '/intake/telemetry-status'}),
204
+ )
205
+ })
206
+
207
+ it('checkConsent returns false when user has denied telemetry', async () => {
208
+ mockClient.request.mockResolvedValue({status: 'denied'})
209
+
210
+ const manager = createTelemetryManager({
211
+ sessionId: 'test-session-id',
212
+ getClient,
213
+ projectId: 'abc123',
214
+ })
215
+
216
+ expect(await manager.checkConsent()).toBe(false)
217
+ })
218
+
219
+ it('checkConsent returns false when consent is unset', async () => {
220
+ mockClient.request.mockResolvedValue({status: 'unset'})
221
+
222
+ const manager = createTelemetryManager({
223
+ sessionId: 'test-session-id',
224
+ getClient,
225
+ projectId: 'abc123',
226
+ })
227
+
228
+ expect(await manager.checkConsent()).toBe(false)
229
+ })
230
+
231
+ it('checkConsent returns false on network failure', async () => {
232
+ mockClient.request.mockRejectedValue(new Error('Network error'))
233
+
234
+ const manager = createTelemetryManager({
235
+ sessionId: 'test-session-id',
236
+ getClient,
237
+ projectId: 'abc123',
238
+ })
239
+
240
+ expect(await manager.checkConsent()).toBe(false)
241
+ })
242
+
243
+ it('caches consent after the first call', async () => {
244
+ mockClient.request.mockResolvedValue({status: 'granted'})
245
+
246
+ createTelemetryManager({
247
+ sessionId: 'test-session-id',
248
+ getClient,
249
+ projectId: 'abc123',
250
+ })
251
+
252
+ const storeOptions = vi.mocked(createBatchedStore).mock.calls[0][1]
253
+ const resolveConsent = storeOptions.resolveConsent
254
+
255
+ const first = await resolveConsent()
256
+ const second = await resolveConsent()
257
+
258
+ expect(first).toEqual({status: 'granted'})
259
+ expect(second).toEqual({status: 'granted'})
260
+ expect(mockClient.request).toHaveBeenCalledTimes(1)
261
+ })
262
+ })
263
+ })