@grainql/analytics-web 1.7.2 → 2.0.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 (116) hide show
  1. package/README.md +71 -777
  2. package/dist/cjs/index.d.ts +35 -2
  3. package/dist/cjs/index.d.ts.map +1 -1
  4. package/dist/cjs/index.js.map +1 -1
  5. package/dist/cjs/react/GrainProvider.d.ts +11 -0
  6. package/dist/cjs/react/GrainProvider.d.ts.map +1 -0
  7. package/dist/cjs/react/GrainProvider.js +79 -0
  8. package/dist/cjs/react/GrainProvider.js.map +1 -0
  9. package/dist/cjs/react/context.d.ts +11 -0
  10. package/dist/cjs/react/context.d.ts.map +1 -0
  11. package/dist/cjs/react/context.js +43 -0
  12. package/dist/cjs/react/context.js.map +1 -0
  13. package/dist/cjs/react/hooks/useAllConfigs.d.ts +8 -0
  14. package/dist/cjs/react/hooks/useAllConfigs.d.ts.map +1 -0
  15. package/dist/cjs/react/hooks/useAllConfigs.js +112 -0
  16. package/dist/cjs/react/hooks/useAllConfigs.js.map +1 -0
  17. package/dist/cjs/react/hooks/useConfig.d.ts +9 -0
  18. package/dist/cjs/react/hooks/useConfig.d.ts.map +1 -0
  19. package/dist/cjs/react/hooks/useConfig.js +116 -0
  20. package/dist/cjs/react/hooks/useConfig.js.map +1 -0
  21. package/dist/cjs/react/hooks/useGrainAnalytics.d.ts +6 -0
  22. package/dist/cjs/react/hooks/useGrainAnalytics.d.ts.map +1 -0
  23. package/dist/cjs/react/hooks/useGrainAnalytics.js +50 -0
  24. package/dist/cjs/react/hooks/useGrainAnalytics.js.map +1 -0
  25. package/dist/cjs/react/hooks/useTrack.d.ts +9 -0
  26. package/dist/cjs/react/hooks/useTrack.d.ts.map +1 -0
  27. package/dist/cjs/react/hooks/useTrack.js +53 -0
  28. package/dist/cjs/react/hooks/useTrack.js.map +1 -0
  29. package/dist/cjs/react/index.d.ts +36 -0
  30. package/dist/cjs/react/index.d.ts.map +1 -0
  31. package/dist/cjs/react/index.js +45 -0
  32. package/dist/cjs/react/index.js.map +1 -0
  33. package/dist/cjs/react/types.d.ts +33 -0
  34. package/dist/cjs/react/types.d.ts.map +1 -0
  35. package/dist/cjs/react/types.js +6 -0
  36. package/dist/cjs/react/types.js.map +1 -0
  37. package/dist/esm/index.d.ts +35 -2
  38. package/dist/esm/index.d.ts.map +1 -1
  39. package/dist/esm/index.js.map +1 -1
  40. package/dist/esm/react/GrainProvider.d.ts +11 -0
  41. package/dist/esm/react/GrainProvider.d.ts.map +1 -0
  42. package/dist/esm/react/GrainProvider.js +43 -0
  43. package/dist/esm/react/GrainProvider.js.map +1 -0
  44. package/dist/esm/react/context.d.ts +11 -0
  45. package/dist/esm/react/context.d.ts.map +1 -0
  46. package/dist/esm/react/context.js +7 -0
  47. package/dist/esm/react/context.js.map +1 -0
  48. package/dist/esm/react/hooks/useAllConfigs.d.ts +8 -0
  49. package/dist/esm/react/hooks/useAllConfigs.d.ts.map +1 -0
  50. package/dist/esm/react/hooks/useAllConfigs.js +76 -0
  51. package/dist/esm/react/hooks/useAllConfigs.js.map +1 -0
  52. package/dist/esm/react/hooks/useConfig.d.ts +9 -0
  53. package/dist/esm/react/hooks/useConfig.d.ts.map +1 -0
  54. package/dist/esm/react/hooks/useConfig.js +80 -0
  55. package/dist/esm/react/hooks/useConfig.js.map +1 -0
  56. package/dist/esm/react/hooks/useGrainAnalytics.d.ts +6 -0
  57. package/dist/esm/react/hooks/useGrainAnalytics.d.ts.map +1 -0
  58. package/dist/esm/react/hooks/useGrainAnalytics.js +14 -0
  59. package/dist/esm/react/hooks/useGrainAnalytics.js.map +1 -0
  60. package/dist/esm/react/hooks/useTrack.d.ts +9 -0
  61. package/dist/esm/react/hooks/useTrack.d.ts.map +1 -0
  62. package/dist/esm/react/hooks/useTrack.js +17 -0
  63. package/dist/esm/react/hooks/useTrack.js.map +1 -0
  64. package/dist/esm/react/index.d.ts +36 -0
  65. package/dist/esm/react/index.d.ts.map +1 -0
  66. package/dist/esm/react/index.js +37 -0
  67. package/dist/esm/react/index.js.map +1 -0
  68. package/dist/esm/react/types.d.ts +33 -0
  69. package/dist/esm/react/types.d.ts.map +1 -0
  70. package/dist/esm/react/types.js +5 -0
  71. package/dist/esm/react/types.js.map +1 -0
  72. package/dist/index.d.ts +35 -2
  73. package/dist/index.d.ts.map +1 -1
  74. package/dist/index.global.dev.js +124 -14
  75. package/dist/index.global.dev.js.map +2 -2
  76. package/dist/index.global.js +2 -2
  77. package/dist/index.global.js.map +3 -3
  78. package/dist/index.js +147 -15
  79. package/dist/index.mjs +147 -15
  80. package/dist/react/index.d.ts +405 -0
  81. package/dist/react/index.d.ts.map +1 -0
  82. package/dist/react/index.js +1181 -0
  83. package/dist/react/index.mjs +1176 -0
  84. package/dist/react/react/GrainProvider.d.ts +11 -0
  85. package/dist/react/react/GrainProvider.d.ts.map +1 -0
  86. package/dist/react/react/GrainProvider.js +45 -0
  87. package/dist/react/react/GrainProvider.mjs +42 -0
  88. package/dist/react/react/context.d.ts +11 -0
  89. package/dist/react/react/context.d.ts.map +1 -0
  90. package/dist/react/react/context.js +9 -0
  91. package/dist/react/react/context.mjs +6 -0
  92. package/dist/react/react/hooks/useAllConfigs.d.ts +8 -0
  93. package/dist/react/react/hooks/useAllConfigs.d.ts.map +1 -0
  94. package/dist/react/react/hooks/useAllConfigs.js +78 -0
  95. package/dist/react/react/hooks/useAllConfigs.mjs +75 -0
  96. package/dist/react/react/hooks/useConfig.d.ts +9 -0
  97. package/dist/react/react/hooks/useConfig.d.ts.map +1 -0
  98. package/dist/react/react/hooks/useConfig.js +82 -0
  99. package/dist/react/react/hooks/useConfig.mjs +79 -0
  100. package/dist/react/react/hooks/useGrainAnalytics.d.ts +6 -0
  101. package/dist/react/react/hooks/useGrainAnalytics.d.ts.map +1 -0
  102. package/dist/react/react/hooks/useGrainAnalytics.js +16 -0
  103. package/dist/react/react/hooks/useGrainAnalytics.mjs +13 -0
  104. package/dist/react/react/hooks/useTrack.d.ts +9 -0
  105. package/dist/react/react/hooks/useTrack.d.ts.map +1 -0
  106. package/dist/react/react/hooks/useTrack.js +19 -0
  107. package/dist/react/react/hooks/useTrack.mjs +16 -0
  108. package/dist/react/react/index.d.ts +36 -0
  109. package/dist/react/react/index.d.ts.map +1 -0
  110. package/dist/react/react/index.js +44 -0
  111. package/dist/react/react/index.mjs +36 -0
  112. package/dist/react/react/types.d.ts +33 -0
  113. package/dist/react/react/types.d.ts.map +1 -0
  114. package/dist/react/react/types.js +5 -0
  115. package/dist/react/react/types.mjs +4 -0
  116. package/package.json +20 -2
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../src/index.ts"],
4
- "sourcesContent": ["/**\n * Grain Analytics Web SDK\n * A lightweight, dependency-free TypeScript SDK for sending analytics events to Grain's REST API\n */\n\nexport interface GrainEvent {\n eventName: string;\n userId?: string;\n properties?: Record<string, unknown>;\n timestamp?: Date;\n}\n\nexport interface EventPayload {\n eventName: string;\n userId: string;\n properties: Record<string, unknown>;\n}\n\nexport type AuthStrategy = 'NONE' | 'SERVER_SIDE' | 'JWT';\n\nexport interface AuthProvider {\n getToken(): Promise<string> | string;\n}\n\nexport interface GrainConfig {\n tenantId: string;\n apiUrl?: string;\n authStrategy?: AuthStrategy;\n secretKey?: string; // For SERVER_SIDE auth\n authProvider?: AuthProvider; // For JWT auth\n userId?: string; // Global user ID for all events\n batchSize?: number;\n flushInterval?: number; // milliseconds\n retryAttempts?: number;\n retryDelay?: number; // milliseconds\n maxEventsPerRequest?: number; // Maximum events to send in a single API request\n debug?: boolean;\n // Remote Config options\n defaultConfigurations?: Record<string, string>; // Default values for configurations\n configCacheKey?: string; // Custom cache key for configurations\n configRefreshInterval?: number; // Auto-refresh interval in milliseconds (default: 5 minutes)\n enableConfigCache?: boolean; // Enable/disable configuration caching (default: true)\n}\n\nexport interface SendEventOptions {\n flush?: boolean; // Force immediate send\n}\n\nexport interface SetPropertyOptions {\n userId?: string; // Override global userId\n}\n\nexport interface PropertyPayload {\n userId: string;\n [key: string]: string; // All property values must be strings\n}\n\n// Remote Config interfaces\nexport interface RemoteConfigRequest {\n userId: string;\n immediateKeys: string[];\n properties?: Record<string, string>;\n}\n\nexport interface RemoteConfigResponse {\n userId: string;\n snapshotId: string;\n configurations: Record<string, string>;\n isFinal: boolean;\n qualifiedSegments: string[];\n qualifiedRuleSets: string[];\n timestamp: string;\n isFromCache: boolean;\n}\n\nexport interface RemoteConfigOptions {\n immediateKeys?: string[];\n properties?: Record<string, string>;\n userId?: string; // Override global userId\n forceRefresh?: boolean; // Force fetch from API, bypass cache\n}\n\nexport interface RemoteConfigCache {\n configurations: Record<string, string>;\n snapshotId: string;\n timestamp: string;\n userId: string;\n}\n\nexport type ConfigChangeListener = (configurations: Record<string, string>) => void;\n\n// Template event interfaces\nexport interface LoginEventProperties extends Record<string, unknown> {\n method?: string; // 'email', 'google', 'facebook', etc.\n success?: boolean;\n errorMessage?: string;\n loginAttempt?: number;\n rememberMe?: boolean;\n twoFactorEnabled?: boolean;\n}\n\nexport interface SignupEventProperties extends Record<string, unknown> {\n method?: string; // 'email', 'google', 'facebook', etc.\n source?: string; // 'landing_page', 'referral', 'ad', etc.\n plan?: string; // 'free', 'pro', 'enterprise', etc.\n success?: boolean;\n errorMessage?: string;\n}\n\nexport interface CheckoutEventProperties extends Record<string, unknown> {\n orderId?: string;\n total?: number;\n currency?: string;\n items?: Array<{\n id: string;\n name: string;\n price: number;\n quantity: number;\n }>;\n paymentMethod?: string; // 'credit_card', 'paypal', 'stripe', etc.\n success?: boolean;\n errorMessage?: string;\n couponCode?: string;\n discount?: number;\n}\n\nexport interface PageViewEventProperties extends Record<string, unknown> {\n page?: string;\n title?: string;\n referrer?: string;\n url?: string;\n userAgent?: string;\n screenResolution?: string;\n viewportSize?: string;\n}\n\nexport interface PurchaseEventProperties extends Record<string, unknown> {\n orderId?: string;\n total?: number;\n currency?: string;\n items?: Array<{\n id: string;\n name: string;\n price: number;\n quantity: number;\n category?: string;\n }>;\n paymentMethod?: string;\n shippingMethod?: string;\n tax?: number;\n shipping?: number;\n discount?: number;\n couponCode?: string;\n}\n\nexport interface SearchEventProperties extends Record<string, unknown> {\n query?: string;\n results?: number;\n filters?: Record<string, unknown>;\n sortBy?: string;\n category?: string;\n success?: boolean;\n}\n\nexport interface AddToCartEventProperties extends Record<string, unknown> {\n itemId?: string;\n itemName?: string;\n price?: number;\n quantity?: number;\n currency?: string;\n category?: string;\n variant?: string;\n}\n\nexport interface RemoveFromCartEventProperties extends Record<string, unknown> {\n itemId?: string;\n itemName?: string;\n price?: number;\n quantity?: number;\n currency?: string;\n category?: string;\n variant?: string;\n}\n\n// Error handling interfaces\nexport interface ErrorDigest {\n eventCount: number;\n totalProperties: number;\n totalSize: number;\n eventNames: string[];\n userIds: string[];\n}\n\nexport interface FormattedError {\n code: string;\n message: string;\n digest: ErrorDigest;\n timestamp: string;\n context: string;\n originalError?: unknown;\n}\n\n/**\n * Main Grain Analytics client\n */\ntype RequiredConfig = Required<Omit<GrainConfig, 'secretKey' | 'authProvider' | 'userId'>> & {\n secretKey?: string;\n authProvider?: AuthProvider;\n userId?: string;\n};\n\nexport class GrainAnalytics {\n private config: RequiredConfig;\n private eventQueue: EventPayload[] = [];\n private flushTimer: number | null = null;\n private isDestroyed = false;\n private globalUserId: string | null = null;\n private persistentAnonymousUserId: string | null = null;\n // Remote Config properties\n private configCache: RemoteConfigCache | null = null;\n private configRefreshTimer: number | null = null;\n private configChangeListeners: ConfigChangeListener[] = [];\n private configFetchPromise: Promise<RemoteConfigResponse> | null = null;\n\n constructor(config: GrainConfig) {\n this.config = {\n apiUrl: 'https://api.grainql.com',\n authStrategy: 'NONE',\n batchSize: 50,\n flushInterval: 5000, // 5 seconds\n retryAttempts: 3,\n retryDelay: 1000, // 1 second\n maxEventsPerRequest: 160, // Maximum events per API request\n debug: false,\n // Remote Config defaults\n defaultConfigurations: {},\n configCacheKey: 'grain_config',\n configRefreshInterval: 300000, // 5 minutes\n enableConfigCache: true,\n ...config,\n tenantId: config.tenantId,\n };\n\n // Set global userId if provided in config\n if (config.userId) {\n this.globalUserId = config.userId;\n }\n\n this.validateConfig();\n this.initializePersistentAnonymousUserId();\n this.setupBeforeUnload();\n this.startFlushTimer();\n this.initializeConfigCache();\n }\n\n private validateConfig(): void {\n if (!this.config.tenantId) {\n throw new Error('Grain Analytics: tenantId is required');\n }\n\n if (this.config.authStrategy === 'SERVER_SIDE' && !this.config.secretKey) {\n throw new Error('Grain Analytics: secretKey is required for SERVER_SIDE auth strategy');\n }\n\n if (this.config.authStrategy === 'JWT' && !this.config.authProvider) {\n throw new Error('Grain Analytics: authProvider is required for JWT auth strategy');\n }\n }\n\n /**\n * Generate a UUID v4 string\n */\n private generateUUID(): string {\n if (typeof crypto !== 'undefined' && crypto.randomUUID) {\n return crypto.randomUUID();\n }\n \n // Fallback for environments without crypto.randomUUID\n return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function(c) {\n const r = Math.random() * 16 | 0;\n const v = c === 'x' ? r : (r & 0x3 | 0x8);\n return v.toString(16);\n });\n }\n\n /**\n * Format UUID for anonymous user ID (remove dashes and prefix with 'temp:')\n */\n private formatAnonymousUserId(uuid: string): string {\n return `temp:${uuid.replace(/-/g, '')}`;\n }\n\n /**\n * Initialize persistent anonymous user ID from localStorage or create new one\n */\n private initializePersistentAnonymousUserId(): void {\n if (typeof window === 'undefined') return;\n\n const storageKey = `grain_anonymous_user_id_${this.config.tenantId}`;\n \n try {\n const stored = localStorage.getItem(storageKey);\n if (stored) {\n this.persistentAnonymousUserId = stored;\n this.log('Loaded persistent anonymous user ID:', this.persistentAnonymousUserId);\n } else {\n // Generate new anonymous user ID\n const uuid = this.generateUUID();\n this.persistentAnonymousUserId = this.formatAnonymousUserId(uuid);\n localStorage.setItem(storageKey, this.persistentAnonymousUserId);\n this.log('Generated new persistent anonymous user ID:', this.persistentAnonymousUserId);\n }\n } catch (error) {\n this.log('Failed to initialize persistent anonymous user ID:', error);\n // Fallback: generate temporary ID without persistence\n const uuid = this.generateUUID();\n this.persistentAnonymousUserId = this.formatAnonymousUserId(uuid);\n }\n }\n\n /**\n * Get the effective user ID (global userId or persistent anonymous ID)\n */\n private getEffectiveUserId(): string {\n return this.globalUserId || this.persistentAnonymousUserId || 'anonymous';\n }\n\n private log(...args: unknown[]): void {\n if (this.config.debug) {\n console.log('[Grain Analytics]', ...args);\n }\n }\n\n /**\n * Create error digest from events\n */\n private createErrorDigest(events: EventPayload[]): ErrorDigest {\n const eventNames = [...new Set(events.map(e => e.eventName))];\n const userIds = [...new Set(events.map(e => e.userId))];\n \n let totalProperties = 0;\n let totalSize = 0;\n \n events.forEach(event => {\n const properties = event.properties || {};\n totalProperties += Object.keys(properties).length;\n totalSize += JSON.stringify(event).length;\n });\n\n return {\n eventCount: events.length,\n totalProperties,\n totalSize,\n eventNames,\n userIds,\n };\n }\n\n /**\n * Format error with beautiful structure\n */\n private formatError(\n error: unknown,\n context: string,\n events?: EventPayload[]\n ): FormattedError {\n const digest = events ? this.createErrorDigest(events) : {\n eventCount: 0,\n totalProperties: 0,\n totalSize: 0,\n eventNames: [],\n userIds: [],\n };\n\n let code = 'UNKNOWN_ERROR';\n let message = 'An unknown error occurred';\n\n if (error instanceof Error) {\n message = error.message;\n \n // Determine error code based on error type and message\n if (message.includes('fetch failed') || message.includes('network error')) {\n code = 'NETWORK_ERROR';\n } else if (message.includes('timeout')) {\n code = 'TIMEOUT_ERROR';\n } else if (message.includes('HTTP 4')) {\n code = 'CLIENT_ERROR';\n } else if (message.includes('HTTP 5')) {\n code = 'SERVER_ERROR';\n } else if (message.includes('JSON')) {\n code = 'PARSE_ERROR';\n } else if (message.includes('auth') || message.includes('unauthorized')) {\n code = 'AUTH_ERROR';\n } else if (message.includes('rate limit') || message.includes('429')) {\n code = 'RATE_LIMIT_ERROR';\n } else {\n code = 'GENERAL_ERROR';\n }\n } else if (typeof error === 'string') {\n message = error;\n code = 'STRING_ERROR';\n } else if (error && typeof error === 'object' && 'status' in error) {\n const status = (error as { status: number }).status;\n code = `HTTP_${status}`;\n message = `HTTP ${status} error`;\n }\n\n return {\n code,\n message,\n digest,\n timestamp: new Date().toISOString(),\n context,\n originalError: error,\n };\n }\n\n /**\n * Log formatted error gracefully\n */\n private logError(formattedError: FormattedError): void {\n const { code, message, digest, timestamp, context } = formattedError;\n \n const errorOutput = {\n '\uD83D\uDEA8 Grain Analytics Error': {\n 'Error Code': code,\n 'Message': message,\n 'Context': context,\n 'Timestamp': timestamp,\n 'Event Digest': {\n 'Events': digest.eventCount,\n 'Properties': digest.totalProperties,\n 'Size (bytes)': digest.totalSize,\n 'Event Names': digest.eventNames.length > 0 ? digest.eventNames.join(', ') : 'None',\n 'User IDs': digest.userIds.length > 0 ? digest.userIds.slice(0, 3).join(', ') + (digest.userIds.length > 3 ? '...' : '') : 'None',\n }\n }\n };\n\n console.error('\uD83D\uDEA8 Grain Analytics Error:', errorOutput);\n \n // Also log in a more compact format for debugging\n if (this.config.debug) {\n console.error(`[Grain Analytics] ${code}: ${message} (${context}) - Events: ${digest.eventCount}, Props: ${digest.totalProperties}, Size: ${digest.totalSize}B`);\n }\n }\n\n /**\n * Safely execute a function with error handling\n */\n private async safeExecute<T>(\n fn: () => Promise<T>,\n context: string,\n events?: EventPayload[]\n ): Promise<T | null> {\n try {\n return await fn();\n } catch (error) {\n const formattedError = this.formatError(error, context, events);\n this.logError(formattedError);\n return null;\n }\n }\n\n private formatEvent(event: GrainEvent): EventPayload {\n return {\n eventName: event.eventName,\n userId: event.userId || this.getEffectiveUserId(),\n properties: event.properties || {},\n };\n }\n\n private async getAuthHeaders(): Promise<Record<string, string>> {\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n };\n\n switch (this.config.authStrategy) {\n case 'NONE':\n break;\n case 'SERVER_SIDE':\n headers['Authorization'] = `Chase ${this.config.secretKey}`;\n break;\n case 'JWT':\n if (this.config.authProvider) {\n const token = await this.config.authProvider.getToken();\n headers['Authorization'] = `Bearer ${token}`;\n }\n break;\n }\n\n return headers;\n }\n\n private async delay(ms: number): Promise<void> {\n return new Promise(resolve => setTimeout(resolve, ms));\n }\n\n private isRetriableError(error: unknown): boolean {\n if (error instanceof Error) {\n // Check for specific network or fetch errors\n const message = error.message.toLowerCase();\n if (message.includes('fetch failed')) return true;\n if (message === 'network error') return true; // Exact match to avoid \"Non-network error\"\n if (message.includes('timeout')) return true;\n if (message.includes('connection')) return true;\n }\n \n // Check for HTTP status codes that are retriable\n if (typeof error === 'object' && error !== null && 'status' in error) {\n const status = (error as { status: number }).status;\n return status >= 500 || status === 429; // Server errors or rate limiting\n }\n \n return false;\n }\n\n private async sendEvents(events: EventPayload[]): Promise<void> {\n if (events.length === 0) return;\n\n let lastError: unknown;\n\n for (let attempt = 0; attempt <= this.config.retryAttempts; attempt++) {\n try {\n const headers = await this.getAuthHeaders();\n const url = `${this.config.apiUrl}/v1/events/${encodeURIComponent(this.config.tenantId)}/multi`;\n\n this.log(`Sending ${events.length} events to ${url} (attempt ${attempt + 1})`);\n\n const response = await fetch(url, {\n method: 'POST',\n headers,\n body: JSON.stringify(events),\n });\n\n if (!response.ok) {\n let errorMessage = `HTTP ${response.status}`;\n try {\n const errorBody = await response.json();\n if (errorBody?.message) {\n errorMessage = errorBody.message;\n }\n } catch {\n const errorText = await response.text();\n if (errorText) {\n errorMessage = errorText;\n }\n }\n \n const error = new Error(`Failed to send events: ${errorMessage}`) as Error & { status?: number };\n error.status = response.status;\n throw error;\n }\n\n this.log(`Successfully sent ${events.length} events`);\n return; // Success, exit retry loop\n \n } catch (error) {\n lastError = error;\n \n if (attempt === this.config.retryAttempts) {\n // Last attempt, don't retry - log error gracefully\n const formattedError = this.formatError(error, `sendEvents (attempt ${attempt + 1}/${this.config.retryAttempts + 1})`, events);\n this.logError(formattedError);\n return; // Don't throw, just return gracefully\n }\n \n if (!this.isRetriableError(error)) {\n // Non-retriable error, don't retry - log error gracefully\n const formattedError = this.formatError(error, `sendEvents (non-retriable error)`, events);\n this.logError(formattedError);\n return; // Don't throw, just return gracefully\n }\n \n const delayMs = this.config.retryDelay * Math.pow(2, attempt); // Exponential backoff\n this.log(`Retrying in ${delayMs}ms after error:`, error);\n await this.delay(delayMs);\n }\n }\n }\n\n private async sendEventsWithBeacon(events: EventPayload[]): Promise<void> {\n if (events.length === 0) return;\n\n try {\n const headers = await this.getAuthHeaders();\n const url = `${this.config.apiUrl}/v1/events/${encodeURIComponent(this.config.tenantId)}/multi`;\n\n const body = JSON.stringify({ events });\n\n // Try beacon API first (more reliable for page unload)\n if (typeof navigator !== 'undefined' && 'sendBeacon' in navigator) {\n const blob = new Blob([body], { type: 'application/json' });\n const success = navigator.sendBeacon(url, blob);\n \n if (success) {\n this.log(`Successfully sent ${events.length} events via beacon`);\n return;\n }\n }\n\n // Fallback to fetch with keepalive\n await fetch(url, {\n method: 'POST',\n headers,\n body,\n keepalive: true,\n });\n\n this.log(`Successfully sent ${events.length} events via fetch (keepalive)`);\n } catch (error) {\n // Log error gracefully for beacon failures (page unload scenarios)\n const formattedError = this.formatError(error, 'sendEventsWithBeacon', events);\n this.logError(formattedError);\n }\n }\n\n private startFlushTimer(): void {\n if (this.flushTimer) {\n clearInterval(this.flushTimer);\n }\n\n this.flushTimer = window.setInterval(() => {\n if (this.eventQueue.length > 0) {\n this.flush().catch((error) => {\n const formattedError = this.formatError(error, 'auto-flush');\n this.logError(formattedError);\n });\n }\n }, this.config.flushInterval);\n }\n\n private setupBeforeUnload(): void {\n if (typeof window === 'undefined') return;\n\n const handleBeforeUnload = () => {\n if (this.eventQueue.length > 0) {\n // Use beacon API for reliable delivery during page unload\n const eventsToSend = [...this.eventQueue];\n this.eventQueue = [];\n \n const chunks = this.chunkEvents(eventsToSend, this.config.maxEventsPerRequest);\n \n // Send first chunk with beacon (most important for page unload)\n if (chunks.length > 0) {\n this.sendEventsWithBeacon(chunks[0]).catch(() => {\n // Silently fail - page is unloading\n });\n }\n }\n };\n\n // Handle page unload\n window.addEventListener('beforeunload', handleBeforeUnload);\n window.addEventListener('pagehide', handleBeforeUnload);\n \n // Handle visibility change (page hidden)\n document.addEventListener('visibilitychange', () => {\n if (document.visibilityState === 'hidden' && this.eventQueue.length > 0) {\n const eventsToSend = [...this.eventQueue];\n this.eventQueue = [];\n \n const chunks = this.chunkEvents(eventsToSend, this.config.maxEventsPerRequest);\n \n // Send first chunk with beacon (most important for page hidden)\n if (chunks.length > 0) {\n this.sendEventsWithBeacon(chunks[0]).catch(() => {\n // Silently fail\n });\n }\n }\n });\n }\n\n /**\n * Track an analytics event\n */\n async track(eventName: string, properties?: Record<string, unknown>, options?: SendEventOptions): Promise<void>;\n async track(event: GrainEvent, options?: SendEventOptions): Promise<void>;\n async track(\n eventOrName: string | GrainEvent,\n propertiesOrOptions?: Record<string, unknown> | SendEventOptions,\n options?: SendEventOptions\n ): Promise<void> {\n try {\n if (this.isDestroyed) {\n const error = new Error('Grain Analytics: Client has been destroyed');\n const formattedError = this.formatError(error, 'track (client destroyed)');\n this.logError(formattedError);\n return;\n }\n\n let event: GrainEvent;\n let opts: SendEventOptions = {};\n\n if (typeof eventOrName === 'string') {\n event = {\n eventName: eventOrName,\n properties: propertiesOrOptions as Record<string, unknown>,\n };\n opts = options || {};\n } else {\n event = eventOrName;\n opts = propertiesOrOptions as SendEventOptions || {};\n }\n\n const formattedEvent = this.formatEvent(event);\n this.eventQueue.push(formattedEvent);\n\n this.log(`Queued event: ${event.eventName}`, event.properties);\n\n // Check if we should flush immediately\n if (opts.flush || this.eventQueue.length >= this.config.batchSize) {\n await this.flush();\n }\n } catch (error) {\n const formattedError = this.formatError(error, 'track');\n this.logError(formattedError);\n }\n }\n\n /**\n * Identify a user (sets userId for subsequent events)\n */\n identify(userId: string): void {\n this.log(`Identified user: ${userId}`);\n this.globalUserId = userId;\n // Clear persistent anonymous user ID since we now have a real user ID\n this.persistentAnonymousUserId = null;\n }\n\n /**\n * Set global user ID for all subsequent events\n */\n setUserId(userId: string | null): void {\n this.log(`Set global user ID: ${userId}`);\n this.globalUserId = userId;\n // Clear persistent anonymous user ID if setting a real user ID\n if (userId) {\n this.persistentAnonymousUserId = null;\n }\n }\n\n /**\n * Get current global user ID\n */\n getUserId(): string | null {\n return this.globalUserId;\n }\n\n /**\n * Get current effective user ID (global userId or persistent anonymous ID)\n */\n getEffectiveUserIdPublic(): string {\n return this.getEffectiveUserId();\n }\n\n /**\n * Set user properties\n */\n async setProperty(properties: Record<string, unknown>, options?: SetPropertyOptions): Promise<void> {\n try {\n if (this.isDestroyed) {\n const error = new Error('Grain Analytics: Client has been destroyed');\n const formattedError = this.formatError(error, 'setProperty (client destroyed)');\n this.logError(formattedError);\n return;\n }\n\n const userId = options?.userId || this.getEffectiveUserId();\n \n // Validate property count (max 4 properties)\n const propertyKeys = Object.keys(properties);\n if (propertyKeys.length > 4) {\n const error = new Error('Grain Analytics: Maximum 4 properties allowed per request');\n const formattedError = this.formatError(error, 'setProperty (validation)');\n this.logError(formattedError);\n return;\n }\n\n if (propertyKeys.length === 0) {\n const error = new Error('Grain Analytics: At least one property is required');\n const formattedError = this.formatError(error, 'setProperty (validation)');\n this.logError(formattedError);\n return;\n }\n\n // Serialize all values to strings\n const serializedProperties: Record<string, string> = {};\n for (const [key, value] of Object.entries(properties)) {\n if (value === null || value === undefined) {\n serializedProperties[key] = '';\n } else if (typeof value === 'string') {\n serializedProperties[key] = value;\n } else {\n serializedProperties[key] = JSON.stringify(value);\n }\n }\n\n const payload: PropertyPayload = {\n userId,\n ...serializedProperties,\n };\n\n await this.sendProperties(payload);\n } catch (error) {\n const formattedError = this.formatError(error, 'setProperty');\n this.logError(formattedError);\n }\n }\n\n /**\n * Send properties to the API\n */\n private async sendProperties(payload: PropertyPayload): Promise<void> {\n let lastError: unknown;\n\n for (let attempt = 0; attempt <= this.config.retryAttempts; attempt++) {\n try {\n const headers = await this.getAuthHeaders();\n const url = `${this.config.apiUrl}/v1/events/${encodeURIComponent(this.config.tenantId)}/properties`;\n\n this.log(`Setting properties for user ${payload.userId} (attempt ${attempt + 1})`);\n\n const response = await fetch(url, {\n method: 'POST',\n headers,\n body: JSON.stringify(payload),\n });\n\n if (!response.ok) {\n let errorMessage = `HTTP ${response.status}`;\n try {\n const errorBody = await response.json();\n if (errorBody?.message) {\n errorMessage = errorBody.message;\n }\n } catch {\n const errorText = await response.text();\n if (errorText) {\n errorMessage = errorText;\n }\n }\n \n const error = new Error(`Failed to set properties: ${errorMessage}`) as Error & { status?: number };\n error.status = response.status;\n throw error;\n }\n\n this.log(`Successfully set properties for user ${payload.userId}`);\n return; // Success, exit retry loop\n \n } catch (error) {\n lastError = error;\n \n if (attempt === this.config.retryAttempts) {\n // Last attempt, don't retry - log error gracefully\n const formattedError = this.formatError(error, `sendProperties (attempt ${attempt + 1}/${this.config.retryAttempts + 1})`);\n this.logError(formattedError);\n return; // Don't throw, just return gracefully\n }\n \n if (!this.isRetriableError(error)) {\n // Non-retriable error, don't retry - log error gracefully\n const formattedError = this.formatError(error, 'sendProperties (non-retriable error)');\n this.logError(formattedError);\n return; // Don't throw, just return gracefully\n }\n \n const delayMs = this.config.retryDelay * Math.pow(2, attempt); // Exponential backoff\n this.log(`Retrying in ${delayMs}ms after error:`, error);\n await this.delay(delayMs);\n }\n }\n }\n\n // Template event methods\n\n /**\n * Track user login event\n */\n async trackLogin(properties?: LoginEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('login', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackLogin');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track user signup event\n */\n async trackSignup(properties?: SignupEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('signup', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackSignup');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track checkout event\n */\n async trackCheckout(properties?: CheckoutEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('checkout', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackCheckout');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track page view event\n */\n async trackPageView(properties?: PageViewEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('page_view', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackPageView');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track purchase event\n */\n async trackPurchase(properties?: PurchaseEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('purchase', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackPurchase');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track search event\n */\n async trackSearch(properties?: SearchEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('search', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackSearch');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track add to cart event\n */\n async trackAddToCart(properties?: AddToCartEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('add_to_cart', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackAddToCart');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track remove from cart event\n */\n async trackRemoveFromCart(properties?: RemoveFromCartEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('remove_from_cart', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackRemoveFromCart');\n this.logError(formattedError);\n }\n }\n\n /**\n * Manually flush all queued events\n */\n async flush(): Promise<void> {\n try {\n if (this.eventQueue.length === 0) return;\n\n const eventsToSend = [...this.eventQueue];\n this.eventQueue = [];\n\n // Split events into chunks to respect maxEventsPerRequest limit\n const chunks = this.chunkEvents(eventsToSend, this.config.maxEventsPerRequest);\n \n // Send all chunks sequentially to maintain order\n for (const chunk of chunks) {\n await this.sendEvents(chunk);\n }\n } catch (error) {\n const formattedError = this.formatError(error, 'flush');\n this.logError(formattedError);\n }\n }\n\n // Remote Config Methods\n\n /**\n * Initialize configuration cache from localStorage\n */\n private initializeConfigCache(): void {\n if (!this.config.enableConfigCache || typeof window === 'undefined') return;\n\n try {\n const cached = localStorage.getItem(this.config.configCacheKey);\n if (cached) {\n this.configCache = JSON.parse(cached);\n this.log('Loaded configuration from cache:', this.configCache);\n }\n } catch (error) {\n this.log('Failed to load configuration cache:', error);\n }\n }\n\n /**\n * Save configuration cache to localStorage\n */\n private saveConfigCache(cache: RemoteConfigCache): void {\n if (!this.config.enableConfigCache || typeof window === 'undefined') return;\n\n try {\n localStorage.setItem(this.config.configCacheKey, JSON.stringify(cache));\n this.log('Saved configuration to cache:', cache);\n } catch (error) {\n this.log('Failed to save configuration cache:', error);\n }\n }\n\n /**\n * Get configuration value with fallback to defaults\n */\n getConfig(key: string): string | undefined {\n // First check cache\n if (this.configCache?.configurations?.[key]) {\n return this.configCache.configurations[key];\n }\n\n // Then check defaults\n if (this.config.defaultConfigurations?.[key]) {\n return this.config.defaultConfigurations[key];\n }\n\n return undefined;\n }\n\n /**\n * Get all configurations with fallback to defaults\n */\n getAllConfigs(): Record<string, string> {\n const configs = { ...this.config.defaultConfigurations };\n \n if (this.configCache?.configurations) {\n Object.assign(configs, this.configCache.configurations);\n }\n\n return configs;\n }\n\n /**\n * Fetch configurations from API\n */\n async fetchConfig(options: RemoteConfigOptions = {}): Promise<RemoteConfigResponse | null> {\n try {\n if (this.isDestroyed) {\n const error = new Error('Grain Analytics: Client has been destroyed');\n const formattedError = this.formatError(error, 'fetchConfig (client destroyed)');\n this.logError(formattedError);\n return null;\n }\n\n const userId = options.userId || this.getEffectiveUserId();\n const immediateKeys = options.immediateKeys || [];\n const properties = options.properties || {};\n\n const request: RemoteConfigRequest = {\n userId,\n immediateKeys,\n properties,\n };\n\n let lastError: unknown;\n\n for (let attempt = 0; attempt <= this.config.retryAttempts; attempt++) {\n try {\n const headers = await this.getAuthHeaders();\n const url = `${this.config.apiUrl}/v1/client/${encodeURIComponent(this.config.tenantId)}/config/configurations`;\n\n this.log(`Fetching configurations for user ${userId} (attempt ${attempt + 1})`);\n\n const response = await fetch(url, {\n method: 'POST',\n headers,\n body: JSON.stringify(request),\n });\n\n if (!response.ok) {\n let errorMessage = `HTTP ${response.status}`;\n try {\n const errorBody = await response.json();\n if (errorBody?.message) {\n errorMessage = errorBody.message;\n }\n } catch {\n const errorText = await response.text();\n if (errorText) {\n errorMessage = errorText;\n }\n }\n \n const error = new Error(`Failed to fetch configurations: ${errorMessage}`) as Error & { status?: number };\n error.status = response.status;\n throw error;\n }\n\n const configResponse: RemoteConfigResponse = await response.json();\n \n // Update cache if successful\n if (configResponse.configurations) {\n this.updateConfigCache(configResponse, userId);\n }\n\n this.log(`Successfully fetched configurations for user ${userId}:`, configResponse);\n return configResponse;\n \n } catch (error) {\n lastError = error;\n \n if (attempt === this.config.retryAttempts) {\n // Last attempt, don't retry - log error gracefully\n const formattedError = this.formatError(error, `fetchConfig (attempt ${attempt + 1}/${this.config.retryAttempts + 1})`);\n this.logError(formattedError);\n return null;\n }\n \n if (!this.isRetriableError(error)) {\n // Non-retriable error, don't retry - log error gracefully\n const formattedError = this.formatError(error, 'fetchConfig (non-retriable error)');\n this.logError(formattedError);\n return null;\n }\n \n const delayMs = this.config.retryDelay * Math.pow(2, attempt);\n this.log(`Retrying config fetch in ${delayMs}ms after error:`, error);\n await this.delay(delayMs);\n }\n }\n\n return null;\n } catch (error) {\n const formattedError = this.formatError(error, 'fetchConfig');\n this.logError(formattedError);\n return null;\n }\n }\n\n /**\n * Get configuration asynchronously (cache-first with fallback to API)\n */\n async getConfigAsync(key: string, options: RemoteConfigOptions = {}): Promise<string | undefined> {\n try {\n // Return immediately if we have it in cache and not forcing refresh\n if (!options.forceRefresh && this.configCache?.configurations?.[key]) {\n return this.configCache.configurations[key];\n }\n\n // Return default if available and not forcing refresh\n if (!options.forceRefresh && this.config.defaultConfigurations?.[key]) {\n return this.config.defaultConfigurations[key];\n }\n\n // Fetch from API\n const response = await this.fetchConfig(options);\n if (response) {\n return response.configurations[key];\n }\n \n // Return default as fallback\n return this.config.defaultConfigurations?.[key];\n } catch (error) {\n const formattedError = this.formatError(error, 'getConfigAsync');\n this.logError(formattedError);\n // Return default as fallback\n return this.config.defaultConfigurations?.[key];\n }\n }\n\n /**\n * Get all configurations asynchronously (cache-first with fallback to API)\n */\n async getAllConfigsAsync(options: RemoteConfigOptions = {}): Promise<Record<string, string>> {\n try {\n // Return cache if available and not forcing refresh\n if (!options.forceRefresh && this.configCache?.configurations) {\n return { ...this.config.defaultConfigurations, ...this.configCache.configurations };\n }\n\n // Fetch from API\n const response = await this.fetchConfig(options);\n if (response) {\n return { ...this.config.defaultConfigurations, ...response.configurations };\n }\n \n // Return defaults as fallback\n return { ...this.config.defaultConfigurations };\n } catch (error) {\n const formattedError = this.formatError(error, 'getAllConfigsAsync');\n this.logError(formattedError);\n // Return defaults as fallback\n return { ...this.config.defaultConfigurations };\n }\n }\n\n /**\n * Update configuration cache and notify listeners\n */\n private updateConfigCache(response: RemoteConfigResponse, userId: string): void {\n const newCache: RemoteConfigCache = {\n configurations: response.configurations,\n snapshotId: response.snapshotId,\n timestamp: response.timestamp,\n userId,\n };\n\n const oldConfigs = this.configCache?.configurations || {};\n this.configCache = newCache;\n this.saveConfigCache(newCache);\n\n // Notify listeners if configurations changed\n if (JSON.stringify(oldConfigs) !== JSON.stringify(response.configurations)) {\n this.notifyConfigChangeListeners(response.configurations);\n }\n }\n\n /**\n * Add configuration change listener\n */\n addConfigChangeListener(listener: ConfigChangeListener): void {\n this.configChangeListeners.push(listener);\n }\n\n /**\n * Remove configuration change listener\n */\n removeConfigChangeListener(listener: ConfigChangeListener): void {\n const index = this.configChangeListeners.indexOf(listener);\n if (index > -1) {\n this.configChangeListeners.splice(index, 1);\n }\n }\n\n /**\n * Notify all configuration change listeners\n */\n private notifyConfigChangeListeners(configurations: Record<string, string>): void {\n this.configChangeListeners.forEach(listener => {\n try {\n listener(configurations);\n } catch (error) {\n console.error('[Grain Analytics] Config change listener error:', error);\n }\n });\n }\n\n /**\n * Start automatic configuration refresh timer\n */\n private startConfigRefreshTimer(): void {\n if (this.configRefreshTimer) {\n clearInterval(this.configRefreshTimer);\n }\n\n this.configRefreshTimer = window.setInterval(() => {\n if (!this.isDestroyed && this.globalUserId) {\n this.fetchConfig().catch((error) => {\n const formattedError = this.formatError(error, 'auto-config refresh');\n this.logError(formattedError);\n });\n }\n }, this.config.configRefreshInterval);\n }\n\n /**\n * Stop automatic configuration refresh timer\n */\n private stopConfigRefreshTimer(): void {\n if (this.configRefreshTimer) {\n clearInterval(this.configRefreshTimer);\n this.configRefreshTimer = null;\n }\n }\n\n /**\n * Preload configurations for immediate access\n */\n async preloadConfig(immediateKeys: string[] = [], properties?: Record<string, string>): Promise<void> {\n try {\n if (!this.globalUserId) {\n this.log('Cannot preload config: no user ID set');\n return;\n }\n\n const response = await this.fetchConfig({ immediateKeys, properties });\n if (response) {\n this.startConfigRefreshTimer();\n }\n } catch (error) {\n const formattedError = this.formatError(error, 'preloadConfig');\n this.logError(formattedError);\n }\n }\n\n /**\n * Split events array into chunks of specified size\n */\n private chunkEvents(events: EventPayload[], chunkSize: number): EventPayload[][] {\n const chunks: EventPayload[][] = [];\n for (let i = 0; i < events.length; i += chunkSize) {\n chunks.push(events.slice(i, i + chunkSize));\n }\n return chunks;\n }\n\n /**\n * Destroy the client and clean up resources\n */\n destroy(): void {\n this.isDestroyed = true;\n \n if (this.flushTimer) {\n clearInterval(this.flushTimer);\n this.flushTimer = null;\n }\n\n // Stop config refresh timer\n this.stopConfigRefreshTimer();\n\n // Clear config change listeners\n this.configChangeListeners = [];\n\n // Send any remaining events (in chunks if necessary)\n if (this.eventQueue.length > 0) {\n const eventsToSend = [...this.eventQueue];\n this.eventQueue = [];\n \n const chunks = this.chunkEvents(eventsToSend, this.config.maxEventsPerRequest);\n \n // Send first chunk with beacon (most important for page unload)\n if (chunks.length > 0) {\n this.sendEventsWithBeacon(chunks[0]).catch(() => {\n // Silently fail during cleanup\n });\n \n // If there are more chunks, try to send them with regular fetch\n for (let i = 1; i < chunks.length; i++) {\n this.sendEventsWithBeacon(chunks[i]).catch(() => {\n // Silently fail during cleanup\n });\n }\n }\n }\n }\n}\n\n/**\n * Create a new Grain Analytics client\n */\nexport function createGrainAnalytics(config: GrainConfig): GrainAnalytics {\n return new GrainAnalytics(config);\n}\n\n// Default export for convenience\nexport default GrainAnalytics;\n\n// Global interface for IIFE build\ndeclare global {\n interface Window {\n Grain?: {\n GrainAnalytics: typeof GrainAnalytics;\n createGrainAnalytics: typeof createGrainAnalytics;\n };\n }\n}\n\n// Auto-setup for IIFE build\nif (typeof window !== 'undefined') {\n window.Grain = {\n GrainAnalytics,\n createGrainAnalytics,\n };\n}"],
5
- "mappings": "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",
6
- "names": ["src_exports", "__export", "GrainAnalytics", "createGrainAnalytics", "src_default", "config", "c", "r", "uuid", "storageKey", "stored", "error", "args", "events", "eventNames", "e", "userIds", "totalProperties", "totalSize", "event", "properties", "context", "digest", "code", "message", "status", "formattedError", "timestamp", "errorOutput", "fn", "headers", "token", "ms", "resolve", "lastError", "attempt", "url", "response", "errorMessage", "errorBody", "errorText", "delayMs", "body", "blob", "handleBeforeUnload", "eventsToSend", "chunks", "eventOrName", "propertiesOrOptions", "options", "opts", "formattedEvent", "userId", "propertyKeys", "serializedProperties", "key", "value", "payload", "chunk", "cached", "cache", "configs", "immediateKeys", "request", "configResponse", "newCache", "oldConfigs", "listener", "index", "configurations", "chunkSize", "i"]
4
+ "sourcesContent": ["/**\n * Grain Analytics Web SDK\n * A lightweight, dependency-free TypeScript SDK for sending analytics events to Grain's REST API\n */\n\nexport interface GrainEvent {\n eventName: string;\n userId?: string;\n properties?: Record<string, unknown>;\n timestamp?: Date;\n}\n\nexport interface EventPayload {\n eventName: string;\n userId: string;\n properties: Record<string, unknown>;\n}\n\nexport type AuthStrategy = 'NONE' | 'SERVER_SIDE' | 'JWT';\n\nexport interface AuthProvider {\n getToken(): Promise<string> | string;\n}\n\nexport interface GrainConfig {\n tenantId: string;\n apiUrl?: string;\n authStrategy?: AuthStrategy;\n secretKey?: string; // For SERVER_SIDE auth\n authProvider?: AuthProvider; // For JWT auth\n userId?: string; // Global user ID for all events\n batchSize?: number;\n flushInterval?: number; // milliseconds\n retryAttempts?: number;\n retryDelay?: number; // milliseconds\n maxEventsPerRequest?: number; // Maximum events to send in a single API request\n debug?: boolean;\n // Remote Config options\n defaultConfigurations?: Record<string, string>; // Default values for configurations\n configCacheKey?: string; // Custom cache key for configurations\n configRefreshInterval?: number; // Auto-refresh interval in milliseconds (default: 5 minutes)\n enableConfigCache?: boolean; // Enable/disable configuration caching (default: true)\n}\n\nexport interface SendEventOptions {\n flush?: boolean; // Force immediate send\n}\n\nexport interface SetPropertyOptions {\n userId?: string; // Override global userId\n}\n\nexport interface LoginOptions {\n userId?: string; // User ID to set\n authToken?: string; // Auth token to set for JWT auth\n authStrategy?: AuthStrategy; // Override auth strategy\n}\n\nexport interface PropertyPayload {\n userId: string;\n [key: string]: string; // All property values must be strings\n}\n\n// Remote Config interfaces\nexport interface RemoteConfigRequest {\n userId: string;\n immediateKeys: string[];\n properties?: Record<string, string>;\n}\n\nexport interface RemoteConfigResponse {\n userId: string;\n snapshotId: string;\n configurations: Record<string, string>;\n isFinal: boolean;\n qualifiedSegments: string[];\n qualifiedRuleSets: string[];\n timestamp: string;\n isFromCache: boolean;\n}\n\nexport interface RemoteConfigOptions {\n immediateKeys?: string[];\n properties?: Record<string, string>;\n userId?: string; // Override global userId\n forceRefresh?: boolean; // Force fetch from API, bypass cache\n}\n\nexport interface RemoteConfigCache {\n configurations: Record<string, string>;\n snapshotId: string;\n timestamp: string;\n userId: string;\n}\n\nexport type ConfigChangeListener = (configurations: Record<string, string>) => void;\n\n// Template event interfaces\nexport interface LoginEventProperties extends Record<string, unknown> {\n method?: string; // 'email', 'google', 'facebook', etc.\n success?: boolean;\n errorMessage?: string;\n loginAttempt?: number;\n rememberMe?: boolean;\n twoFactorEnabled?: boolean;\n}\n\nexport interface SignupEventProperties extends Record<string, unknown> {\n method?: string; // 'email', 'google', 'facebook', etc.\n source?: string; // 'landing_page', 'referral', 'ad', etc.\n plan?: string; // 'free', 'pro', 'enterprise', etc.\n success?: boolean;\n errorMessage?: string;\n}\n\nexport interface CheckoutEventProperties extends Record<string, unknown> {\n orderId?: string;\n total?: number;\n currency?: string;\n items?: Array<{\n id: string;\n name: string;\n price: number;\n quantity: number;\n }>;\n paymentMethod?: string; // 'credit_card', 'paypal', 'stripe', etc.\n success?: boolean;\n errorMessage?: string;\n couponCode?: string;\n discount?: number;\n}\n\nexport interface PageViewEventProperties extends Record<string, unknown> {\n page?: string;\n title?: string;\n referrer?: string;\n url?: string;\n userAgent?: string;\n screenResolution?: string;\n viewportSize?: string;\n}\n\nexport interface PurchaseEventProperties extends Record<string, unknown> {\n orderId?: string;\n total?: number;\n currency?: string;\n items?: Array<{\n id: string;\n name: string;\n price: number;\n quantity: number;\n category?: string;\n }>;\n paymentMethod?: string;\n shippingMethod?: string;\n tax?: number;\n shipping?: number;\n discount?: number;\n couponCode?: string;\n}\n\nexport interface SearchEventProperties extends Record<string, unknown> {\n query?: string;\n results?: number;\n filters?: Record<string, unknown>;\n sortBy?: string;\n category?: string;\n success?: boolean;\n}\n\nexport interface AddToCartEventProperties extends Record<string, unknown> {\n itemId?: string;\n itemName?: string;\n price?: number;\n quantity?: number;\n currency?: string;\n category?: string;\n variant?: string;\n}\n\nexport interface RemoveFromCartEventProperties extends Record<string, unknown> {\n itemId?: string;\n itemName?: string;\n price?: number;\n quantity?: number;\n currency?: string;\n category?: string;\n variant?: string;\n}\n\n// Error handling interfaces\nexport interface ErrorDigest {\n eventCount: number;\n totalProperties: number;\n totalSize: number;\n eventNames: string[];\n userIds: string[];\n}\n\nexport interface FormattedError {\n code: string;\n message: string;\n digest: ErrorDigest;\n timestamp: string;\n context: string;\n originalError?: unknown;\n}\n\n/**\n * Main Grain Analytics client\n * \n * Features:\n * - Automatic UUIDv4 generation for anonymous users\n * - Login/logout functionality for dynamic auth token injection\n * - Persistent anonymous user ID across sessions\n * - Support for multiple auth strategies (NONE, SERVER_SIDE, JWT)\n */\ntype RequiredConfig = Required<Omit<GrainConfig, 'secretKey' | 'authProvider' | 'userId'>> & {\n secretKey?: string;\n authProvider?: AuthProvider;\n userId?: string;\n};\n\nexport class GrainAnalytics {\n private config: RequiredConfig;\n private eventQueue: EventPayload[] = [];\n private flushTimer: number | null = null;\n private isDestroyed = false;\n private globalUserId: string | null = null;\n private persistentAnonymousUserId: string | null = null;\n // Remote Config properties\n private configCache: RemoteConfigCache | null = null;\n private configRefreshTimer: number | null = null;\n private configChangeListeners: ConfigChangeListener[] = [];\n private configFetchPromise: Promise<RemoteConfigResponse> | null = null;\n\n constructor(config: GrainConfig) {\n this.config = {\n apiUrl: 'https://api.grainql.com',\n authStrategy: 'NONE',\n batchSize: 50,\n flushInterval: 5000, // 5 seconds\n retryAttempts: 3,\n retryDelay: 1000, // 1 second\n maxEventsPerRequest: 160, // Maximum events per API request\n debug: false,\n // Remote Config defaults\n defaultConfigurations: {},\n configCacheKey: 'grain_config',\n configRefreshInterval: 300000, // 5 minutes\n enableConfigCache: true,\n ...config,\n tenantId: config.tenantId,\n };\n\n // Set global userId if provided in config\n if (config.userId) {\n this.globalUserId = config.userId;\n }\n\n this.validateConfig();\n this.initializePersistentAnonymousUserId();\n this.setupBeforeUnload();\n this.startFlushTimer();\n this.initializeConfigCache();\n }\n\n private validateConfig(): void {\n if (!this.config.tenantId) {\n throw new Error('Grain Analytics: tenantId is required');\n }\n\n if (this.config.authStrategy === 'SERVER_SIDE' && !this.config.secretKey) {\n throw new Error('Grain Analytics: secretKey is required for SERVER_SIDE auth strategy');\n }\n\n if (this.config.authStrategy === 'JWT' && !this.config.authProvider) {\n throw new Error('Grain Analytics: authProvider is required for JWT auth strategy');\n }\n }\n\n /**\n * Generate a UUID v4 string\n */\n private generateUUID(): string {\n if (typeof crypto !== 'undefined' && crypto.randomUUID) {\n return crypto.randomUUID();\n }\n \n // Fallback for environments without crypto.randomUUID\n return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function(c) {\n const r = Math.random() * 16 | 0;\n const v = c === 'x' ? r : (r & 0x3 | 0x8);\n return v.toString(16);\n });\n }\n\n /**\n * Generate a proper UUIDv4 identifier for anonymous user ID\n */\n private generateAnonymousUserId(): string {\n return this.generateUUID();\n }\n\n /**\n * Initialize persistent anonymous user ID from localStorage or create new one\n */\n private initializePersistentAnonymousUserId(): void {\n if (typeof window === 'undefined') return;\n\n const storageKey = `grain_anonymous_user_id_${this.config.tenantId}`;\n \n try {\n const stored = localStorage.getItem(storageKey);\n if (stored) {\n this.persistentAnonymousUserId = stored;\n this.log('Loaded persistent anonymous user ID:', this.persistentAnonymousUserId);\n } else {\n // Generate new UUIDv4 anonymous user ID\n this.persistentAnonymousUserId = this.generateAnonymousUserId();\n localStorage.setItem(storageKey, this.persistentAnonymousUserId);\n this.log('Generated new persistent anonymous user ID:', this.persistentAnonymousUserId);\n }\n } catch (error) {\n this.log('Failed to initialize persistent anonymous user ID:', error);\n // Fallback: generate temporary ID without persistence\n this.persistentAnonymousUserId = this.generateAnonymousUserId();\n }\n }\n\n /**\n * Get the effective user ID (global userId or persistent anonymous ID)\n */\n private getEffectiveUserId(): string {\n if (this.globalUserId) {\n return this.globalUserId;\n }\n \n if (this.persistentAnonymousUserId) {\n return this.persistentAnonymousUserId;\n }\n \n // Generate a new UUIDv4 identifier as fallback\n this.persistentAnonymousUserId = this.generateAnonymousUserId();\n \n // Try to persist it\n if (typeof window !== 'undefined') {\n try {\n const storageKey = `grain_anonymous_user_id_${this.config.tenantId}`;\n localStorage.setItem(storageKey, this.persistentAnonymousUserId);\n } catch (error) {\n this.log('Failed to persist generated anonymous user ID:', error);\n }\n }\n \n return this.persistentAnonymousUserId;\n }\n\n private log(...args: unknown[]): void {\n if (this.config.debug) {\n console.log('[Grain Analytics]', ...args);\n }\n }\n\n /**\n * Create error digest from events\n */\n private createErrorDigest(events: EventPayload[]): ErrorDigest {\n const eventNames = [...new Set(events.map(e => e.eventName))];\n const userIds = [...new Set(events.map(e => e.userId))];\n \n let totalProperties = 0;\n let totalSize = 0;\n \n events.forEach(event => {\n const properties = event.properties || {};\n totalProperties += Object.keys(properties).length;\n totalSize += JSON.stringify(event).length;\n });\n\n return {\n eventCount: events.length,\n totalProperties,\n totalSize,\n eventNames,\n userIds,\n };\n }\n\n /**\n * Format error with beautiful structure\n */\n private formatError(\n error: unknown,\n context: string,\n events?: EventPayload[]\n ): FormattedError {\n const digest = events ? this.createErrorDigest(events) : {\n eventCount: 0,\n totalProperties: 0,\n totalSize: 0,\n eventNames: [],\n userIds: [],\n };\n\n let code = 'UNKNOWN_ERROR';\n let message = 'An unknown error occurred';\n\n if (error instanceof Error) {\n message = error.message;\n \n // Determine error code based on error type and message\n if (message.includes('fetch failed') || message.includes('network error')) {\n code = 'NETWORK_ERROR';\n } else if (message.includes('timeout')) {\n code = 'TIMEOUT_ERROR';\n } else if (message.includes('HTTP 4')) {\n code = 'CLIENT_ERROR';\n } else if (message.includes('HTTP 5')) {\n code = 'SERVER_ERROR';\n } else if (message.includes('JSON')) {\n code = 'PARSE_ERROR';\n } else if (message.includes('auth') || message.includes('unauthorized')) {\n code = 'AUTH_ERROR';\n } else if (message.includes('rate limit') || message.includes('429')) {\n code = 'RATE_LIMIT_ERROR';\n } else {\n code = 'GENERAL_ERROR';\n }\n } else if (typeof error === 'string') {\n message = error;\n code = 'STRING_ERROR';\n } else if (error && typeof error === 'object' && 'status' in error) {\n const status = (error as { status: number }).status;\n code = `HTTP_${status}`;\n message = `HTTP ${status} error`;\n }\n\n return {\n code,\n message,\n digest,\n timestamp: new Date().toISOString(),\n context,\n originalError: error,\n };\n }\n\n /**\n * Log formatted error gracefully\n */\n private logError(formattedError: FormattedError): void {\n const { code, message, digest, timestamp, context } = formattedError;\n \n const errorOutput = {\n '\uD83D\uDEA8 Grain Analytics Error': {\n 'Error Code': code,\n 'Message': message,\n 'Context': context,\n 'Timestamp': timestamp,\n 'Event Digest': {\n 'Events': digest.eventCount,\n 'Properties': digest.totalProperties,\n 'Size (bytes)': digest.totalSize,\n 'Event Names': digest.eventNames.length > 0 ? digest.eventNames.join(', ') : 'None',\n 'User IDs': digest.userIds.length > 0 ? digest.userIds.slice(0, 3).join(', ') + (digest.userIds.length > 3 ? '...' : '') : 'None',\n }\n }\n };\n\n console.error('\uD83D\uDEA8 Grain Analytics Error:', errorOutput);\n \n // Also log in a more compact format for debugging\n if (this.config.debug) {\n console.error(`[Grain Analytics] ${code}: ${message} (${context}) - Events: ${digest.eventCount}, Props: ${digest.totalProperties}, Size: ${digest.totalSize}B`);\n }\n }\n\n /**\n * Safely execute a function with error handling\n */\n private async safeExecute<T>(\n fn: () => Promise<T>,\n context: string,\n events?: EventPayload[]\n ): Promise<T | null> {\n try {\n return await fn();\n } catch (error) {\n const formattedError = this.formatError(error, context, events);\n this.logError(formattedError);\n return null;\n }\n }\n\n private formatEvent(event: GrainEvent): EventPayload {\n return {\n eventName: event.eventName,\n userId: event.userId || this.getEffectiveUserId(),\n properties: event.properties || {},\n };\n }\n\n private async getAuthHeaders(): Promise<Record<string, string>> {\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n };\n\n switch (this.config.authStrategy) {\n case 'NONE':\n break;\n case 'SERVER_SIDE':\n headers['Authorization'] = `Chase ${this.config.secretKey}`;\n break;\n case 'JWT':\n if (this.config.authProvider) {\n const token = await this.config.authProvider.getToken();\n headers['Authorization'] = `Bearer ${token}`;\n }\n break;\n }\n\n return headers;\n }\n\n private async delay(ms: number): Promise<void> {\n return new Promise(resolve => setTimeout(resolve, ms));\n }\n\n private isRetriableError(error: unknown): boolean {\n if (error instanceof Error) {\n // Check for specific network or fetch errors\n const message = error.message.toLowerCase();\n if (message.includes('fetch failed')) return true;\n if (message === 'network error') return true; // Exact match to avoid \"Non-network error\"\n if (message.includes('timeout')) return true;\n if (message.includes('connection')) return true;\n }\n \n // Check for HTTP status codes that are retriable\n if (typeof error === 'object' && error !== null && 'status' in error) {\n const status = (error as { status: number }).status;\n return status >= 500 || status === 429; // Server errors or rate limiting\n }\n \n return false;\n }\n\n private async sendEvents(events: EventPayload[]): Promise<void> {\n if (events.length === 0) return;\n\n let lastError: unknown;\n\n for (let attempt = 0; attempt <= this.config.retryAttempts; attempt++) {\n try {\n const headers = await this.getAuthHeaders();\n const url = `${this.config.apiUrl}/v1/events/${encodeURIComponent(this.config.tenantId)}/multi`;\n\n this.log(`Sending ${events.length} events to ${url} (attempt ${attempt + 1})`);\n\n const response = await fetch(url, {\n method: 'POST',\n headers,\n body: JSON.stringify(events),\n });\n\n if (!response.ok) {\n let errorMessage = `HTTP ${response.status}`;\n try {\n const errorBody = await response.json();\n if (errorBody?.message) {\n errorMessage = errorBody.message;\n }\n } catch {\n const errorText = await response.text();\n if (errorText) {\n errorMessage = errorText;\n }\n }\n \n const error = new Error(`Failed to send events: ${errorMessage}`) as Error & { status?: number };\n error.status = response.status;\n throw error;\n }\n\n this.log(`Successfully sent ${events.length} events`);\n return; // Success, exit retry loop\n \n } catch (error) {\n lastError = error;\n \n if (attempt === this.config.retryAttempts) {\n // Last attempt, don't retry - log error gracefully\n const formattedError = this.formatError(error, `sendEvents (attempt ${attempt + 1}/${this.config.retryAttempts + 1})`, events);\n this.logError(formattedError);\n return; // Don't throw, just return gracefully\n }\n \n if (!this.isRetriableError(error)) {\n // Non-retriable error, don't retry - log error gracefully\n const formattedError = this.formatError(error, `sendEvents (non-retriable error)`, events);\n this.logError(formattedError);\n return; // Don't throw, just return gracefully\n }\n \n const delayMs = this.config.retryDelay * Math.pow(2, attempt); // Exponential backoff\n this.log(`Retrying in ${delayMs}ms after error:`, error);\n await this.delay(delayMs);\n }\n }\n }\n\n private async sendEventsWithBeacon(events: EventPayload[]): Promise<void> {\n if (events.length === 0) return;\n\n try {\n const headers = await this.getAuthHeaders();\n const url = `${this.config.apiUrl}/v1/events/${encodeURIComponent(this.config.tenantId)}/multi`;\n\n const body = JSON.stringify({ events });\n\n // Try beacon API first (more reliable for page unload)\n if (typeof navigator !== 'undefined' && 'sendBeacon' in navigator) {\n const blob = new Blob([body], { type: 'application/json' });\n const success = navigator.sendBeacon(url, blob);\n \n if (success) {\n this.log(`Successfully sent ${events.length} events via beacon`);\n return;\n }\n }\n\n // Fallback to fetch with keepalive\n await fetch(url, {\n method: 'POST',\n headers,\n body,\n keepalive: true,\n });\n\n this.log(`Successfully sent ${events.length} events via fetch (keepalive)`);\n } catch (error) {\n // Log error gracefully for beacon failures (page unload scenarios)\n const formattedError = this.formatError(error, 'sendEventsWithBeacon', events);\n this.logError(formattedError);\n }\n }\n\n private startFlushTimer(): void {\n if (this.flushTimer) {\n clearInterval(this.flushTimer);\n }\n\n this.flushTimer = window.setInterval(() => {\n if (this.eventQueue.length > 0) {\n this.flush().catch((error) => {\n const formattedError = this.formatError(error, 'auto-flush');\n this.logError(formattedError);\n });\n }\n }, this.config.flushInterval);\n }\n\n private setupBeforeUnload(): void {\n if (typeof window === 'undefined') return;\n\n const handleBeforeUnload = () => {\n if (this.eventQueue.length > 0) {\n // Use beacon API for reliable delivery during page unload\n const eventsToSend = [...this.eventQueue];\n this.eventQueue = [];\n \n const chunks = this.chunkEvents(eventsToSend, this.config.maxEventsPerRequest);\n \n // Send first chunk with beacon (most important for page unload)\n if (chunks.length > 0) {\n this.sendEventsWithBeacon(chunks[0]).catch(() => {\n // Silently fail - page is unloading\n });\n }\n }\n };\n\n // Handle page unload\n window.addEventListener('beforeunload', handleBeforeUnload);\n window.addEventListener('pagehide', handleBeforeUnload);\n \n // Handle visibility change (page hidden)\n document.addEventListener('visibilitychange', () => {\n if (document.visibilityState === 'hidden' && this.eventQueue.length > 0) {\n const eventsToSend = [...this.eventQueue];\n this.eventQueue = [];\n \n const chunks = this.chunkEvents(eventsToSend, this.config.maxEventsPerRequest);\n \n // Send first chunk with beacon (most important for page hidden)\n if (chunks.length > 0) {\n this.sendEventsWithBeacon(chunks[0]).catch(() => {\n // Silently fail\n });\n }\n }\n });\n }\n\n /**\n * Track an analytics event\n */\n async track(eventName: string, properties?: Record<string, unknown>, options?: SendEventOptions): Promise<void>;\n async track(event: GrainEvent, options?: SendEventOptions): Promise<void>;\n async track(\n eventOrName: string | GrainEvent,\n propertiesOrOptions?: Record<string, unknown> | SendEventOptions,\n options?: SendEventOptions\n ): Promise<void> {\n try {\n if (this.isDestroyed) {\n const error = new Error('Grain Analytics: Client has been destroyed');\n const formattedError = this.formatError(error, 'track (client destroyed)');\n this.logError(formattedError);\n return;\n }\n\n let event: GrainEvent;\n let opts: SendEventOptions = {};\n\n if (typeof eventOrName === 'string') {\n event = {\n eventName: eventOrName,\n properties: propertiesOrOptions as Record<string, unknown>,\n };\n opts = options || {};\n } else {\n event = eventOrName;\n opts = propertiesOrOptions as SendEventOptions || {};\n }\n\n const formattedEvent = this.formatEvent(event);\n this.eventQueue.push(formattedEvent);\n\n this.log(`Queued event: ${event.eventName}`, event.properties);\n\n // Check if we should flush immediately\n if (opts.flush || this.eventQueue.length >= this.config.batchSize) {\n await this.flush();\n }\n } catch (error) {\n const formattedError = this.formatError(error, 'track');\n this.logError(formattedError);\n }\n }\n\n /**\n * Identify a user (sets userId for subsequent events)\n */\n identify(userId: string): void {\n this.log(`Identified user: ${userId}`);\n this.globalUserId = userId;\n // Clear persistent anonymous user ID since we now have a real user ID\n this.persistentAnonymousUserId = null;\n }\n\n /**\n * Set global user ID for all subsequent events\n */\n setUserId(userId: string | null): void {\n this.log(`Set global user ID: ${userId}`);\n this.globalUserId = userId;\n \n if (userId) {\n // Clear persistent anonymous user ID if setting a real user ID\n this.persistentAnonymousUserId = null;\n } else {\n // If clearing user ID, ensure we have a UUIDv4 identifier\n if (!this.persistentAnonymousUserId) {\n this.persistentAnonymousUserId = this.generateAnonymousUserId();\n \n // Try to persist the new anonymous ID\n if (typeof window !== 'undefined') {\n try {\n const storageKey = `grain_anonymous_user_id_${this.config.tenantId}`;\n localStorage.setItem(storageKey, this.persistentAnonymousUserId);\n } catch (error) {\n this.log('Failed to persist new anonymous user ID:', error);\n }\n }\n }\n }\n }\n\n /**\n * Get current global user ID\n */\n getUserId(): string | null {\n return this.globalUserId;\n }\n\n /**\n * Get current effective user ID (global userId or persistent anonymous ID)\n */\n getEffectiveUserIdPublic(): string {\n return this.getEffectiveUserId();\n }\n\n /**\n * Login with auth token or userId on the fly\n * \n * @example\n * // Login with userId only\n * client.login({ userId: 'user123' });\n * \n * // Login with auth token (automatically sets authStrategy to JWT)\n * client.login({ authToken: 'jwt-token-here' });\n * \n * // Login with both userId and auth token\n * client.login({ userId: 'user123', authToken: 'jwt-token-here' });\n * \n * // Override auth strategy\n * client.login({ userId: 'user123', authStrategy: 'SERVER_SIDE' });\n */\n login(options: LoginOptions): void {\n try {\n if (this.isDestroyed) {\n const error = new Error('Grain Analytics: Client has been destroyed');\n const formattedError = this.formatError(error, 'login (client destroyed)');\n this.logError(formattedError);\n return;\n }\n\n // Set userId if provided\n if (options.userId) {\n this.log(`Login: Setting user ID to ${options.userId}`);\n this.globalUserId = options.userId;\n // Clear persistent anonymous user ID since we now have a real user ID\n this.persistentAnonymousUserId = null;\n }\n\n // Handle auth token if provided\n if (options.authToken) {\n this.log('Login: Setting auth token');\n // Update auth strategy to JWT if not already set\n if (this.config.authStrategy === 'NONE') {\n this.config.authStrategy = 'JWT';\n }\n \n // Create a simple auth provider that returns the provided token\n this.config.authProvider = {\n getToken: () => options.authToken!\n };\n }\n\n // Override auth strategy if provided\n if (options.authStrategy) {\n this.log(`Login: Setting auth strategy to ${options.authStrategy}`);\n this.config.authStrategy = options.authStrategy;\n }\n\n this.log(`Login successful. Effective user ID: ${this.getEffectiveUserId()}`);\n } catch (error) {\n const formattedError = this.formatError(error, 'login');\n this.logError(formattedError);\n }\n }\n\n /**\n * Logout and clear user session, fall back to UUIDv4 identifier\n * \n * @example\n * // Logout user and return to anonymous mode\n * client.logout();\n * \n * // After logout, events will use the persistent UUIDv4 identifier\n * client.track('page_view', { page: 'home' });\n */\n logout(): void {\n try {\n if (this.isDestroyed) {\n const error = new Error('Grain Analytics: Client has been destroyed');\n const formattedError = this.formatError(error, 'logout (client destroyed)');\n this.logError(formattedError);\n return;\n }\n\n this.log('Logout: Clearing user session');\n \n // Clear global user ID\n this.globalUserId = null;\n \n // Reset auth strategy to NONE\n this.config.authStrategy = 'NONE';\n this.config.authProvider = undefined;\n \n // Generate new UUIDv4 identifier if we don't have one\n if (!this.persistentAnonymousUserId) {\n this.persistentAnonymousUserId = this.generateAnonymousUserId();\n \n // Try to persist the new anonymous ID\n if (typeof window !== 'undefined') {\n try {\n const storageKey = `grain_anonymous_user_id_${this.config.tenantId}`;\n localStorage.setItem(storageKey, this.persistentAnonymousUserId);\n } catch (error) {\n this.log('Failed to persist new anonymous user ID after logout:', error);\n }\n }\n }\n \n this.log(`Logout successful. Effective user ID: ${this.getEffectiveUserId()}`);\n } catch (error) {\n const formattedError = this.formatError(error, 'logout');\n this.logError(formattedError);\n }\n }\n\n /**\n * Set user properties\n */\n async setProperty(properties: Record<string, unknown>, options?: SetPropertyOptions): Promise<void> {\n try {\n if (this.isDestroyed) {\n const error = new Error('Grain Analytics: Client has been destroyed');\n const formattedError = this.formatError(error, 'setProperty (client destroyed)');\n this.logError(formattedError);\n return;\n }\n\n const userId = options?.userId || this.getEffectiveUserId();\n \n // Validate property count (max 4 properties)\n const propertyKeys = Object.keys(properties);\n if (propertyKeys.length > 4) {\n const error = new Error('Grain Analytics: Maximum 4 properties allowed per request');\n const formattedError = this.formatError(error, 'setProperty (validation)');\n this.logError(formattedError);\n return;\n }\n\n if (propertyKeys.length === 0) {\n const error = new Error('Grain Analytics: At least one property is required');\n const formattedError = this.formatError(error, 'setProperty (validation)');\n this.logError(formattedError);\n return;\n }\n\n // Serialize all values to strings\n const serializedProperties: Record<string, string> = {};\n for (const [key, value] of Object.entries(properties)) {\n if (value === null || value === undefined) {\n serializedProperties[key] = '';\n } else if (typeof value === 'string') {\n serializedProperties[key] = value;\n } else {\n serializedProperties[key] = JSON.stringify(value);\n }\n }\n\n const payload: PropertyPayload = {\n userId,\n ...serializedProperties,\n };\n\n await this.sendProperties(payload);\n } catch (error) {\n const formattedError = this.formatError(error, 'setProperty');\n this.logError(formattedError);\n }\n }\n\n /**\n * Send properties to the API\n */\n private async sendProperties(payload: PropertyPayload): Promise<void> {\n let lastError: unknown;\n\n for (let attempt = 0; attempt <= this.config.retryAttempts; attempt++) {\n try {\n const headers = await this.getAuthHeaders();\n const url = `${this.config.apiUrl}/v1/events/${encodeURIComponent(this.config.tenantId)}/properties`;\n\n this.log(`Setting properties for user ${payload.userId} (attempt ${attempt + 1})`);\n\n const response = await fetch(url, {\n method: 'POST',\n headers,\n body: JSON.stringify(payload),\n });\n\n if (!response.ok) {\n let errorMessage = `HTTP ${response.status}`;\n try {\n const errorBody = await response.json();\n if (errorBody?.message) {\n errorMessage = errorBody.message;\n }\n } catch {\n const errorText = await response.text();\n if (errorText) {\n errorMessage = errorText;\n }\n }\n \n const error = new Error(`Failed to set properties: ${errorMessage}`) as Error & { status?: number };\n error.status = response.status;\n throw error;\n }\n\n this.log(`Successfully set properties for user ${payload.userId}`);\n return; // Success, exit retry loop\n \n } catch (error) {\n lastError = error;\n \n if (attempt === this.config.retryAttempts) {\n // Last attempt, don't retry - log error gracefully\n const formattedError = this.formatError(error, `sendProperties (attempt ${attempt + 1}/${this.config.retryAttempts + 1})`);\n this.logError(formattedError);\n return; // Don't throw, just return gracefully\n }\n \n if (!this.isRetriableError(error)) {\n // Non-retriable error, don't retry - log error gracefully\n const formattedError = this.formatError(error, 'sendProperties (non-retriable error)');\n this.logError(formattedError);\n return; // Don't throw, just return gracefully\n }\n \n const delayMs = this.config.retryDelay * Math.pow(2, attempt); // Exponential backoff\n this.log(`Retrying in ${delayMs}ms after error:`, error);\n await this.delay(delayMs);\n }\n }\n }\n\n // Template event methods\n\n /**\n * Track user login event\n */\n async trackLogin(properties?: LoginEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('login', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackLogin');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track user signup event\n */\n async trackSignup(properties?: SignupEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('signup', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackSignup');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track checkout event\n */\n async trackCheckout(properties?: CheckoutEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('checkout', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackCheckout');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track page view event\n */\n async trackPageView(properties?: PageViewEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('page_view', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackPageView');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track purchase event\n */\n async trackPurchase(properties?: PurchaseEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('purchase', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackPurchase');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track search event\n */\n async trackSearch(properties?: SearchEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('search', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackSearch');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track add to cart event\n */\n async trackAddToCart(properties?: AddToCartEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('add_to_cart', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackAddToCart');\n this.logError(formattedError);\n }\n }\n\n /**\n * Track remove from cart event\n */\n async trackRemoveFromCart(properties?: RemoveFromCartEventProperties, options?: SendEventOptions): Promise<void> {\n try {\n return await this.track('remove_from_cart', properties, options);\n } catch (error) {\n const formattedError = this.formatError(error, 'trackRemoveFromCart');\n this.logError(formattedError);\n }\n }\n\n /**\n * Manually flush all queued events\n */\n async flush(): Promise<void> {\n try {\n if (this.eventQueue.length === 0) return;\n\n const eventsToSend = [...this.eventQueue];\n this.eventQueue = [];\n\n // Split events into chunks to respect maxEventsPerRequest limit\n const chunks = this.chunkEvents(eventsToSend, this.config.maxEventsPerRequest);\n \n // Send all chunks sequentially to maintain order\n for (const chunk of chunks) {\n await this.sendEvents(chunk);\n }\n } catch (error) {\n const formattedError = this.formatError(error, 'flush');\n this.logError(formattedError);\n }\n }\n\n // Remote Config Methods\n\n /**\n * Initialize configuration cache from localStorage\n */\n private initializeConfigCache(): void {\n if (!this.config.enableConfigCache || typeof window === 'undefined') return;\n\n try {\n const cached = localStorage.getItem(this.config.configCacheKey);\n if (cached) {\n this.configCache = JSON.parse(cached);\n this.log('Loaded configuration from cache:', this.configCache);\n }\n } catch (error) {\n this.log('Failed to load configuration cache:', error);\n }\n }\n\n /**\n * Save configuration cache to localStorage\n */\n private saveConfigCache(cache: RemoteConfigCache): void {\n if (!this.config.enableConfigCache || typeof window === 'undefined') return;\n\n try {\n localStorage.setItem(this.config.configCacheKey, JSON.stringify(cache));\n this.log('Saved configuration to cache:', cache);\n } catch (error) {\n this.log('Failed to save configuration cache:', error);\n }\n }\n\n /**\n * Get configuration value with fallback to defaults\n */\n getConfig(key: string): string | undefined {\n // First check cache\n if (this.configCache?.configurations?.[key]) {\n return this.configCache.configurations[key];\n }\n\n // Then check defaults\n if (this.config.defaultConfigurations?.[key]) {\n return this.config.defaultConfigurations[key];\n }\n\n return undefined;\n }\n\n /**\n * Get all configurations with fallback to defaults\n */\n getAllConfigs(): Record<string, string> {\n const configs = { ...this.config.defaultConfigurations };\n \n if (this.configCache?.configurations) {\n Object.assign(configs, this.configCache.configurations);\n }\n\n return configs;\n }\n\n /**\n * Fetch configurations from API\n */\n async fetchConfig(options: RemoteConfigOptions = {}): Promise<RemoteConfigResponse | null> {\n try {\n if (this.isDestroyed) {\n const error = new Error('Grain Analytics: Client has been destroyed');\n const formattedError = this.formatError(error, 'fetchConfig (client destroyed)');\n this.logError(formattedError);\n return null;\n }\n\n const userId = options.userId || this.getEffectiveUserId();\n const immediateKeys = options.immediateKeys || [];\n const properties = options.properties || {};\n\n const request: RemoteConfigRequest = {\n userId,\n immediateKeys,\n properties,\n };\n\n let lastError: unknown;\n\n for (let attempt = 0; attempt <= this.config.retryAttempts; attempt++) {\n try {\n const headers = await this.getAuthHeaders();\n const url = `${this.config.apiUrl}/v1/client/${encodeURIComponent(this.config.tenantId)}/config/configurations`;\n\n this.log(`Fetching configurations for user ${userId} (attempt ${attempt + 1})`);\n\n const response = await fetch(url, {\n method: 'POST',\n headers,\n body: JSON.stringify(request),\n });\n\n if (!response.ok) {\n let errorMessage = `HTTP ${response.status}`;\n try {\n const errorBody = await response.json();\n if (errorBody?.message) {\n errorMessage = errorBody.message;\n }\n } catch {\n const errorText = await response.text();\n if (errorText) {\n errorMessage = errorText;\n }\n }\n \n const error = new Error(`Failed to fetch configurations: ${errorMessage}`) as Error & { status?: number };\n error.status = response.status;\n throw error;\n }\n\n const configResponse: RemoteConfigResponse = await response.json();\n \n // Update cache if successful\n if (configResponse.configurations) {\n this.updateConfigCache(configResponse, userId);\n }\n\n this.log(`Successfully fetched configurations for user ${userId}:`, configResponse);\n return configResponse;\n \n } catch (error) {\n lastError = error;\n \n if (attempt === this.config.retryAttempts) {\n // Last attempt, don't retry - log error gracefully\n const formattedError = this.formatError(error, `fetchConfig (attempt ${attempt + 1}/${this.config.retryAttempts + 1})`);\n this.logError(formattedError);\n return null;\n }\n \n if (!this.isRetriableError(error)) {\n // Non-retriable error, don't retry - log error gracefully\n const formattedError = this.formatError(error, 'fetchConfig (non-retriable error)');\n this.logError(formattedError);\n return null;\n }\n \n const delayMs = this.config.retryDelay * Math.pow(2, attempt);\n this.log(`Retrying config fetch in ${delayMs}ms after error:`, error);\n await this.delay(delayMs);\n }\n }\n\n return null;\n } catch (error) {\n const formattedError = this.formatError(error, 'fetchConfig');\n this.logError(formattedError);\n return null;\n }\n }\n\n /**\n * Get configuration asynchronously (cache-first with fallback to API)\n */\n async getConfigAsync(key: string, options: RemoteConfigOptions = {}): Promise<string | undefined> {\n try {\n // Return immediately if we have it in cache and not forcing refresh\n if (!options.forceRefresh && this.configCache?.configurations?.[key]) {\n return this.configCache.configurations[key];\n }\n\n // Return default if available and not forcing refresh\n if (!options.forceRefresh && this.config.defaultConfigurations?.[key]) {\n return this.config.defaultConfigurations[key];\n }\n\n // Fetch from API\n const response = await this.fetchConfig(options);\n if (response) {\n return response.configurations[key];\n }\n \n // Return default as fallback\n return this.config.defaultConfigurations?.[key];\n } catch (error) {\n const formattedError = this.formatError(error, 'getConfigAsync');\n this.logError(formattedError);\n // Return default as fallback\n return this.config.defaultConfigurations?.[key];\n }\n }\n\n /**\n * Get all configurations asynchronously (cache-first with fallback to API)\n */\n async getAllConfigsAsync(options: RemoteConfigOptions = {}): Promise<Record<string, string>> {\n try {\n // Return cache if available and not forcing refresh\n if (!options.forceRefresh && this.configCache?.configurations) {\n return { ...this.config.defaultConfigurations, ...this.configCache.configurations };\n }\n\n // Fetch from API\n const response = await this.fetchConfig(options);\n if (response) {\n return { ...this.config.defaultConfigurations, ...response.configurations };\n }\n \n // Return defaults as fallback\n return { ...this.config.defaultConfigurations };\n } catch (error) {\n const formattedError = this.formatError(error, 'getAllConfigsAsync');\n this.logError(formattedError);\n // Return defaults as fallback\n return { ...this.config.defaultConfigurations };\n }\n }\n\n /**\n * Update configuration cache and notify listeners\n */\n private updateConfigCache(response: RemoteConfigResponse, userId: string): void {\n const newCache: RemoteConfigCache = {\n configurations: response.configurations,\n snapshotId: response.snapshotId,\n timestamp: response.timestamp,\n userId,\n };\n\n const oldConfigs = this.configCache?.configurations || {};\n this.configCache = newCache;\n this.saveConfigCache(newCache);\n\n // Notify listeners if configurations changed\n if (JSON.stringify(oldConfigs) !== JSON.stringify(response.configurations)) {\n this.notifyConfigChangeListeners(response.configurations);\n }\n }\n\n /**\n * Add configuration change listener\n */\n addConfigChangeListener(listener: ConfigChangeListener): void {\n this.configChangeListeners.push(listener);\n }\n\n /**\n * Remove configuration change listener\n */\n removeConfigChangeListener(listener: ConfigChangeListener): void {\n const index = this.configChangeListeners.indexOf(listener);\n if (index > -1) {\n this.configChangeListeners.splice(index, 1);\n }\n }\n\n /**\n * Notify all configuration change listeners\n */\n private notifyConfigChangeListeners(configurations: Record<string, string>): void {\n this.configChangeListeners.forEach(listener => {\n try {\n listener(configurations);\n } catch (error) {\n console.error('[Grain Analytics] Config change listener error:', error);\n }\n });\n }\n\n /**\n * Start automatic configuration refresh timer\n */\n private startConfigRefreshTimer(): void {\n if (this.configRefreshTimer) {\n clearInterval(this.configRefreshTimer);\n }\n\n this.configRefreshTimer = window.setInterval(() => {\n if (!this.isDestroyed) {\n // Use effective userId (will be generated if not set)\n this.fetchConfig().catch((error) => {\n const formattedError = this.formatError(error, 'auto-config refresh');\n this.logError(formattedError);\n });\n }\n }, this.config.configRefreshInterval);\n }\n\n /**\n * Stop automatic configuration refresh timer\n */\n private stopConfigRefreshTimer(): void {\n if (this.configRefreshTimer) {\n clearInterval(this.configRefreshTimer);\n this.configRefreshTimer = null;\n }\n }\n\n /**\n * Preload configurations for immediate access\n */\n async preloadConfig(immediateKeys: string[] = [], properties?: Record<string, string>): Promise<void> {\n try {\n // Use effective userId (will be generated if not set)\n const effectiveUserId = this.getEffectiveUserId();\n this.log(`Preloading config for user: ${effectiveUserId}`);\n\n const response = await this.fetchConfig({ immediateKeys, properties });\n if (response) {\n this.startConfigRefreshTimer();\n }\n } catch (error) {\n const formattedError = this.formatError(error, 'preloadConfig');\n this.logError(formattedError);\n }\n }\n\n /**\n * Split events array into chunks of specified size\n */\n private chunkEvents(events: EventPayload[], chunkSize: number): EventPayload[][] {\n const chunks: EventPayload[][] = [];\n for (let i = 0; i < events.length; i += chunkSize) {\n chunks.push(events.slice(i, i + chunkSize));\n }\n return chunks;\n }\n\n /**\n * Destroy the client and clean up resources\n */\n destroy(): void {\n this.isDestroyed = true;\n \n if (this.flushTimer) {\n clearInterval(this.flushTimer);\n this.flushTimer = null;\n }\n\n // Stop config refresh timer\n this.stopConfigRefreshTimer();\n\n // Clear config change listeners\n this.configChangeListeners = [];\n\n // Send any remaining events (in chunks if necessary)\n if (this.eventQueue.length > 0) {\n const eventsToSend = [...this.eventQueue];\n this.eventQueue = [];\n \n const chunks = this.chunkEvents(eventsToSend, this.config.maxEventsPerRequest);\n \n // Send first chunk with beacon (most important for page unload)\n if (chunks.length > 0) {\n this.sendEventsWithBeacon(chunks[0]).catch(() => {\n // Silently fail during cleanup\n });\n \n // If there are more chunks, try to send them with regular fetch\n for (let i = 1; i < chunks.length; i++) {\n this.sendEventsWithBeacon(chunks[i]).catch(() => {\n // Silently fail during cleanup\n });\n }\n }\n }\n }\n}\n\n/**\n * Create a new Grain Analytics client\n */\nexport function createGrainAnalytics(config: GrainConfig): GrainAnalytics {\n return new GrainAnalytics(config);\n}\n\n// Default export for convenience\nexport default GrainAnalytics;\n\n// Global interface for IIFE build\ndeclare global {\n interface Window {\n Grain?: {\n GrainAnalytics: typeof GrainAnalytics;\n createGrainAnalytics: typeof createGrainAnalytics;\n };\n }\n}\n\n// Auto-setup for IIFE build\nif (typeof window !== 'undefined') {\n window.Grain = {\n GrainAnalytics,\n createGrainAnalytics,\n };\n}"],
5
+ "mappings": "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",
6
+ "names": ["src_exports", "__export", "GrainAnalytics", "createGrainAnalytics", "src_default", "config", "c", "r", "storageKey", "stored", "error", "args", "events", "eventNames", "e", "userIds", "totalProperties", "totalSize", "event", "properties", "context", "digest", "code", "message", "status", "formattedError", "timestamp", "errorOutput", "fn", "headers", "token", "ms", "resolve", "lastError", "attempt", "url", "response", "errorMessage", "errorBody", "errorText", "delayMs", "body", "blob", "handleBeforeUnload", "eventsToSend", "chunks", "eventOrName", "propertiesOrOptions", "options", "opts", "formattedEvent", "userId", "propertyKeys", "serializedProperties", "key", "value", "payload", "chunk", "cached", "cache", "configs", "immediateKeys", "request", "configResponse", "newCache", "oldConfigs", "listener", "index", "configurations", "effectiveUserId", "chunkSize", "i"]
7
7
  }
package/dist/index.js CHANGED
@@ -71,10 +71,10 @@ class GrainAnalytics {
71
71
  });
72
72
  }
73
73
  /**
74
- * Format UUID for anonymous user ID (remove dashes and prefix with 'temp:')
74
+ * Generate a proper UUIDv4 identifier for anonymous user ID
75
75
  */
76
- formatAnonymousUserId(uuid) {
77
- return `temp:${uuid.replace(/-/g, '')}`;
76
+ generateAnonymousUserId() {
77
+ return this.generateUUID();
78
78
  }
79
79
  /**
80
80
  * Initialize persistent anonymous user ID from localStorage or create new one
@@ -90,9 +90,8 @@ class GrainAnalytics {
90
90
  this.log('Loaded persistent anonymous user ID:', this.persistentAnonymousUserId);
91
91
  }
92
92
  else {
93
- // Generate new anonymous user ID
94
- const uuid = this.generateUUID();
95
- this.persistentAnonymousUserId = this.formatAnonymousUserId(uuid);
93
+ // Generate new UUIDv4 anonymous user ID
94
+ this.persistentAnonymousUserId = this.generateAnonymousUserId();
96
95
  localStorage.setItem(storageKey, this.persistentAnonymousUserId);
97
96
  this.log('Generated new persistent anonymous user ID:', this.persistentAnonymousUserId);
98
97
  }
@@ -100,15 +99,32 @@ class GrainAnalytics {
100
99
  catch (error) {
101
100
  this.log('Failed to initialize persistent anonymous user ID:', error);
102
101
  // Fallback: generate temporary ID without persistence
103
- const uuid = this.generateUUID();
104
- this.persistentAnonymousUserId = this.formatAnonymousUserId(uuid);
102
+ this.persistentAnonymousUserId = this.generateAnonymousUserId();
105
103
  }
106
104
  }
107
105
  /**
108
106
  * Get the effective user ID (global userId or persistent anonymous ID)
109
107
  */
110
108
  getEffectiveUserId() {
111
- return this.globalUserId || this.persistentAnonymousUserId || 'anonymous';
109
+ if (this.globalUserId) {
110
+ return this.globalUserId;
111
+ }
112
+ if (this.persistentAnonymousUserId) {
113
+ return this.persistentAnonymousUserId;
114
+ }
115
+ // Generate a new UUIDv4 identifier as fallback
116
+ this.persistentAnonymousUserId = this.generateAnonymousUserId();
117
+ // Try to persist it
118
+ if (typeof window !== 'undefined') {
119
+ try {
120
+ const storageKey = `grain_anonymous_user_id_${this.config.tenantId}`;
121
+ localStorage.setItem(storageKey, this.persistentAnonymousUserId);
122
+ }
123
+ catch (error) {
124
+ this.log('Failed to persist generated anonymous user ID:', error);
125
+ }
126
+ }
127
+ return this.persistentAnonymousUserId;
112
128
  }
113
129
  log(...args) {
114
130
  if (this.config.debug) {
@@ -466,10 +482,26 @@ class GrainAnalytics {
466
482
  setUserId(userId) {
467
483
  this.log(`Set global user ID: ${userId}`);
468
484
  this.globalUserId = userId;
469
- // Clear persistent anonymous user ID if setting a real user ID
470
485
  if (userId) {
486
+ // Clear persistent anonymous user ID if setting a real user ID
471
487
  this.persistentAnonymousUserId = null;
472
488
  }
489
+ else {
490
+ // If clearing user ID, ensure we have a UUIDv4 identifier
491
+ if (!this.persistentAnonymousUserId) {
492
+ this.persistentAnonymousUserId = this.generateAnonymousUserId();
493
+ // Try to persist the new anonymous ID
494
+ if (typeof window !== 'undefined') {
495
+ try {
496
+ const storageKey = `grain_anonymous_user_id_${this.config.tenantId}`;
497
+ localStorage.setItem(storageKey, this.persistentAnonymousUserId);
498
+ }
499
+ catch (error) {
500
+ this.log('Failed to persist new anonymous user ID:', error);
501
+ }
502
+ }
503
+ }
504
+ }
473
505
  }
474
506
  /**
475
507
  * Get current global user ID
@@ -483,6 +515,106 @@ class GrainAnalytics {
483
515
  getEffectiveUserIdPublic() {
484
516
  return this.getEffectiveUserId();
485
517
  }
518
+ /**
519
+ * Login with auth token or userId on the fly
520
+ *
521
+ * @example
522
+ * // Login with userId only
523
+ * client.login({ userId: 'user123' });
524
+ *
525
+ * // Login with auth token (automatically sets authStrategy to JWT)
526
+ * client.login({ authToken: 'jwt-token-here' });
527
+ *
528
+ * // Login with both userId and auth token
529
+ * client.login({ userId: 'user123', authToken: 'jwt-token-here' });
530
+ *
531
+ * // Override auth strategy
532
+ * client.login({ userId: 'user123', authStrategy: 'SERVER_SIDE' });
533
+ */
534
+ login(options) {
535
+ try {
536
+ if (this.isDestroyed) {
537
+ const error = new Error('Grain Analytics: Client has been destroyed');
538
+ const formattedError = this.formatError(error, 'login (client destroyed)');
539
+ this.logError(formattedError);
540
+ return;
541
+ }
542
+ // Set userId if provided
543
+ if (options.userId) {
544
+ this.log(`Login: Setting user ID to ${options.userId}`);
545
+ this.globalUserId = options.userId;
546
+ // Clear persistent anonymous user ID since we now have a real user ID
547
+ this.persistentAnonymousUserId = null;
548
+ }
549
+ // Handle auth token if provided
550
+ if (options.authToken) {
551
+ this.log('Login: Setting auth token');
552
+ // Update auth strategy to JWT if not already set
553
+ if (this.config.authStrategy === 'NONE') {
554
+ this.config.authStrategy = 'JWT';
555
+ }
556
+ // Create a simple auth provider that returns the provided token
557
+ this.config.authProvider = {
558
+ getToken: () => options.authToken
559
+ };
560
+ }
561
+ // Override auth strategy if provided
562
+ if (options.authStrategy) {
563
+ this.log(`Login: Setting auth strategy to ${options.authStrategy}`);
564
+ this.config.authStrategy = options.authStrategy;
565
+ }
566
+ this.log(`Login successful. Effective user ID: ${this.getEffectiveUserId()}`);
567
+ }
568
+ catch (error) {
569
+ const formattedError = this.formatError(error, 'login');
570
+ this.logError(formattedError);
571
+ }
572
+ }
573
+ /**
574
+ * Logout and clear user session, fall back to UUIDv4 identifier
575
+ *
576
+ * @example
577
+ * // Logout user and return to anonymous mode
578
+ * client.logout();
579
+ *
580
+ * // After logout, events will use the persistent UUIDv4 identifier
581
+ * client.track('page_view', { page: 'home' });
582
+ */
583
+ logout() {
584
+ try {
585
+ if (this.isDestroyed) {
586
+ const error = new Error('Grain Analytics: Client has been destroyed');
587
+ const formattedError = this.formatError(error, 'logout (client destroyed)');
588
+ this.logError(formattedError);
589
+ return;
590
+ }
591
+ this.log('Logout: Clearing user session');
592
+ // Clear global user ID
593
+ this.globalUserId = null;
594
+ // Reset auth strategy to NONE
595
+ this.config.authStrategy = 'NONE';
596
+ this.config.authProvider = undefined;
597
+ // Generate new UUIDv4 identifier if we don't have one
598
+ if (!this.persistentAnonymousUserId) {
599
+ this.persistentAnonymousUserId = this.generateAnonymousUserId();
600
+ // Try to persist the new anonymous ID
601
+ if (typeof window !== 'undefined') {
602
+ try {
603
+ const storageKey = `grain_anonymous_user_id_${this.config.tenantId}`;
604
+ localStorage.setItem(storageKey, this.persistentAnonymousUserId);
605
+ }
606
+ catch (error) {
607
+ this.log('Failed to persist new anonymous user ID after logout:', error);
608
+ }
609
+ }
610
+ }
611
+ this.log(`Logout successful. Effective user ID: ${this.getEffectiveUserId()}`);
612
+ }
613
+ catch (error) {
614
+ const formattedError = this.formatError(error, 'logout');
615
+ this.logError(formattedError);
616
+ }
617
+ }
486
618
  /**
487
619
  * Set user properties
488
620
  */
@@ -952,7 +1084,8 @@ class GrainAnalytics {
952
1084
  clearInterval(this.configRefreshTimer);
953
1085
  }
954
1086
  this.configRefreshTimer = window.setInterval(() => {
955
- if (!this.isDestroyed && this.globalUserId) {
1087
+ if (!this.isDestroyed) {
1088
+ // Use effective userId (will be generated if not set)
956
1089
  this.fetchConfig().catch((error) => {
957
1090
  const formattedError = this.formatError(error, 'auto-config refresh');
958
1091
  this.logError(formattedError);
@@ -974,10 +1107,9 @@ class GrainAnalytics {
974
1107
  */
975
1108
  async preloadConfig(immediateKeys = [], properties) {
976
1109
  try {
977
- if (!this.globalUserId) {
978
- this.log('Cannot preload config: no user ID set');
979
- return;
980
- }
1110
+ // Use effective userId (will be generated if not set)
1111
+ const effectiveUserId = this.getEffectiveUserId();
1112
+ this.log(`Preloading config for user: ${effectiveUserId}`);
981
1113
  const response = await this.fetchConfig({ immediateKeys, properties });
982
1114
  if (response) {
983
1115
  this.startConfigRefreshTimer();