@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.
- package/dist/_chunks-dts/utils.d.ts +2450 -0
- package/dist/_chunks-es/_internal.js +129 -0
- package/dist/_chunks-es/_internal.js.map +1 -0
- package/dist/_chunks-es/createGroqSearchFilter.js +1537 -0
- package/dist/_chunks-es/createGroqSearchFilter.js.map +1 -0
- package/dist/_chunks-es/telemetryManager.js +87 -0
- package/dist/_chunks-es/telemetryManager.js.map +1 -0
- package/dist/_chunks-es/version.js +7 -0
- package/dist/_chunks-es/version.js.map +1 -0
- package/dist/_exports/_internal.d.ts +64 -0
- package/dist/_exports/_internal.js +20 -0
- package/dist/_exports/_internal.js.map +1 -0
- package/dist/index.d.ts +2 -2343
- package/dist/index.js +465 -1813
- package/dist/index.js.map +1 -1
- package/package.json +17 -12
- package/src/_exports/_internal.ts +14 -0
- package/src/_exports/index.ts +18 -1
- package/src/auth/authStore.test.ts +150 -1
- package/src/auth/authStore.ts +11 -11
- package/src/auth/dashboardAuth.ts +2 -2
- package/src/auth/handleAuthCallback.ts +9 -3
- package/src/auth/logout.test.ts +1 -1
- package/src/auth/logout.ts +1 -1
- package/src/auth/refreshStampedToken.test.ts +118 -1
- package/src/auth/refreshStampedToken.ts +3 -2
- package/src/auth/standaloneAuth.ts +9 -3
- package/src/auth/studioAuth.ts +34 -7
- package/src/auth/studioModeAuth.ts +2 -1
- package/src/auth/subscribeToStateAndFetchCurrentUser.test.ts +10 -2
- package/src/auth/subscribeToStateAndFetchCurrentUser.ts +5 -1
- package/src/auth/subscribeToStorageEventsAndSetToken.ts +2 -2
- package/src/auth/utils.ts +33 -0
- package/src/client/clientStore.test.ts +44 -30
- package/src/client/clientStore.ts +49 -48
- package/src/comlink/controller/actions/getOrCreateChannel.ts +2 -2
- package/src/comlink/node/actions/getOrCreateNode.ts +2 -2
- package/src/comlink/node/getNodeState.ts +2 -1
- package/src/config/sanityConfig.ts +78 -12
- package/src/document/actions.ts +18 -11
- package/src/document/applyDocumentActions.test.ts +7 -6
- package/src/document/applyDocumentActions.ts +10 -4
- package/src/document/documentStore.test.ts +542 -188
- package/src/document/documentStore.ts +142 -76
- package/src/document/events.ts +7 -2
- package/src/document/permissions.test.ts +18 -16
- package/src/document/permissions.ts +35 -11
- package/src/document/processActions.test.ts +359 -32
- package/src/document/processActions.ts +106 -78
- package/src/document/reducers.test.ts +117 -29
- package/src/document/reducers.ts +47 -40
- package/src/document/sharedListener.ts +16 -6
- package/src/document/util.ts +14 -0
- package/src/favorites/favorites.test.ts +9 -2
- package/src/presence/bifurTransport.test.ts +46 -6
- package/src/presence/bifurTransport.ts +19 -2
- package/src/presence/presenceStore.test.ts +96 -0
- package/src/presence/presenceStore.ts +96 -24
- package/src/preview/getPreviewState.test.ts +115 -98
- package/src/preview/getPreviewState.ts +38 -60
- package/src/preview/previewProjectionUtils.test.ts +179 -0
- package/src/preview/previewProjectionUtils.ts +93 -0
- package/src/preview/resolvePreview.test.ts +42 -25
- package/src/preview/resolvePreview.ts +33 -10
- package/src/preview/{previewStore.ts → types.ts} +8 -17
- package/src/projection/getProjectionState.test.ts +16 -16
- package/src/projection/getProjectionState.ts +6 -5
- package/src/projection/projectionQuery.ts +2 -3
- package/src/projection/projectionStore.test.ts +2 -2
- package/src/projection/resolveProjection.ts +2 -2
- package/src/projection/subscribeToStateAndFetchBatches.test.ts +1 -1
- package/src/projection/subscribeToStateAndFetchBatches.ts +12 -11
- package/src/projection/types.ts +1 -1
- package/src/query/queryStore.test.ts +12 -12
- package/src/query/queryStore.ts +12 -11
- package/src/query/reducers.ts +3 -3
- package/src/releases/getPerspectiveState.ts +7 -6
- package/src/releases/releasesStore.test.ts +20 -5
- package/src/releases/releasesStore.ts +20 -8
- package/src/store/createActionBinder.test.ts +31 -31
- package/src/store/createActionBinder.ts +43 -38
- package/src/store/createSanityInstance.ts +2 -3
- package/src/store/createStateSourceAction.test.ts +62 -0
- package/src/store/createStateSourceAction.ts +34 -39
- package/src/telemetry/__telemetry__/sdk.telemetry.ts +42 -0
- package/src/telemetry/devMode.test.ts +52 -0
- package/src/telemetry/devMode.ts +40 -0
- package/src/telemetry/initTelemetry.test.ts +225 -0
- package/src/telemetry/initTelemetry.ts +205 -0
- package/src/telemetry/telemetryManager.test.ts +263 -0
- package/src/telemetry/telemetryManager.ts +187 -0
- package/src/users/reducers.ts +3 -4
- package/src/users/usersStore.test.ts +1 -0
- package/src/users/usersStore.ts +5 -1
- package/src/utils/createFetcherStore.test.ts +6 -4
- package/src/utils/createFetcherStore.ts +8 -5
- package/src/utils/getStagingApiHost.test.ts +21 -0
- package/src/utils/getStagingApiHost.ts +14 -0
- package/src/utils/ids.test.ts +1 -29
- package/src/utils/ids.ts +0 -10
- package/src/utils/isImportError.test.ts +72 -0
- package/src/utils/isImportError.ts +34 -0
- package/src/utils/object.test.ts +95 -0
- package/src/utils/object.ts +142 -0
- package/src/utils/setCleanupTimeout.ts +24 -0
- package/src/preview/previewQuery.test.ts +0 -236
- package/src/preview/previewQuery.ts +0 -153
- package/src/preview/previewStore.test.ts +0 -36
- package/src/preview/subscribeToStateAndFetchBatches.test.ts +0 -221
- package/src/preview/subscribeToStateAndFetchBatches.ts +0 -112
- 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
|
+
})
|