@drivemetadata-ai/sdk 0.1.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 (44) hide show
  1. package/CHANGELOG.md +12 -0
  2. package/LICENSE +21 -0
  3. package/README.md +241 -0
  4. package/dist/angular/index.cjs +1675 -0
  5. package/dist/angular/index.cjs.map +1 -0
  6. package/dist/angular/index.d.cts +26 -0
  7. package/dist/angular/index.d.ts +26 -0
  8. package/dist/angular/index.js +1649 -0
  9. package/dist/angular/index.js.map +1 -0
  10. package/dist/browser/index.cjs +1635 -0
  11. package/dist/browser/index.cjs.map +1 -0
  12. package/dist/browser/index.d.cts +21 -0
  13. package/dist/browser/index.d.ts +21 -0
  14. package/dist/browser/index.js +1595 -0
  15. package/dist/browser/index.js.map +1 -0
  16. package/dist/next/index.cjs +1693 -0
  17. package/dist/next/index.cjs.map +1 -0
  18. package/dist/next/index.d.cts +18 -0
  19. package/dist/next/index.d.ts +18 -0
  20. package/dist/next/index.js +1650 -0
  21. package/dist/next/index.js.map +1 -0
  22. package/dist/node/index.cjs +384 -0
  23. package/dist/node/index.cjs.map +1 -0
  24. package/dist/node/index.d.cts +91 -0
  25. package/dist/node/index.d.ts +91 -0
  26. package/dist/node/index.js +354 -0
  27. package/dist/node/index.js.map +1 -0
  28. package/dist/react/index.cjs +1721 -0
  29. package/dist/react/index.cjs.map +1 -0
  30. package/dist/react/index.d.cts +26 -0
  31. package/dist/react/index.d.ts +26 -0
  32. package/dist/react/index.js +1674 -0
  33. package/dist/react/index.js.map +1 -0
  34. package/dist/types-mgbdL1V7.d.cts +123 -0
  35. package/dist/types-mgbdL1V7.d.ts +123 -0
  36. package/docs/angular-integration.md +106 -0
  37. package/docs/architecture.md +109 -0
  38. package/docs/index.md +18 -0
  39. package/docs/integration.md +520 -0
  40. package/docs/node-server-integration.md +147 -0
  41. package/docs/npm-browser-sdk.md +143 -0
  42. package/docs/react-next-integration.md +168 -0
  43. package/docs/security-privacy.md +128 -0
  44. package/package.json +101 -0
@@ -0,0 +1,354 @@
1
+ // src/node/errors.ts
2
+ var DmdServerSdkError = class extends Error {
3
+ constructor(message, cause) {
4
+ super(message);
5
+ this.cause = cause;
6
+ this.name = "DmdServerSdkError";
7
+ }
8
+ };
9
+ var DmdServerValidationError = class extends DmdServerSdkError {
10
+ constructor(message) {
11
+ super(message);
12
+ this.name = "DmdServerValidationError";
13
+ }
14
+ };
15
+ var DmdServerRequestError = class extends DmdServerSdkError {
16
+ constructor(message, status, cause) {
17
+ super(message, cause);
18
+ this.status = status;
19
+ this.name = "DmdServerRequestError";
20
+ }
21
+ };
22
+
23
+ // src/core/uuid.ts
24
+ function createUuid() {
25
+ const cryptoApi = globalThis.crypto;
26
+ if (typeof cryptoApi?.randomUUID === "function") {
27
+ return cryptoApi.randomUUID();
28
+ }
29
+ const bytes = new Uint8Array(16);
30
+ if (typeof cryptoApi?.getRandomValues === "function") {
31
+ cryptoApi.getRandomValues(bytes);
32
+ } else {
33
+ for (let index = 0; index < bytes.length; index += 1) {
34
+ bytes[index] = Math.floor(Math.random() * 256);
35
+ }
36
+ }
37
+ bytes[6] = (bytes[6] ?? 0) & 15 | 64;
38
+ bytes[8] = (bytes[8] ?? 0) & 63 | 128;
39
+ const hex = Array.from(bytes, (byte) => byte.toString(16).padStart(2, "0")).join("");
40
+ return `${hex.slice(0, 8)}-${hex.slice(8, 12)}-${hex.slice(12, 16)}-${hex.slice(16, 20)}-${hex.slice(20)}`;
41
+ }
42
+ function isUuid(value) {
43
+ return typeof value === "string" && /^[0-9a-f]{8}-[0-9a-f]{4}-[1-8][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i.test(value);
44
+ }
45
+ function ensureUuid(value) {
46
+ return isUuid(value) ? value : createUuid();
47
+ }
48
+
49
+ // src/core/backend-payload.ts
50
+ function formatUtcTimestamp(value) {
51
+ const date = typeof value === "string" || typeof value === "number" || value instanceof Date ? new Date(value) : /* @__PURE__ */ new Date();
52
+ const safeDate = Number.isNaN(date.getTime()) ? /* @__PURE__ */ new Date() : date;
53
+ return safeDate.toISOString().replace("T", " ").slice(0, 19);
54
+ }
55
+ function cleanObject(value) {
56
+ if (Array.isArray(value)) {
57
+ const cleaned = value.map((item) => cleanObject(item)).filter((item) => item !== null && item !== void 0 && item !== "");
58
+ return cleaned.length > 0 ? cleaned : void 0;
59
+ }
60
+ if (value && typeof value === "object") {
61
+ const cleaned = Object.fromEntries(
62
+ Object.entries(value).map(([key, item]) => [key, cleanObject(item)]).filter(([, item]) => {
63
+ if (item === null || item === void 0 || item === "") return false;
64
+ if (typeof item === "object" && !Array.isArray(item) && Object.keys(item).length === 0) return false;
65
+ return true;
66
+ })
67
+ );
68
+ return Object.keys(cleaned).length > 0 ? cleaned : void 0;
69
+ }
70
+ return value;
71
+ }
72
+ function createBackendCollectorPayload(input) {
73
+ const eventData = input.eventData && typeof input.eventData === "object" ? input.eventData : {};
74
+ const page = eventData.page && typeof eventData.page === "object" ? eventData.page : {};
75
+ const timestamp = formatUtcTimestamp(eventData.timestamp ?? input.timestamp);
76
+ const normalizedEventData = {
77
+ ...eventData,
78
+ timestamp,
79
+ requestSentAt: formatUtcTimestamp(eventData.requestSentAt ?? input.requestSentAt ?? timestamp),
80
+ requestReceivedAt: formatUtcTimestamp(eventData.requestReceivedAt ?? input.requestReceivedAt ?? timestamp)
81
+ };
82
+ const metaData = {
83
+ ...normalizedEventData,
84
+ requestId: ensureUuid(typeof input.requestId === "string" ? input.requestId : void 0),
85
+ timestamp,
86
+ eventType: input.eventType,
87
+ requestFrom: input.requestFrom ?? "3",
88
+ clientId: input.clientId,
89
+ workspaceId: input.workspaceId,
90
+ token: input.token,
91
+ anonymousId: ensureUuid(
92
+ typeof eventData.anonymousId === "string" ? eventData.anonymousId : typeof input.anonymousId === "string" ? input.anonymousId : void 0
93
+ ),
94
+ sessionId: ensureUuid(
95
+ typeof eventData.sessionId === "string" ? eventData.sessionId : typeof input.sessionId === "string" ? input.sessionId : void 0
96
+ ),
97
+ ua: input.ua,
98
+ appDetails: { app_id: input.appId },
99
+ page: { ...page, url: page.url ?? input.pageUrl },
100
+ requestSentAt: normalizedEventData.requestSentAt,
101
+ requestReceivedAt: normalizedEventData.requestReceivedAt
102
+ };
103
+ const payload = { metaData };
104
+ return cleanObject(payload);
105
+ }
106
+
107
+ // src/core/backend-schema.ts
108
+ var requiredMetaDataFields = [
109
+ "requestId",
110
+ "timestamp",
111
+ "eventType",
112
+ "requestFrom",
113
+ "clientId",
114
+ "workspaceId",
115
+ "anonymousId",
116
+ "sessionId"
117
+ ];
118
+ function isPresent(value) {
119
+ return value !== null && value !== void 0 && value !== "";
120
+ }
121
+ function validateBackendCollectorPayload(payload) {
122
+ const errors = [];
123
+ const metaData = payload.metaData;
124
+ if (!metaData || typeof metaData !== "object" || Array.isArray(metaData)) {
125
+ return {
126
+ ok: false,
127
+ errors: ["metaData is required"]
128
+ };
129
+ }
130
+ const metadataRecord = metaData;
131
+ for (const field of requiredMetaDataFields) {
132
+ if (!isPresent(metadataRecord[field])) {
133
+ errors.push(`metaData.${field} is required`);
134
+ }
135
+ }
136
+ if (isPresent(metadataRecord.eventType) && typeof metadataRecord.eventType !== "string") {
137
+ errors.push("metaData.eventType must be a string");
138
+ }
139
+ return {
140
+ ok: errors.length === 0,
141
+ errors
142
+ };
143
+ }
144
+
145
+ // src/node/retry.ts
146
+ function isRetryableStatus(status) {
147
+ return status === 408 || status === 429 || status >= 500;
148
+ }
149
+ function getRetryDelayMs(attemptIndex, retry = {}) {
150
+ const minDelayMs = retry.minDelayMs ?? 250;
151
+ const maxDelayMs = retry.maxDelayMs ?? 2e3;
152
+ const exponentialDelay = minDelayMs * 2 ** attemptIndex;
153
+ return Math.min(exponentialDelay, maxDelayMs);
154
+ }
155
+ function getRetryAttempts(retry) {
156
+ return retry?.attempts ?? 0;
157
+ }
158
+
159
+ // src/node/validation.ts
160
+ function requireString(value, message) {
161
+ if (typeof value !== "string" || value.trim() === "") {
162
+ throw new DmdServerValidationError(message);
163
+ }
164
+ return value;
165
+ }
166
+ function validateServerConfig(config) {
167
+ requireString(config.writeKey || config.token, "DMD server SDK config writeKey or token is required");
168
+ }
169
+ function validateTrackPayload(payload) {
170
+ requireString(payload.event, "DMD server SDK track event is required");
171
+ if (!payload.userId && !payload.anonymousId) {
172
+ throw new DmdServerValidationError("DMD server SDK track requires userId or anonymousId");
173
+ }
174
+ }
175
+ function validateIdentifyPayload(payload) {
176
+ requireString(payload.userId, "DMD server SDK identify requires userId");
177
+ }
178
+ function validatePagePayload(payload) {
179
+ if (!payload.userId && !payload.anonymousId) {
180
+ throw new DmdServerValidationError("DMD server SDK page requires userId or anonymousId");
181
+ }
182
+ }
183
+ function validateAliasPayload(payload) {
184
+ const hasPreviousId = typeof payload.previousId === "string" && payload.previousId.trim() !== "";
185
+ const hasUserId = typeof payload.userId === "string" && payload.userId.trim() !== "";
186
+ if (!hasPreviousId || !hasUserId) {
187
+ throw new DmdServerValidationError("DMD server SDK alias requires previousId and userId");
188
+ }
189
+ }
190
+ function validateGroupPayload(payload) {
191
+ requireString(payload.groupId, "DMD server SDK group requires groupId");
192
+ if (!payload.userId && !payload.anonymousId) {
193
+ throw new DmdServerValidationError("DMD server SDK group requires userId or anonymousId");
194
+ }
195
+ }
196
+ function validateBatchPayload(items) {
197
+ if (!Array.isArray(items) || items.length === 0) {
198
+ throw new DmdServerValidationError("DMD server SDK batch requires at least one item");
199
+ }
200
+ for (const item of items) {
201
+ if (item.type === "track") validateTrackPayload(item.payload);
202
+ else if (item.type === "identify") validateIdentifyPayload(item.payload);
203
+ else if (item.type === "page") validatePagePayload(item.payload);
204
+ else if (item.type === "alias") validateAliasPayload(item.payload);
205
+ else if (item.type === "group") validateGroupPayload(item.payload);
206
+ else throw new DmdServerValidationError("DMD server SDK batch item has invalid type");
207
+ }
208
+ }
209
+
210
+ // src/node/client.ts
211
+ var DEFAULT_ENDPOINT = "https://sdk.drivemetadata.com/v2/data-collector";
212
+ function sleep(ms) {
213
+ return new Promise((resolve) => setTimeout(resolve, ms));
214
+ }
215
+ function withTimestamp(payload) {
216
+ return {
217
+ ...payload,
218
+ timestamp: payload.timestamp ?? (/* @__PURE__ */ new Date()).toISOString()
219
+ };
220
+ }
221
+ function withBatchItemTimestamp(item) {
222
+ if (item.type === "track") return { type: item.type, payload: withTimestamp(item.payload) };
223
+ if (item.type === "identify") return { type: item.type, payload: withTimestamp(item.payload) };
224
+ if (item.type === "page") return { type: item.type, payload: withTimestamp(item.payload) };
225
+ if (item.type === "alias") return { type: item.type, payload: withTimestamp(item.payload) };
226
+ return { type: item.type, payload: withTimestamp(item.payload) };
227
+ }
228
+ function toCollectorPayload(config, type, payload, token) {
229
+ const basePayload = payload;
230
+ const properties = "properties" in payload && payload.properties ? payload.properties : {};
231
+ const collectorPayload = createBackendCollectorPayload({
232
+ requestId: ensureUuid(payload.messageId),
233
+ timestamp: payload.timestamp ?? (/* @__PURE__ */ new Date()).toISOString(),
234
+ eventType: type === "track" ? basePayload.event : type,
235
+ requestFrom: config.requestFrom ?? "3",
236
+ clientId: config.clientId,
237
+ workspaceId: config.workspaceId,
238
+ token,
239
+ anonymousId: basePayload.anonymousId,
240
+ sessionId: basePayload.sessionId,
241
+ appId: config.appId,
242
+ eventData: {
243
+ ...properties,
244
+ anonymousId: basePayload.anonymousId,
245
+ sessionId: basePayload.sessionId,
246
+ timestamp: payload.timestamp ?? (/* @__PURE__ */ new Date()).toISOString()
247
+ }
248
+ });
249
+ const validation = validateBackendCollectorPayload(collectorPayload);
250
+ if (!validation.ok) {
251
+ throw new DmdServerValidationError(`DMD server SDK backend payload invalid: ${validation.errors.join(", ")}`);
252
+ }
253
+ return collectorPayload;
254
+ }
255
+ function createTimeoutController(timeoutMs) {
256
+ if (!timeoutMs) {
257
+ return { cleanup: () => void 0 };
258
+ }
259
+ const controller = new AbortController();
260
+ const timer = setTimeout(() => controller.abort(), timeoutMs);
261
+ return {
262
+ signal: controller.signal,
263
+ cleanup: () => clearTimeout(timer)
264
+ };
265
+ }
266
+ function createDmdServerClient(config) {
267
+ validateServerConfig(config);
268
+ const endpoint = config.endpoint ?? DEFAULT_ENDPOINT;
269
+ const fetchImpl = config.fetch ?? globalThis.fetch;
270
+ const authorizationToken = config.writeKey || config.token;
271
+ if (typeof fetchImpl !== "function") {
272
+ throw new Error("DMD server SDK requires fetch. Use Node.js 18+ or pass config.fetch.");
273
+ }
274
+ async function send(requestPayload) {
275
+ const attempts = getRetryAttempts(config.retry);
276
+ let lastError;
277
+ for (let attemptIndex = 0; attemptIndex <= attempts; attemptIndex += 1) {
278
+ try {
279
+ const requestInit = {
280
+ method: "POST",
281
+ headers: {
282
+ "Content-Type": "application/json"
283
+ },
284
+ body: JSON.stringify(requestPayload)
285
+ };
286
+ const timeout = createTimeoutController(config.timeoutMs);
287
+ if (timeout.signal !== void 0) {
288
+ requestInit.signal = timeout.signal;
289
+ }
290
+ const response = await fetchImpl(endpoint, requestInit).finally(timeout.cleanup);
291
+ if (response.ok) {
292
+ return;
293
+ }
294
+ if (!isRetryableStatus(response.status) || attemptIndex === attempts) {
295
+ throw new DmdServerRequestError(
296
+ `DMD server SDK request failed with status ${response.status}`,
297
+ response.status
298
+ );
299
+ }
300
+ await sleep(getRetryDelayMs(attemptIndex, config.retry));
301
+ } catch (error) {
302
+ lastError = error;
303
+ if (error instanceof DmdServerRequestError || attemptIndex === attempts) {
304
+ throw error;
305
+ }
306
+ await sleep(getRetryDelayMs(attemptIndex, config.retry));
307
+ }
308
+ }
309
+ throw new DmdServerRequestError("DMD server SDK request failed", void 0, lastError);
310
+ }
311
+ function envelope(type, payload) {
312
+ return toCollectorPayload(config, type, payload, authorizationToken);
313
+ }
314
+ return {
315
+ async track(payload) {
316
+ validateTrackPayload(payload);
317
+ await send(envelope("track", withTimestamp(payload)));
318
+ },
319
+ async identify(payload) {
320
+ validateIdentifyPayload(payload);
321
+ await send(envelope("identify", withTimestamp(payload)));
322
+ },
323
+ async page(payload) {
324
+ validatePagePayload(payload);
325
+ await send(envelope("page", withTimestamp(payload)));
326
+ },
327
+ async alias(payload) {
328
+ validateAliasPayload(payload);
329
+ await send(envelope("alias", withTimestamp(payload)));
330
+ },
331
+ async group(payload) {
332
+ validateGroupPayload(payload);
333
+ await send(envelope("group", withTimestamp(payload)));
334
+ },
335
+ async batch(items) {
336
+ validateBatchPayload(items);
337
+ await send({
338
+ type: "batch",
339
+ batch: items.map(withBatchItemTimestamp),
340
+ sentAt: (/* @__PURE__ */ new Date()).toISOString()
341
+ });
342
+ },
343
+ async flush() {
344
+ return Promise.resolve();
345
+ }
346
+ };
347
+ }
348
+ export {
349
+ DmdServerRequestError,
350
+ DmdServerSdkError,
351
+ DmdServerValidationError,
352
+ createDmdServerClient
353
+ };
354
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/node/errors.ts","../../src/core/uuid.ts","../../src/core/backend-payload.ts","../../src/core/backend-schema.ts","../../src/node/retry.ts","../../src/node/validation.ts","../../src/node/client.ts"],"sourcesContent":["export class DmdServerSdkError extends Error {\n constructor(message: string, public readonly cause?: unknown) {\n super(message);\n this.name = 'DmdServerSdkError';\n }\n}\n\nexport class DmdServerValidationError extends DmdServerSdkError {\n constructor(message: string) {\n super(message);\n this.name = 'DmdServerValidationError';\n }\n}\n\nexport class DmdServerRequestError extends DmdServerSdkError {\n constructor(message: string, public readonly status?: number, cause?: unknown) {\n super(message, cause);\n this.name = 'DmdServerRequestError';\n }\n}\n","export function createUuid(): string {\n const cryptoApi = globalThis.crypto;\n if (typeof cryptoApi?.randomUUID === 'function') {\n return cryptoApi.randomUUID();\n }\n\n const bytes = new Uint8Array(16);\n if (typeof cryptoApi?.getRandomValues === 'function') {\n cryptoApi.getRandomValues(bytes);\n } else {\n for (let index = 0; index < bytes.length; index += 1) {\n bytes[index] = Math.floor(Math.random() * 256);\n }\n }\n\n bytes[6] = ((bytes[6] ?? 0) & 0x0f) | 0x40;\n bytes[8] = ((bytes[8] ?? 0) & 0x3f) | 0x80;\n\n const hex = Array.from(bytes, byte => byte.toString(16).padStart(2, '0')).join('');\n return `${hex.slice(0, 8)}-${hex.slice(8, 12)}-${hex.slice(12, 16)}-${hex.slice(16, 20)}-${hex.slice(20)}`;\n}\n\nexport function isUuid(value: unknown): value is string {\n return typeof value === 'string'\n && /^[0-9a-f]{8}-[0-9a-f]{4}-[1-8][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i.test(value);\n}\n\nexport function ensureUuid(value?: string): string {\n return isUuid(value) ? value : createUuid();\n}\n","import { ensureUuid } from './uuid';\n\nfunction formatUtcTimestamp(value: unknown): string {\n const date =\n typeof value === 'string' || typeof value === 'number' || value instanceof Date\n ? new Date(value)\n : new Date();\n const safeDate = Number.isNaN(date.getTime()) ? new Date() : date;\n return safeDate.toISOString().replace('T', ' ').slice(0, 19);\n}\n\nfunction cleanObject(value: unknown): unknown {\n if (Array.isArray(value)) {\n const cleaned = value\n .map(item => cleanObject(item))\n .filter(item => item !== null && item !== undefined && item !== '');\n return cleaned.length > 0 ? cleaned : undefined;\n }\n\n if (value && typeof value === 'object') {\n const cleaned = Object.fromEntries(\n Object.entries(value as Record<string, unknown>)\n .map(([key, item]) => [key, cleanObject(item)])\n .filter(([, item]) => {\n if (item === null || item === undefined || item === '') return false;\n if (typeof item === 'object' && !Array.isArray(item) && Object.keys(item).length === 0) return false;\n return true;\n })\n );\n return Object.keys(cleaned).length > 0 ? cleaned : undefined;\n }\n\n return value;\n}\n\nexport function createBackendCollectorPayload(input: Record<string, unknown>): Record<string, unknown> {\n const eventData = input.eventData && typeof input.eventData === 'object'\n ? input.eventData as Record<string, unknown>\n : {};\n const page = eventData.page && typeof eventData.page === 'object'\n ? eventData.page as Record<string, unknown>\n : {};\n const timestamp = formatUtcTimestamp(eventData.timestamp ?? input.timestamp);\n const normalizedEventData = {\n ...eventData,\n timestamp,\n requestSentAt: formatUtcTimestamp(eventData.requestSentAt ?? input.requestSentAt ?? timestamp),\n requestReceivedAt: formatUtcTimestamp(eventData.requestReceivedAt ?? input.requestReceivedAt ?? timestamp)\n };\n\n const metaData: Record<string, unknown> = {\n ...normalizedEventData,\n requestId: ensureUuid(typeof input.requestId === 'string' ? input.requestId : undefined),\n timestamp,\n eventType: input.eventType,\n requestFrom: input.requestFrom ?? '3',\n clientId: input.clientId,\n workspaceId: input.workspaceId,\n token: input.token,\n anonymousId: ensureUuid(\n typeof eventData.anonymousId === 'string'\n ? eventData.anonymousId\n : typeof input.anonymousId === 'string'\n ? input.anonymousId\n : undefined\n ),\n sessionId: ensureUuid(\n typeof eventData.sessionId === 'string'\n ? eventData.sessionId\n : typeof input.sessionId === 'string'\n ? input.sessionId\n : undefined\n ),\n ua: input.ua,\n appDetails: { app_id: input.appId },\n page: { ...page, url: page.url ?? input.pageUrl },\n requestSentAt: normalizedEventData.requestSentAt,\n requestReceivedAt: normalizedEventData.requestReceivedAt\n };\n\n const payload = { metaData };\n\n return cleanObject(payload) as Record<string, unknown>;\n}\n","const requiredMetaDataFields = [\n 'requestId',\n 'timestamp',\n 'eventType',\n 'requestFrom',\n 'clientId',\n 'workspaceId',\n 'anonymousId',\n 'sessionId'\n] as const;\n\nexport interface DmdBackendPayloadValidation {\n ok: boolean;\n errors: string[];\n}\n\nfunction isPresent(value: unknown): boolean {\n return value !== null && value !== undefined && value !== '';\n}\n\nexport function validateBackendCollectorPayload(payload: Record<string, unknown>): DmdBackendPayloadValidation {\n const errors: string[] = [];\n const metaData = payload.metaData;\n\n if (!metaData || typeof metaData !== 'object' || Array.isArray(metaData)) {\n return {\n ok: false,\n errors: ['metaData is required']\n };\n }\n\n const metadataRecord = metaData as Record<string, unknown>;\n for (const field of requiredMetaDataFields) {\n if (!isPresent(metadataRecord[field])) {\n errors.push(`metaData.${field} is required`);\n }\n }\n\n if (isPresent(metadataRecord.eventType) && typeof metadataRecord.eventType !== 'string') {\n errors.push('metaData.eventType must be a string');\n }\n\n return {\n ok: errors.length === 0,\n errors\n };\n}\n","import type { DmdServerRetryConfig } from './types';\n\nexport function isRetryableStatus(status: number): boolean {\n return status === 408 || status === 429 || status >= 500;\n}\n\nexport function getRetryDelayMs(attemptIndex: number, retry: DmdServerRetryConfig = {}): number {\n const minDelayMs = retry.minDelayMs ?? 250;\n const maxDelayMs = retry.maxDelayMs ?? 2000;\n const exponentialDelay = minDelayMs * 2 ** attemptIndex;\n\n return Math.min(exponentialDelay, maxDelayMs);\n}\n\nexport function getRetryAttempts(retry: DmdServerRetryConfig | undefined): number {\n return retry?.attempts ?? 0;\n}\n","import { DmdServerValidationError } from './errors';\nimport type {\n DmdServerAliasPayload,\n DmdServerBatchItem,\n DmdServerConfig,\n DmdServerGroupPayload,\n DmdServerIdentifyPayload,\n DmdServerPagePayload,\n DmdServerTrackPayload\n} from './types';\n\nfunction requireString(value: string | undefined, message: string): string {\n if (typeof value !== 'string' || value.trim() === '') {\n throw new DmdServerValidationError(message);\n }\n\n return value;\n}\n\nexport function validateServerConfig(config: DmdServerConfig): void {\n requireString(config.writeKey || config.token, 'DMD server SDK config writeKey or token is required');\n}\n\nexport function validateTrackPayload(payload: DmdServerTrackPayload): void {\n requireString(payload.event, 'DMD server SDK track event is required');\n\n if (!payload.userId && !payload.anonymousId) {\n throw new DmdServerValidationError('DMD server SDK track requires userId or anonymousId');\n }\n}\n\nexport function validateIdentifyPayload(payload: DmdServerIdentifyPayload): void {\n requireString(payload.userId, 'DMD server SDK identify requires userId');\n}\n\nexport function validatePagePayload(payload: DmdServerPagePayload): void {\n if (!payload.userId && !payload.anonymousId) {\n throw new DmdServerValidationError('DMD server SDK page requires userId or anonymousId');\n }\n}\n\nexport function validateAliasPayload(payload: DmdServerAliasPayload): void {\n const hasPreviousId = typeof payload.previousId === 'string' && payload.previousId.trim() !== '';\n const hasUserId = typeof payload.userId === 'string' && payload.userId.trim() !== '';\n if (!hasPreviousId || !hasUserId) {\n throw new DmdServerValidationError('DMD server SDK alias requires previousId and userId');\n }\n}\n\nexport function validateGroupPayload(payload: DmdServerGroupPayload): void {\n requireString(payload.groupId, 'DMD server SDK group requires groupId');\n if (!payload.userId && !payload.anonymousId) {\n throw new DmdServerValidationError('DMD server SDK group requires userId or anonymousId');\n }\n}\n\nexport function validateBatchPayload(items: DmdServerBatchItem[]): void {\n if (!Array.isArray(items) || items.length === 0) {\n throw new DmdServerValidationError('DMD server SDK batch requires at least one item');\n }\n\n for (const item of items) {\n if (item.type === 'track') validateTrackPayload(item.payload);\n else if (item.type === 'identify') validateIdentifyPayload(item.payload);\n else if (item.type === 'page') validatePagePayload(item.payload);\n else if (item.type === 'alias') validateAliasPayload(item.payload);\n else if (item.type === 'group') validateGroupPayload(item.payload);\n else throw new DmdServerValidationError('DMD server SDK batch item has invalid type');\n }\n}\n","import { DmdServerRequestError, DmdServerValidationError } from './errors';\nimport { createBackendCollectorPayload } from '../core/backend-payload';\nimport { validateBackendCollectorPayload } from '../core/backend-schema';\nimport { ensureUuid } from '../core/uuid';\nimport { getRetryAttempts, getRetryDelayMs, isRetryableStatus } from './retry';\nimport type {\n DmdServerAliasPayload,\n DmdServerBatchItem,\n DmdServerClient,\n DmdServerConfig,\n DmdServerGroupPayload,\n DmdServerIdentifyPayload,\n DmdServerPagePayload,\n DmdServerSingleRequestPayload,\n DmdServerTrackPayload\n} from './types';\nimport {\n validateAliasPayload,\n validateBatchPayload,\n validateGroupPayload,\n validateIdentifyPayload,\n validatePagePayload,\n validateServerConfig,\n validateTrackPayload\n} from './validation';\n\nconst DEFAULT_ENDPOINT = 'https://sdk.drivemetadata.com/v2/data-collector';\n\nfunction sleep(ms: number): Promise<void> {\n return new Promise(resolve => setTimeout(resolve, ms));\n}\n\nfunction withTimestamp<T extends { timestamp?: string }>(payload: T): T {\n return {\n ...payload,\n timestamp: payload.timestamp ?? new Date().toISOString()\n };\n}\n\nfunction withBatchItemTimestamp(item: DmdServerBatchItem): DmdServerBatchItem {\n if (item.type === 'track') return { type: item.type, payload: withTimestamp(item.payload) };\n if (item.type === 'identify') return { type: item.type, payload: withTimestamp(item.payload) };\n if (item.type === 'page') return { type: item.type, payload: withTimestamp(item.payload) };\n if (item.type === 'alias') return { type: item.type, payload: withTimestamp(item.payload) };\n return { type: item.type, payload: withTimestamp(item.payload) };\n}\n\nfunction toCollectorPayload(\n config: DmdServerConfig,\n type: DmdServerSingleRequestPayload['type'],\n payload: DmdServerSingleRequestPayload['payload'],\n token: string | undefined\n) {\n const basePayload = payload as DmdServerTrackPayload & { anonymousId?: string; sessionId?: string };\n const properties = 'properties' in payload && payload.properties ? payload.properties : {};\n const collectorPayload = createBackendCollectorPayload({\n requestId: ensureUuid(payload.messageId),\n timestamp: payload.timestamp ?? new Date().toISOString(),\n eventType: type === 'track' ? basePayload.event : type,\n requestFrom: config.requestFrom ?? '3',\n clientId: config.clientId,\n workspaceId: config.workspaceId,\n token,\n anonymousId: basePayload.anonymousId,\n sessionId: basePayload.sessionId,\n appId: config.appId,\n eventData: {\n ...properties,\n anonymousId: basePayload.anonymousId,\n sessionId: basePayload.sessionId,\n timestamp: payload.timestamp ?? new Date().toISOString()\n }\n });\n const validation = validateBackendCollectorPayload(collectorPayload);\n if (!validation.ok) {\n throw new DmdServerValidationError(`DMD server SDK backend payload invalid: ${validation.errors.join(', ')}`);\n }\n return collectorPayload;\n}\n\nfunction createTimeoutController(timeoutMs: number | undefined): { signal?: AbortSignal; cleanup: () => void } {\n if (!timeoutMs) {\n return { cleanup: () => undefined };\n }\n\n const controller = new AbortController();\n const timer = setTimeout(() => controller.abort(), timeoutMs);\n return {\n signal: controller.signal,\n cleanup: () => clearTimeout(timer)\n };\n}\n\nexport function createDmdServerClient(config: DmdServerConfig): DmdServerClient {\n validateServerConfig(config);\n\n const endpoint = config.endpoint ?? DEFAULT_ENDPOINT;\n const fetchImpl = config.fetch ?? globalThis.fetch;\n const authorizationToken = config.writeKey || config.token;\n\n if (typeof fetchImpl !== 'function') {\n throw new Error('DMD server SDK requires fetch. Use Node.js 18+ or pass config.fetch.');\n }\n\n async function send(requestPayload: Record<string, unknown>): Promise<void> {\n const attempts = getRetryAttempts(config.retry);\n let lastError: unknown;\n\n for (let attemptIndex = 0; attemptIndex <= attempts; attemptIndex += 1) {\n try {\n const requestInit: RequestInit = {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json'\n },\n body: JSON.stringify(requestPayload)\n };\n const timeout = createTimeoutController(config.timeoutMs);\n if (timeout.signal !== undefined) {\n requestInit.signal = timeout.signal;\n }\n\n const response = await fetchImpl(endpoint, requestInit).finally(timeout.cleanup);\n\n if (response.ok) {\n return;\n }\n\n if (!isRetryableStatus(response.status) || attemptIndex === attempts) {\n throw new DmdServerRequestError(\n `DMD server SDK request failed with status ${response.status}`,\n response.status\n );\n }\n\n await sleep(getRetryDelayMs(attemptIndex, config.retry));\n } catch (error) {\n lastError = error;\n\n if (error instanceof DmdServerRequestError || attemptIndex === attempts) {\n throw error;\n }\n\n await sleep(getRetryDelayMs(attemptIndex, config.retry));\n }\n }\n\n throw new DmdServerRequestError('DMD server SDK request failed', undefined, lastError);\n }\n\n function envelope(type: DmdServerSingleRequestPayload['type'], payload: DmdServerSingleRequestPayload['payload']) {\n return toCollectorPayload(config, type, payload, authorizationToken);\n }\n\n return {\n async track(payload: DmdServerTrackPayload) {\n validateTrackPayload(payload);\n await send(envelope('track', withTimestamp(payload)));\n },\n async identify(payload: DmdServerIdentifyPayload) {\n validateIdentifyPayload(payload);\n await send(envelope('identify', withTimestamp(payload)));\n },\n async page(payload: DmdServerPagePayload) {\n validatePagePayload(payload);\n await send(envelope('page', withTimestamp(payload)));\n },\n async alias(payload: DmdServerAliasPayload) {\n validateAliasPayload(payload);\n await send(envelope('alias', withTimestamp(payload)));\n },\n async group(payload: DmdServerGroupPayload) {\n validateGroupPayload(payload);\n await send(envelope('group', withTimestamp(payload)));\n },\n async batch(items: DmdServerBatchItem[]) {\n validateBatchPayload(items);\n await send({\n type: 'batch',\n batch: items.map(withBatchItemTimestamp),\n sentAt: new Date().toISOString()\n });\n },\n async flush() {\n return Promise.resolve();\n }\n };\n}\n"],"mappings":"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","names":[]}