shopkit-analytics 1.0.21 → 1.0.22
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/adapters/index.js +78 -20
- package/dist/adapters/index.js.map +1 -1
- package/dist/adapters/index.mjs +2 -2
- package/dist/{chunk-NSUG7HIX.mjs → chunk-7CBBOZY6.mjs} +79 -21
- package/dist/chunk-7CBBOZY6.mjs.map +1 -0
- package/dist/{chunk-VN6OWWKF.mjs → chunk-D7O7U5E7.mjs} +2 -2
- package/dist/{chunk-BE3W2SD6.mjs → chunk-EOVVGPT2.mjs} +2 -2
- package/dist/chunk-EOVVGPT2.mjs.map +1 -0
- package/dist/events/index.js +51 -20
- package/dist/events/index.js.map +1 -1
- package/dist/events/index.mjs +3 -3
- package/dist/index.js +79 -21
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +3 -3
- package/dist/types.d.mts +13 -4
- package/dist/types.d.ts +13 -4
- package/dist/types.js +1 -1
- package/dist/types.js.map +1 -1
- package/dist/types.mjs +1 -1
- package/package.json +1 -1
- package/dist/chunk-BE3W2SD6.mjs.map +0 -1
- package/dist/chunk-NSUG7HIX.mjs.map +0 -1
- /package/dist/{chunk-VN6OWWKF.mjs.map → chunk-D7O7U5E7.mjs.map} +0 -0
package/dist/events/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../src/events/publisher.ts","../../src/events/subscriber.ts","../../src/events/index.ts","../../src/types.ts","../../src/logger/index.ts","../../src/experiment/constants.ts","../../src/experiment/experiment-tracker.ts","../../src/adapters/base-adapter.ts","../../src/utils/event-id.ts","../../src/adapters/pixel-adapter.ts","../../src/adapters/google-adapter.ts","../../src/constants.ts","../../src/adapters/moengage-adapter.ts","../../src/adapters/posthog-adapter.ts","../../src/adapters/shopify-adapter.ts","../../src/events/init.ts"],"sourcesContent":["import type { TEvent } from \"../types\";\nimport type { TAdapterParams } from \"../adapter-params\";\n\n/**\n * EventPublisher class responsible for publishing events to subscribers\n */\nclass EventPublisher {\n private static instance: EventPublisher;\n private subscribers: Array<(event: TEvent, params?: TAdapterParams) => void> =\n [];\n\n /**\n * Private constructor to enforce singleton pattern\n */\n private constructor() {}\n\n /**\n * Get the singleton instance of EventPublisher\n */\n public static getInstance(): EventPublisher {\n if (!EventPublisher.instance) {\n EventPublisher.instance = new EventPublisher();\n }\n return EventPublisher.instance;\n }\n\n /**\n * Subscribe to events\n * @param callback Function to be called when an event is published\n * @returns Unsubscribe function\n */\n public subscribe(\n callback: (event: TEvent, params?: TAdapterParams) => void\n ): () => void {\n this.subscribers.push(callback);\n\n // Return unsubscribe function\n return () => {\n this.subscribers = this.subscribers.filter(\n (subscriber) => subscriber !== callback\n );\n };\n }\n\n /**\n * Publish an event to all subscribers\n * @param event The event to publish\n * @param params Optional adapter-specific parameters\n */\n public publish(event: TEvent, params?: TAdapterParams): void {\n try {\n console.log(\"EventPublisher: Received event\", {\n type: event.type,\n data: event,\n params: params,\n });\n\n // Add timestamp if not already present\n const eventWithTimestamp = !event.timestamp\n ? { ...event, timestamp: Date.now() }\n : event;\n\n console.log(\n \"EventPublisher: Number of subscribers:\",\n this.subscribers.length\n );\n\n // Notify all subscribers\n for (const subscriber of this.subscribers) {\n try {\n console.log(\"EventPublisher: Notifying subscriber\");\n subscriber(eventWithTimestamp, params);\n console.log(\"EventPublisher: Subscriber notified successfully\");\n } catch (error) {\n console.error(\"EventPublisher: Error in subscriber:\", error);\n // Continue with other subscribers even if one fails\n }\n }\n\n console.log(\"EventPublisher: Event processing completed\");\n } catch (error) {\n console.error(\"Error publishing event:\", error, event);\n // Silently fail to prevent app crashes due to event publishing issues\n }\n }\n}\n\n// Export singleton instance\nexport const eventPublisher = EventPublisher.getInstance();\n\n/**\n * Helper function to publish events\n * @param event The event to publish\n * @param params Optional adapter-specific parameters\n */\nexport function publishEvent(event: TEvent, params?: TAdapterParams): void {\n try {\n eventPublisher.publish(event, params);\n } catch (error) {\n console.error(\"Error in publishEvent helper:\", error);\n // Silently fail to prevent app crashes due to event publishing issues\n }\n}\n","import type { TEvent } from \"../types\";\nimport type { TAdapterParams } from \"../adapter-params\";\nimport { eventPublisher } from \"./publisher\";\n\n/**\n * Interface for tracking adapters\n */\nexport interface TrackingAdapter {\n /**\n * Name of the tracking adapter\n */\n name: string;\n\n /**\n * Whether the adapter is enabled\n */\n isEnabled: () => boolean;\n\n /**\n * Initialize the adapter\n */\n initialize: () => Promise<void>;\n\n /**\n * Track an event\n */\n trackEvent: (event: TEvent, params?: TAdapterParams) => Promise<void>;\n}\n\n/**\n * EventSubscriber class responsible for managing tracking adapters and forwarding events\n */\nclass EventSubscriber {\n private static instance: EventSubscriber;\n private adapters: TrackingAdapter[] = [];\n private initialized = false;\n private eventQueue: Array<{ event: TEvent; params?: TAdapterParams }> = [];\n private unsubscribe: (() => void) | null = null;\n\n /**\n * Private constructor to enforce singleton pattern\n */\n private constructor() {}\n\n /**\n * Get the singleton instance of EventSubscriber\n */\n public static getInstance(): EventSubscriber {\n if (!EventSubscriber.instance) {\n EventSubscriber.instance = new EventSubscriber();\n }\n return EventSubscriber.instance;\n }\n\n /**\n * Register a tracking adapter\n * @param adapter The tracking adapter to register\n */\n public registerAdapter(adapter: TrackingAdapter): void {\n // Check if adapter with the same name already exists\n const existingAdapter = this.adapters.find((a) => a.name === adapter.name);\n if (existingAdapter) {\n console.warn(\n `Tracking adapter with name \"${adapter.name}\" is already registered. Skipping.`\n );\n return;\n }\n\n this.adapters.push(adapter);\n\n // If already initialized, initialize the new adapter\n if (this.initialized) {\n this.initializeAdapter(adapter).catch((error) => {\n console.error(`Failed to initialize adapter \"${adapter.name}\":`, error);\n });\n }\n }\n\n /**\n * Unregister a tracking adapter by name\n * @param adapterName The name of the adapter to unregister\n */\n public unregisterAdapter(adapterName: string): void {\n this.adapters = this.adapters.filter(\n (adapter) => adapter.name !== adapterName\n );\n }\n\n /**\n * Initialize all registered adapters\n */\n public async initialize(): Promise<void> {\n console.log(\"Initializing EventSubscriber...\");\n\n if (this.initialized) {\n console.log(\"EventSubscriber already initialized\");\n return;\n }\n\n // Subscribe to events\n this.unsubscribe = eventPublisher.subscribe(this.handleEvent.bind(this));\n console.log(\"Subscribed to event publisher\");\n\n // Initialize all adapters\n console.log(`Initializing ${this.adapters.length} adapters...`);\n await Promise.all(\n this.adapters.map((adapter) => this.initializeAdapter(adapter))\n );\n\n this.initialized = true;\n console.log(\"EventSubscriber initialization complete\");\n\n // Process any queued events\n this.processEventQueue();\n }\n\n /**\n * Initialize a single adapter\n * @param adapter The adapter to initialize\n */\n private async initializeAdapter(adapter: TrackingAdapter): Promise<void> {\n try {\n await adapter.initialize();\n console.log(\n `Tracking adapter \"${adapter.name}\" initialized successfully.`\n );\n } catch (error) {\n console.error(`Failed to initialize adapter \"${adapter.name}\":`, error);\n }\n }\n\n /**\n * Handle an event by forwarding it to all enabled adapters\n * @param event The event to handle\n * @param params Optional adapter-specific parameters\n */\n private handleEvent(event: TEvent, params?: TAdapterParams): void {\n console.log(\"Handling event:\", event.type, \"with params:\", params);\n\n if (!this.initialized) {\n console.log(\"Subscriber not initialized, queueing event\");\n this.eventQueue.push({ event, params });\n return;\n }\n\n // Forward the event to all enabled adapters\n for (const adapter of this.adapters) {\n if (adapter.isEnabled()) {\n console.log(`Forwarding event to adapter: ${adapter.name}`);\n adapter.trackEvent(event, params).catch((error) => {\n console.error(\n `Error tracking event in adapter \"${adapter.name}\":`,\n error\n );\n });\n } else {\n console.log(`Adapter ${adapter.name} is disabled, skipping`);\n }\n }\n }\n\n /**\n * Process any events that were queued before initialization\n */\n private processEventQueue(): void {\n if (this.eventQueue.length === 0) {\n return;\n }\n\n console.log(`Processing ${this.eventQueue.length} queued events.`);\n\n // Process each queued event\n for (const { event, params } of this.eventQueue) {\n this.handleEvent(event, params);\n }\n\n // Clear the queue\n this.eventQueue = [];\n }\n\n /**\n * Shutdown the subscriber and all adapters\n */\n public shutdown(): void {\n if (this.unsubscribe) {\n this.unsubscribe();\n this.unsubscribe = null;\n }\n\n this.initialized = false;\n this.eventQueue = [];\n }\n\n /**\n * Get all registered adapters\n */\n public getAdapters(): TrackingAdapter[] {\n return [...this.adapters];\n }\n}\n\n// Export singleton instance\nexport const eventSubscriber = EventSubscriber.getInstance();\n","// Export event types\nexport * from \"../types\";\n\n// Export publisher\nexport { eventPublisher, publishEvent } from \"./publisher\";\n\n// Export subscriber\nexport { eventSubscriber } from \"./subscriber\";\nexport type { TrackingAdapter } from \"./subscriber\";\n\n// Export initialization functions\nexport { initializeEventTracking, initTracking } from \"./init\";\n\n// Export constants\nexport * from \"../constants\";\n","/**\n * Event types that can be tracked across the application\n */\nexport enum EventType {\n PRODUCT_VIEW = \"product_view\",\n PAGE_VIEW = \"page_view\",\n BUTTON_CLICK = \"button_click\",\n FORM_SUBMISSION = \"form_submission\",\n VIEW_ITEM = \"view_item\",\n VIEWED_PRODUCT = \"viewed_product\",\n CHECKOUT_PAYMENT = \"checkout_payment\",\n COLLECTION_VIEW = \"collection_view\",\n ADD_TO_CART = \"add_to_cart\",\n REMOVE_FROM_CART = \"remove_from_cart\",\n CHECKOUT_STARTED = \"checkout_started\",\n CHECKOUT_COMPLETED = \"checkout_completed\",\n SEARCH = \"search\",\n USER_SIGNUP = \"user_signup\",\n USER_LOGIN = \"user_login\",\n CUSTOM = \"custom\",\n USER_ENGAGEMENT = \"user_engagement\",\n SESSION_START = \"session_start\",\n FIRST_VISIT = \"first_visit\",\n SCROLL = \"scroll\",\n FORM_START = \"form_start\",\n ADD_PAYMENT_INFO = \"add_payment_info\",\n CART_VIEWED = \"cart_viewed\",\n VIEW_CONTENT = \"view_content\",\n PURCHASE = \"purchase\",\n VIEW_SEARCH_RESULTS = \"view_search_results\",\n ORDER_PLACED = \"order_placed\",\n BEGIN_CHECKOUT = \"begin_checkout\",\n\n // GoKwik Checkout events (actually used)\n STARTED_CHECKOUT_GK = \"started_checkout_gk\",\n MOBILE_ADDED_GK = \"mobile_added_gk\",\n ADDRESS_SELECTED_GK = \"address_selected_gk\",\n ADDRESS_COMPLETED_GK = \"address_completed_gk\",\n PAYMENT_METHOD_SELECTED_GK = \"payment_method_selected_gk\",\n PAYMENT_COMPLETED_GK = \"payment_completed_gk\",\n ORDER_SUCCESS = \"order_success\",\n ORDER_COMPLETED = \"order_completed\",\n PIN_CODE_ADDED_GK = \"pin_code_added_gk\",\n ADDRESS_ADDED_GK = \"address_added_gk\",\n\n // Track Specific Event\n SPECIFIC = \"specific\",\n\n // Shopify specific\n SHOPIFY_PAGE_VIEW= \"shopify_page_view\"\n}\n\n/**\n * Base event interface that all events should implement\n */\nexport interface BaseEvent {\n type: EventType;\n timestamp?: number;\n eventId?: string;\n event_category?: string;\n description?: string;\n [key: string]: any;\n}\n\n/**\n * Page view event\n */\nexport interface IPageViewEvent extends BaseEvent {\n type: EventType.PAGE_VIEW;\n path: string;\n title: string;\n referrer?: string;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n campaign?: string;\n campaign_id?: string;\n content?: string;\n debug_mode?: boolean;\n engagement_time_msec?: number;\n entrances?: number;\n ga_session_id?: string;\n ga_session_number?: number;\n gclid?: string;\n ignore_referrer?: boolean;\n medium?: string;\n page_location?: string;\n page_path?: string;\n page_referrer?: string;\n page_title?: string;\n page_source?: string;\n page_term?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n}\n\n/**\n * User engagement event\n */\nexport interface IUserEngagementEvent extends BaseEvent {\n type: EventType.USER_ENGAGEMENT;\n engagementTime?: number; // in seconds\n pagePath?: string;\n pageTitle?: string;\n scrollDepth?: number; // in percentage\n formId?: string;\n formName?: string;\n formFields?: Array<{\n fieldName: string;\n fieldValue: string;\n }>;\n buttonId?: string;\n buttonText?: string;\n location?: string;\n customProperties?: Record<string, any>;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n campaign?: string;\n ga_session_id?: string;\n ga_session_number?: number;\n gclid?: string;\n ignore_referrer?: boolean;\n medium?: string;\n page_location?: string;\n page_path?: string;\n page_referrer?: string;\n page_title?: string;\n source?: string;\n srsltid?: string;\n term?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n}\n\n/**\n * Session start event\n */\nexport interface ISessionStartEvent extends BaseEvent {\n type: EventType.SESSION_START;\n sessionId: string;\n userId?: string;\n userAgent?: string;\n referrer?: string;\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n}\n\n/**\n * First visit event\n */\nexport interface IFirstVisitEvent extends BaseEvent {\n type: EventType.FIRST_VISIT;\n userId?: string;\n userAgent?: string;\n referrer?: string;\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n}\n\n/**\n * Scroll event\n */\nexport interface IScrollEvent extends BaseEvent {\n type: EventType.SCROLL;\n scrollDepth: number; // in percentage\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n engagement_time_msec?: number;\n ga_session_id?: string;\n ga_session_number?: number;\n ignore_referrer?: boolean;\n page_location?: string;\n page_referrer?: string;\n page_title?: string;\n percent_scrolled?: number;\n}\n\n/**\n * Form start event\n */\nexport interface IFormStartEvent extends BaseEvent {\n type: EventType.FORM_START;\n formId: string;\n formName?: string;\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n}\n\n/**\n * Form submission event\n */\nexport interface IFormSubmissionEvent extends BaseEvent {\n type: EventType.FORM_SUBMISSION;\n formId: string;\n formName?: string;\n success: boolean;\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n}\n\n/**\n * User signup event\n */\nexport interface IUserSignupEvent extends BaseEvent {\n type: EventType.USER_SIGNUP;\n userId?: string;\n method?: string;\n success: boolean;\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n}\n\n/**\n * User login event\n */\nexport interface IUserLoginEvent extends BaseEvent {\n type: EventType.USER_LOGIN;\n userId?: string;\n method?: string;\n success: boolean;\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n}\n\n/**\n * Button click event\n */\nexport interface IButtonClickEvent extends BaseEvent {\n type: EventType.BUTTON_CLICK;\n buttonId?: string;\n buttonText?: string;\n location: string;\n}\n\n/**\n * View item event for e-commerce tracking\n */\nexport interface IViewItemEvent extends BaseEvent {\n type: EventType.VIEW_ITEM;\n currency: string;\n value: number;\n items: Array<{\n item_id: string;\n item_name: string;\n price: number;\n currency: string;\n }>;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n campaign?: string;\n ecomm_pagetype?: string;\n ecomm_prodid?: string;\n ecomm_totalvalue?: number;\n engagement_time_msec?: number;\n ga_session_id?: string;\n ga_session_number?: number;\n gclid?: string;\n ignore_referrer?: boolean;\n medium?: string;\n page_location?: string;\n page_path?: string;\n page_referrer?: string;\n page_title?: string;\n source?: string;\n srsltid?: string;\n term?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n}\n\n/**\n * Checkout payment event\n */\nexport interface ICheckoutPaymentEvent extends BaseEvent {\n type: EventType.CHECKOUT_PAYMENT;\n cartValue: number;\n currency: string;\n itemCount: number;\n items: Array<{\n productId: string;\n productName: string;\n price: number;\n quantity: number;\n }>;\n}\n\n/**\n * Viewed product event after 20 seconds\n */\nexport interface IViewedProductEvent extends BaseEvent {\n type: EventType.VIEWED_PRODUCT;\n productId: string;\n productName: string;\n price: number;\n currency: string;\n viewDuration?: number;\n}\n\nexport interface IPCollectionViewEvent extends BaseEvent {\n type: EventType.COLLECTION_VIEW;\n productId: string;\n productName: string;\n price?: number;\n currency?: string;\n category?: string;\n}\n\n/**\n * Add to cart event\n */\nexport interface IAddToCartEvent extends BaseEvent {\n type: EventType.ADD_TO_CART;\n productId: string;\n productName: string;\n price: number;\n currency?: string;\n quantity?: number;\n variant?: string;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n ecomm_pagetype?: string;\n ecomm_prodid?: string;\n ecomm_totalvalue?: number;\n value?: number;\n engagement_time_msec?: number;\n ga_session_id?: string;\n ga_session_number?: number;\n ignore_referrer?: boolean;\n page_location?: string;\n page_referrer?: string;\n page_title?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n}\n\n/**\n * Remove from cart event\n */\nexport interface IRemoveFromCartEvent extends BaseEvent {\n type: EventType.REMOVE_FROM_CART;\n productId: string;\n productName: string;\n price: number;\n currency?: string;\n quantity: number;\n variant?: string;\n}\n\n/**\n * Checkout started event\n */\nexport interface ICheckoutStartedEvent extends BaseEvent {\n type: EventType.CHECKOUT_STARTED;\n cartValue: number;\n currency?: string;\n itemCount: number;\n items: Array<{\n productId: string;\n productName: string;\n price: number;\n quantity: number;\n variant?: string;\n }>;\n coupon?: string;\n discount?: number | string;\n}\n\n/**\n * Checkout completed event\n */\nexport interface ICheckoutCompletedEvent extends BaseEvent {\n type: EventType.CHECKOUT_COMPLETED;\n orderId: string;\n cartValue: number;\n currency?: string;\n itemCount: number;\n items: Array<{\n productId: string;\n productName: string;\n price: number;\n quantity: number;\n variant?: string;\n }>;\n}\n\n/**\n * Cart viewed event\n */\nexport interface ICartViewedEvent extends BaseEvent {\n type: EventType.CART_VIEWED;\n cartId: string;\n products: Array<{\n variantId: string;\n price: number;\n quantity: number;\n }>;\n}\n\n/**\n * Search event\n */\nexport interface ISearchEvent extends BaseEvent {\n type: EventType.SEARCH;\n searchTerm: string;\n content_ids: string[];\n resultsCount?: number;\n}\n\n/**\n * View content event (for Facebook Pixel ViewContent)\n */\nexport interface IViewContentEvent extends BaseEvent {\n type: EventType.VIEW_CONTENT;\n content_type?: string;\n content_ids?: string[];\n content_name?: string;\n content_category?: string;\n value?: number;\n currency?: string;\n items?: Array<any>;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n campaign?: string;\n ga_session_id?: string;\n engagement_time_msec?: number;\n session_engaged?: boolean;\n page_title?: string;\n page_location?: string;\n page_referrer?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n}\n\n/**\n * Purchase event (for Facebook Pixel Purchase)\n */\nexport interface IPurchaseEvent extends BaseEvent {\n type: EventType.PURCHASE;\n value: number;\n currency?: string;\n content_type?: string;\n content_ids?: string[];\n num_items?: number;\n contents?: Array<{\n id: string;\n quantity: number;\n item_price?: number;\n }>;\n transaction_id?: string;\n coupon?: string;\n shipping?: number;\n tax?: number;\n items?: Array<any>;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n campaign?: string;\n ga_session_id?: string;\n engagement_time_msec?: number;\n session_engaged?: boolean;\n page_title?: string;\n page_location?: string;\n page_referrer?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n}\n\n/**\n * View search results event (for GA4)\n */\nexport interface IViewSearchResultsEvent extends BaseEvent {\n type: EventType.VIEW_SEARCH_RESULTS;\n search_term?: string;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n campaign?: string;\n campaign_id?: string;\n content?: string;\n ga_session_id?: string;\n ga_session_number?: number;\n gclid?: string;\n medium?: string;\n page_location?: string;\n page_path?: string;\n page_referrer?: string;\n page_title?: string;\n source?: string;\n srsltid?: string;\n term?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n engagement_time_msec?: number;\n}\n\n/**\n * Order placed event (for GA4)\n */\nexport interface IOrderPlacedEvent extends BaseEvent {\n type: EventType.ORDER_PLACED;\n transaction_id?: string;\n order_id?: string;\n total_amount?: number;\n value: number;\n currency?: string;\n items?: Array<any>;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n coupon?: string;\n discount?: number;\n engagement_time_msec?: number;\n ga_session_id?: string;\n ga_session_number?: number;\n ignore_referrer?: boolean;\n page_location?: string;\n page_referrer?: string;\n page_title?: string;\n payment_type?: string;\n shipping?: number;\n shipping_name?: string;\n tax?: number;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n}\n\n/**\n * Begin checkout event (for GA4)\n */\nexport interface IBeginCheckoutEvent extends BaseEvent {\n type: EventType.BEGIN_CHECKOUT;\n currency?: string;\n value: number;\n coupon?: string;\n items?: Array<any>;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n ecomm_pagetype?: string;\n ecomm_prodid?: string;\n ecomm_totalvalue?: number;\n engagement_time_msec?: number;\n ga_session_id?: string;\n ga_session_number?: number;\n ignore_referrer?: boolean;\n page_location?: string;\n page_referrer?: string;\n page_title?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n}\n\n/**\n * Custom event for any other tracking needs\n */\nexport interface ICustomEvent extends BaseEvent {\n type: EventType.CUSTOM;\n}\n\n/**\n * Custom event for any other tracking needs\n */\nexport interface ISpecificEvent extends BaseEvent {\n type: EventType.SPECIFIC;\n}\n\n/**\n * Add Payment Info event\n */\nexport interface IAddPaymentInfoEvent extends BaseEvent {\n type: EventType.ADD_PAYMENT_INFO;\n cartValue: number;\n currency?: string;\n itemCount: number;\n paymentType?: string;\n items: Array<{\n productId: string;\n productName: string;\n price: number;\n quantity: number;\n variant?: string;\n }>;\n}\n\n/**\n * Started checkout GK event\n */\nexport interface IStartedCheckoutGKEvent extends BaseEvent {\n type: EventType.STARTED_CHECKOUT_GK;\n cartValue?: number;\n currency?: string;\n}\n\n/**\n * Mobile added GK event\n */\nexport interface IMobileAddedGKEvent extends BaseEvent {\n type: EventType.MOBILE_ADDED_GK;\n mobile?: string;\n}\n\n/**\n * Address selected GK event\n */\nexport interface IAddressSelectedGKEvent extends BaseEvent {\n type: EventType.ADDRESS_SELECTED_GK;\n addressId?: string;\n}\n\n/**\n * Address completed GK event\n */\nexport interface IAddressCompletedGKEvent extends BaseEvent {\n type: EventType.ADDRESS_COMPLETED_GK;\n addressId?: string;\n}\n\n/**\n * Payment method selected GK event\n */\nexport interface IPaymentMethodSelectedGKEvent extends BaseEvent {\n type: EventType.PAYMENT_METHOD_SELECTED_GK;\n paymentMethod?: string;\n}\n\n/**\n * Payment completed GK event\n */\nexport interface IPaymentCompletedGKEvent extends BaseEvent {\n type: EventType.PAYMENT_COMPLETED_GK;\n amount?: number;\n currency?: string;\n}\n\n/**\n * Order success event\n */\nexport interface IOrderSuccessEvent extends BaseEvent {\n type: EventType.ORDER_SUCCESS;\n orderId?: string;\n amount?: number;\n currency?: string;\n}\n\n/**\n * Order completed event\n */\nexport interface IOrderCompletedEvent extends BaseEvent {\n type: EventType.ORDER_COMPLETED;\n orderId?: string;\n amount?: number;\n currency?: string;\n}\n\n/**\n * PIN code added GK event\n */\nexport interface IPinCodeAddedGKEvent extends BaseEvent {\n type: EventType.PIN_CODE_ADDED_GK;\n pinCode?: string;\n}\n\n/**\n * Address added GK event\n */\nexport interface IAddressAddedGKEvent extends BaseEvent {\n type: EventType.ADDRESS_ADDED_GK;\n addressId?: string;\n}\n\n/**\n * Product view event\n */\nexport interface IProductViewEvent extends BaseEvent {\n type: EventType.PRODUCT_VIEW;\n event_category: string;\n description: string;\n productId: string;\n productName: string;\n price: number;\n currency?: string;\n category?: string;\n}\n\nexport interface IShopifyPageViewEvent extends BaseEvent {\n type: EventType.SHOPIFY_PAGE_VIEW;\n path?: string;\n}\n\n\n/**\n * Union type of all possible events\n */\nexport type TEvent =\n | IProductViewEvent\n | IPageViewEvent\n | IButtonClickEvent\n | IFormSubmissionEvent\n | IViewItemEvent\n | IPCollectionViewEvent\n | IAddToCartEvent\n | IRemoveFromCartEvent\n | ICheckoutStartedEvent\n | ICheckoutCompletedEvent\n | ICartViewedEvent\n | ISearchEvent\n | IUserSignupEvent\n | IUserLoginEvent\n | IViewContentEvent\n | IPurchaseEvent\n | IViewSearchResultsEvent\n | IOrderPlacedEvent\n | IBeginCheckoutEvent\n | ICustomEvent\n | ISpecificEvent\n | IUserEngagementEvent\n | ISessionStartEvent\n | IFirstVisitEvent\n | IScrollEvent\n | IFormStartEvent\n | IAddPaymentInfoEvent\n | IStartedCheckoutGKEvent\n | IMobileAddedGKEvent\n | IAddressSelectedGKEvent\n | IAddressCompletedGKEvent\n | IPaymentMethodSelectedGKEvent\n | IPaymentCompletedGKEvent\n | IOrderSuccessEvent\n | IOrderCompletedEvent\n | IPinCodeAddedGKEvent\n | IAddressAddedGKEvent\n | IViewedProductEvent\n | ICheckoutPaymentEvent\n | IShopifyPageViewEvent;\n","/**\n * Logger configuration options\n */\nexport interface LoggerConfig {\n level?: \"trace\" | \"debug\" | \"info\" | \"warn\" | \"error\" | \"fatal\";\n enabled?: boolean;\n name?: string;\n prettyPrint?: boolean;\n redact?: string[];\n}\n\n/**\n * Logger interface for consistent logging across adapters\n */\nexport interface Logger {\n trace: (message: string, ...args: any[]) => void;\n debug: (message: string, ...args: any[]) => void;\n info: (message: string, ...args: any[]) => void;\n warn: (message: string, ...args: any[]) => void;\n error: (message: string, error?: Error, ...args: any[]) => void;\n fatal: (message: string, error?: Error, ...args: any[]) => void;\n child: (bindings: Record<string, any>) => Logger;\n}\n\n/**\n * Log levels with numeric values for comparison\n */\nconst LOG_LEVELS = {\n trace: 10,\n debug: 20,\n info: 30,\n warn: 40,\n error: 50,\n fatal: 60,\n} as const;\n\n/**\n * No-op logger for when logging is disabled\n */\nclass NoOpLogger implements Logger {\n trace() {}\n debug() {}\n info() {}\n warn() {}\n error() {}\n fatal() {}\n child(): Logger {\n return new NoOpLogger();\n }\n}\n\n/**\n * Simple universal logger that works in all environments\n */\nclass UniversalLogger implements Logger {\n private config: Required<LoggerConfig>;\n private bindings: Record<string, any>;\n\n constructor(\n config: Required<LoggerConfig>,\n bindings: Record<string, any> = {}\n ) {\n this.config = config;\n this.bindings = bindings;\n }\n\n private shouldLog(level: keyof typeof LOG_LEVELS): boolean {\n return LOG_LEVELS[level] >= LOG_LEVELS[this.config.level];\n }\n\n private formatMessage(\n level: string,\n message: string,\n args: any[],\n error?: Error\n ): string {\n const timestamp = new Date().toISOString();\n const name = this.config.name;\n\n let logMessage = this.config.prettyPrint\n ? `[${timestamp}] ${level.toUpperCase()} [${name}]: ${message}`\n : JSON.stringify({\n timestamp,\n level,\n name,\n message,\n ...this.bindings,\n ...(args.length > 0 && { args }),\n ...(error && {\n error: { message: error.message, stack: error.stack },\n }),\n });\n\n if (this.config.prettyPrint && args.length > 0) {\n logMessage += ` ${args\n .map((arg) =>\n typeof arg === \"object\" ? JSON.stringify(arg) : String(arg)\n )\n .join(\" \")}`;\n }\n\n if (this.config.prettyPrint && error) {\n logMessage += ` Error: ${error.message}`;\n if (error.stack) {\n logMessage += `\\n${error.stack}`;\n }\n }\n\n return this.redactSensitiveData(logMessage);\n }\n\n private redactSensitiveData(message: string): string {\n let redactedMessage = message;\n for (const field of this.config.redact) {\n const regex = new RegExp(`\"${field}\"\\\\s*:\\\\s*\"[^\"]*\"`, \"gi\");\n redactedMessage = redactedMessage.replace(\n regex,\n `\"${field}\":\"[REDACTED]\"`\n );\n }\n return redactedMessage;\n }\n\n private log(\n level: keyof typeof LOG_LEVELS,\n message: string,\n args: any[],\n error?: Error\n ): void {\n if (!this.shouldLog(level)) return;\n\n const formattedMessage = this.formatMessage(level, message, args, error);\n\n // Use appropriate console method based on level\n switch (level) {\n case \"trace\":\n console.trace(formattedMessage);\n break;\n case \"debug\":\n console.debug(formattedMessage);\n break;\n case \"info\":\n console.info(formattedMessage);\n break;\n case \"warn\":\n console.warn(formattedMessage);\n break;\n case \"error\":\n case \"fatal\":\n console.error(formattedMessage);\n break;\n }\n }\n\n trace(message: string, ...args: any[]): void {\n this.log(\"trace\", message, args);\n }\n\n debug(message: string, ...args: any[]): void {\n this.log(\"debug\", message, args);\n }\n\n info(message: string, ...args: any[]): void {\n this.log(\"info\", message, args);\n }\n\n warn(message: string, ...args: any[]): void {\n this.log(\"warn\", message, args);\n }\n\n error(message: string, error?: Error, ...args: any[]): void {\n this.log(\"error\", message, args, error);\n }\n\n fatal(message: string, error?: Error, ...args: any[]): void {\n this.log(\"fatal\", message, args, error);\n }\n\n child(bindings: Record<string, any>): Logger {\n return new UniversalLogger(this.config, { ...this.bindings, ...bindings });\n }\n}\n\n/**\n * Create a logger instance\n */\nexport function createLogger(config: LoggerConfig = {}): Logger {\n const fullConfig: Required<LoggerConfig> = {\n level: config.level || \"info\",\n enabled: config.enabled !== false,\n name: config.name || \"@shopkit/analytics\",\n prettyPrint: config.prettyPrint || false,\n redact: config.redact || [],\n };\n\n // Return no-op logger if disabled\n if (!fullConfig.enabled) {\n return new NoOpLogger();\n }\n\n return new UniversalLogger(fullConfig);\n}\n\n/**\n * Default logger instance\n */\nexport const logger = createLogger({\n enabled: true,\n level: \"info\",\n prettyPrint: false,\n});\n\n/**\n * Create adapter-specific logger\n */\nexport function createAdapterLogger(\n adapterName: string,\n config: LoggerConfig = {}\n): Logger {\n const baseLogger = createLogger(config);\n return baseLogger.child({ adapter: adapterName });\n}\n","/**\n * Prima PR A/B test experiment cookie names\n */\nexport const PRIMA_EXPERIMENT_COOKIES = {\n HOME: \"_prima_ptr_ab_home\",\n COLLECTION: \"_prima_ptr_ab_collection\",\n PRODUCT: \"_prima_ptr_ab_product\",\n} as const;\n\n/**\n * Array of all Prima experiment cookie names for iteration\n */\nexport const PRIMA_EXPERIMENT_COOKIE_NAMES = Object.values(\n PRIMA_EXPERIMENT_COOKIES\n);\n","import type { ExperimentData } from \"./types\";\nimport {\n PRIMA_EXPERIMENT_COOKIES,\n PRIMA_EXPERIMENT_COOKIE_NAMES,\n} from \"./constants\";\n\n/**\n * Get cookie value by name\n */\nfunction getCookie(name: string): string | null {\n if (typeof document === \"undefined\") return null;\n\n const value = `; ${document.cookie}`;\n const parts = value.split(`; ${name}=`);\n if (parts.length === 2) {\n return parts.pop()?.split(\";\").shift() || null;\n }\n return null;\n}\n\n/**\n * Get experiment parameters from cookies\n */\nexport function getExperimentParams(): ExperimentData {\n const payload = {\n [PRIMA_EXPERIMENT_COOKIES.HOME]: getCookie(PRIMA_EXPERIMENT_COOKIES.HOME),\n [PRIMA_EXPERIMENT_COOKIES.COLLECTION]: getCookie(\n PRIMA_EXPERIMENT_COOKIES.COLLECTION\n ),\n [PRIMA_EXPERIMENT_COOKIES.PRODUCT]: getCookie(\n PRIMA_EXPERIMENT_COOKIES.PRODUCT\n ),\n };\n\n return payload;\n}\n\n/**\n * Check if any experiment data exists\n */\nexport function hasExperimentData(): boolean {\n return PRIMA_EXPERIMENT_COOKIE_NAMES.some(\n (cookieName) => getCookie(cookieName) !== null\n );\n}\n","import type { TEvent } from \"../types\";\nimport type { TrackingAdapter } from \"../events/subscriber\";\nimport type { TAdapterParams, IBaseAdapterParams } from \"../adapter-params\";\nimport { createAdapterLogger, type Logger, type LoggerConfig } from \"../logger\";\nimport { getExperimentParams } from \"../experiment/experiment-tracker\";\nimport type { ExperimentData } from \"../experiment/types\";\nimport { PRIMA_EXPERIMENT_COOKIES } from \"../experiment\";\n\n/**\n * Abstract base class for tracking adapters\n */\nexport abstract class BaseAdapter implements TrackingAdapter {\n /**\n * Name of the tracking adapter\n */\n public abstract readonly name: string;\n\n /**\n * Configuration for the adapter\n */\n protected config: Record<string, any>;\n\n /**\n * Logger instance for this adapter\n */\n protected logger: Logger;\n\n /**\n * Whether the adapter has been initialized\n */\n protected initialized = false;\n\n /**\n * Constructor\n * @param config Configuration for the adapter\n */\n constructor(config: Record<string, any> = {}) {\n this.config = config;\n\n // Create a generic logger - will be properly initialized by subclasses\n const loggerConfig: LoggerConfig = {\n enabled: config.enableDebugLogs ?? true,\n level: config.logLevel ?? \"info\",\n name: \"@shopkit/analytics\",\n };\n\n this.logger = createAdapterLogger(\"BaseAdapter\", loggerConfig);\n }\n\n /**\n * Initialize logger with proper adapter name\n * Should be called by subclasses after name is available\n */\n protected initializeLogger(): void {\n const loggerConfig: LoggerConfig = {\n enabled: this.config.enableDebugLogs ?? true,\n level: this.config.logLevel ?? \"info\",\n name: \"@shopkit/analytics\",\n };\n\n this.logger = createAdapterLogger(this.name, loggerConfig);\n }\n\n /**\n * Check if the adapter is enabled\n */\n public isEnabled(): boolean {\n return this.initialized;\n }\n\n /**\n * Initialize the adapter\n */\n public abstract initialize(): Promise<void>;\n\n /**\n * Track an event\n * @param event The event to track\n * @param params Optional adapter-specific parameters\n */\n public abstract trackEvent(\n event: TEvent,\n params?: TAdapterParams\n ): Promise<void>;\n\n /**\n * Extract adapter-specific parameters for this adapter\n * @param params The adapter parameters object\n * @returns Adapter-specific parameters or empty object\n */\n protected getAdapterParams(params?: TAdapterParams): IBaseAdapterParams {\n return params?.[this.name] || {};\n }\n\n /**\n * Get custom event name or fall back to default\n * @param params The adapter parameters object\n * @param defaultName The default event name to use\n * @returns Custom event name or default\n */\n protected getEventName(\n params: TAdapterParams | undefined,\n defaultName: string\n ): string {\n const adapterParams = this.getAdapterParams(params);\n return adapterParams.event_name || defaultName;\n }\n\n /**\n * Merge standard event data with adapter-specific parameters\n * @param params The adapter parameters object\n * @param standardData The standard event data\n * @returns Merged data with adapter-specific overrides\n */\n protected mergeEventData(\n standardData: Record<string, any>,\n params: TAdapterParams | undefined\n ): Record<string, any> {\n const adapterParams = this.getAdapterParams(params);\n\n // Remove event_name from adapter params to avoid conflicts\n const { event_name, ...customParams } = adapterParams;\n\n // Merge with priority: adapter-specific > standard\n return {\n ...standardData,\n ...customParams,\n };\n }\n\n /**\n * Get a configuration value\n * @param key The configuration key\n * @param defaultValue The default value if the key is not found\n */\n protected getConfig<T>(key: string, defaultValue?: T): T {\n return key in this.config ? this.config[key] : (defaultValue as T);\n }\n\n /**\n * Get affiliate parameters from session storage or custom implementation\n * Override this method to provide your own affiliate tracking implementation\n * @returns Affiliate parameters or null if not available\n */\n protected getAffiliateParams(): Record<string, string> | null {\n // Default implementation returns null\n // Users can override this method to integrate with their affiliate tracking system\n return null;\n }\n\n /**\n * Get experiment parameters from cookies\n * @returns Experiment parameters\n */\n protected getExperimentParams(): ExperimentData {\n try {\n return getExperimentParams();\n } catch (error) {\n // Silently fail if experiment tracking is not available\n return {\n [PRIMA_EXPERIMENT_COOKIES.HOME]: null,\n [PRIMA_EXPERIMENT_COOKIES.COLLECTION]: null,\n [PRIMA_EXPERIMENT_COOKIES.PRODUCT]: null,\n };\n }\n }\n\n /**\n * Enhance event parameters with affiliate data\n * @param params The original parameters\n * @returns Parameters enhanced with affiliate data\n */\n protected enhanceWithAffiliateParams(\n params: Record<string, any>\n ): Record<string, any> {\n const affiliateParams = this.getAffiliateParams();\n if (!affiliateParams) return params;\n\n return {\n ...params,\n // Add affiliate parameters\n ...affiliateParams,\n };\n }\n\n /**\n * Enhance event parameters with experiment data\n * @param params The original parameters\n * @returns Parameters enhanced with experiment data\n */\n protected enhanceWithExperimentParams(\n params: Record<string, any>\n ): Record<string, any> {\n const experimentParams = this.getExperimentParams();\n\n return {\n ...params,\n experiment: experimentParams,\n };\n }\n}\n","/**\n * Generate a unique event ID for deduplication\n * Format: timestamp_randomString_eventType\n */\nexport function generateEventId(eventType: string): string {\n const timestamp = Date.now();\n const randomString = Math.random().toString(36).substring(2, 8);\n return `${timestamp}_${randomString}_${eventType}`;\n}\n\n/**\n * Extract browser information for CAPI user data\n */\nexport function getBrowserInfo(): {\n clientUserAgent?: string;\n fbc?: string;\n fbp?: string;\n} {\n if (typeof window === \"undefined\") {\n return {};\n }\n\n const userAgent = navigator.userAgent;\n\n // Get Facebook click ID from URL or cookies\n const fbc = getFacebookClickId();\n const fbp = getFacebookBrowserId();\n\n return {\n clientUserAgent: userAgent,\n fbc,\n fbp,\n };\n}\n\n/**\n * Get Facebook click ID (fbc) from URL parameters or cookies\n */\nfunction getFacebookClickId(): string | undefined {\n if (typeof window === \"undefined\") {\n return undefined;\n }\n\n // Check URL parameters first\n const urlParams = new URLSearchParams(window.location.search);\n const fbclid = urlParams.get(\"fbclid\");\n\n if (fbclid) {\n return `fb.1.${Date.now()}.${fbclid}`;\n }\n\n // Check cookies\n const cookies = document.cookie.split(\";\");\n for (const cookie of cookies) {\n const [name, value] = cookie.trim().split(\"=\");\n if (name === \"_fbc\") {\n return value;\n }\n }\n\n return undefined;\n}\n\n/**\n * Get Facebook browser ID (fbp) from cookies\n */\nfunction getFacebookBrowserId(): string | undefined {\n if (typeof window === \"undefined\") {\n return undefined;\n }\n\n const cookies = document.cookie.split(\";\");\n for (const cookie of cookies) {\n const [name, value] = cookie.trim().split(\"=\");\n if (name === \"_fbp\") {\n return value;\n }\n }\n\n return undefined;\n}\n\n/**\n * Get client IP address (this will be handled server-side)\n */\nexport function getClientIpAddress(request?: Request): string | undefined {\n if (!request) {\n return undefined;\n }\n\n // Try different headers for IP address\n const forwarded = request.headers.get(\"x-forwarded-for\");\n const realIp = request.headers.get(\"x-real-ip\");\n const cfConnectingIp = request.headers.get(\"cf-connecting-ip\");\n\n return forwarded?.split(\",\")[0] || realIp || cfConnectingIp || undefined;\n}\n","import { BaseAdapter } from \"./base-adapter\";\nimport {\n type TEvent,\n EventType,\n type IPageViewEvent,\n type IAddToCartEvent,\n type IProductViewEvent,\n type IViewContentEvent,\n type ISearchEvent,\n} from \"../types\";\nimport type { TAdapterParams } from \"../adapter-params\";\nimport { generateEventId, getBrowserInfo } from \"../utils/event-id\";\n\ndeclare global {\n interface Window {\n fbq?: (\n type: string,\n eventName: string,\n params?: Record<string, any>,\n options?: { eventID?: string }\n ) => void;\n }\n}\n\n/**\n * Configuration for the Facebook Pixel adapter\n */\nexport interface PixelAdapterConfig {\n /**\n * Facebook Pixel ID\n */\n pixelId: string;\n /**\n * Enable server-side CAPI backup\n * @default true - CAPI is enabled by default for better data reliability\n */\n enableCAPI?: boolean;\n}\n\n/**\n * Facebook Pixel tracking adapter\n * Assumes that the Facebook Pixel script is already loaded in the page\n */\nexport class PixelAdapter extends BaseAdapter {\n public readonly name = \"FacebookPixel\";\n\n constructor(config: PixelAdapterConfig) {\n super(config);\n // Initialize logger with proper adapter name\n this.initializeLogger();\n }\n\n /**\n * Initialize the Facebook Pixel\n */\n public async initialize(): Promise<void> {\n if (this.initialized) {\n this.logger.debug(\"Already initialized\");\n return;\n }\n\n // Skip initialization if running on server\n if (typeof window === \"undefined\") {\n this.logger.debug(\"Skipping initialization on server\");\n return;\n }\n\n // Check if fbq is available\n if (!window.fbq) {\n this.logger.warn(\n \"Facebook Pixel fbq not found. Make sure the script is loaded.\"\n );\n return;\n } else {\n const pixelId = this.getConfig(\"pixelId\") as string;\n if (!pixelId) {\n this.logger.warn(\"Pixel ID not provided. Skipping fbq init.\");\n } else {\n // Check if pixel is already initialized to avoid duplicate init\n const isAlreadyInitialized =\n (window as any)._fbq?.loaded ||\n (window as any).fbq?.loaded ||\n document\n .querySelector(`script[src*=\"fbevents.js\"]`)\n ?.getAttribute(\"data-pixel-id\") === pixelId;\n\n if (!isAlreadyInitialized) {\n window.fbq(\"init\", pixelId);\n this.logger.info(\"Successfully initialized\", { pixelId: \"***\" });\n } else {\n this.logger.info(\"Pixel already initialized, skipping init\", {\n pixelId: \"***\",\n });\n }\n }\n }\n\n this.initialized = true;\n }\n\n /**\n * Track an event with Facebook Pixel\n * @param event The event to track\n * @param adapterParams Optional adapter-specific parameters\n */\n public async trackEvent(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n // Generate unique event ID for deduplication if not already present\n if (!event.eventId) {\n event.eventId = generateEventId(event.type);\n }\n\n // Ensure timestamp is always available for both client and server tracking\n if (!event.timestamp) {\n event.timestamp = Date.now();\n }\n\n this.logger.debug(\"Tracking event\", {\n eventType: event.type,\n eventId: event.eventId,\n timestamp: event.timestamp,\n });\n\n // Track client-side (browser pixel)\n await this.trackClientSide(event, adapterParams);\n\n // Track server-side (CAPI) as backup\n if (this.getConfig(\"enableCAPI\", true)) {\n await this.trackServerSide(event, adapterParams);\n }\n }\n\n /**\n * Track event on client-side (browser pixel)\n */\n private async trackClientSide(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n if (!this.initialized || typeof window === \"undefined\" || !window.fbq) {\n this.logger.warn(\n \"Cannot track client-side event, adapter not initialized or fbq not available\",\n {\n eventType: event.type,\n initialized: this.initialized,\n hasWindow: typeof window !== \"undefined\",\n hasFbq: !!window?.fbq,\n }\n );\n return;\n }\n\n const { eventName, enhancedParams } = this.formatEventPayload(\n event,\n adapterParams\n );\n\n // Skip tracking if eventName is empty (unknown event type)\n if (!eventName) {\n return;\n }\n\n // Add event_id for deduplication\n if (event.eventId) {\n window.fbq?.(\"track\", eventName, enhancedParams, {\n eventID: event.eventId,\n });\n } else {\n window.fbq?.(\"track\", eventName, enhancedParams);\n }\n\n this.logger.debug(`${eventName} event tracked on client-side`, {\n eventName,\n eventId: event.eventId,\n params: enhancedParams,\n });\n }\n\n /**\n * Track event on server-side (CAPI) as backup\n */\n private async trackServerSide(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n try {\n const { eventName, enhancedParams } = this.formatEventPayload(\n event,\n adapterParams\n );\n\n // Skip tracking if eventName is empty (unknown event type)\n if (!eventName) {\n return;\n }\n\n const browserInfo = getBrowserInfo();\n\n const payload = {\n eventName,\n eventId: event.eventId,\n timestamp: event.timestamp,\n enhancedParams: enhancedParams,\n userInfo: browserInfo,\n };\n\n const response = await fetch(\"/api/events\", {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify(payload),\n });\n\n if (!response.ok) {\n throw new Error(`Server-side tracking failed: ${response.status}`);\n }\n\n this.logger.debug(`${eventName} event sent to server-side`, {\n eventId: event.eventId,\n params: enhancedParams,\n });\n } catch (error) {\n this.logger.error(\n `FB Pixel: Server-side tracking failed for event ${event.type}: ${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n }\n\n /**\n * Utility function to format event payload for both client and server-side tracking\n */\n private formatEventPayload(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): { eventName: string; enhancedParams: Record<string, any> } {\n let baseParams: Record<string, any> = {};\n let eventName = \"\";\n\n switch (event.type) {\n case EventType.PAGE_VIEW:\n const pageViewEvent = event as IPageViewEvent;\n eventName = this.getEventName(adapterParams, \"PageView\");\n baseParams = {\n // Facebook-compatible parameters\n content_name: pageViewEvent.page_title || pageViewEvent.title,\n content_category: pageViewEvent.event_category,\n // Custom parameters that Facebook will accept in custom_data\n page_location: pageViewEvent.page_location || pageViewEvent.path,\n page_title: pageViewEvent.page_title || pageViewEvent.title,\n page_referrer: pageViewEvent.page_referrer || pageViewEvent.referrer,\n page_path: pageViewEvent.page_path || pageViewEvent.path,\n };\n break;\n\n case EventType.VIEW_CONTENT:\n const viewContentEvent = event as IViewContentEvent;\n eventName = this.getEventName(adapterParams, \"ViewContent\");\n baseParams = {\n content_type: viewContentEvent.content_type || \"product\",\n content_ids: viewContentEvent.content_ids || [],\n content_name: viewContentEvent.content_name || \"\",\n content_category: viewContentEvent.content_category || \"\",\n value: viewContentEvent.value || 0,\n currency: viewContentEvent.currency || \"INR\",\n };\n break;\n\n case EventType.PRODUCT_VIEW:\n const productEvent = event as IProductViewEvent;\n eventName = this.getEventName(adapterParams, \"ViewContent\");\n baseParams = {\n content_type: \"product_group\",\n content_ids: [productEvent.productId],\n content_name: productEvent.productName,\n content_category: productEvent.category,\n value: productEvent.price,\n currency: productEvent.currency || \"INR\",\n };\n break;\n\n case EventType.ADD_TO_CART:\n const cartEvent = event as IAddToCartEvent;\n eventName = this.getEventName(adapterParams, \"AddToCart\");\n baseParams = {\n content_type: \"product_group\",\n content_ids: [cartEvent.productId],\n content_name: cartEvent.productName,\n value: cartEvent.price,\n currency: cartEvent.currency || \"INR\",\n };\n break;\n\n case EventType.SEARCH:\n const searchEvent = event as ISearchEvent;\n eventName = this.getEventName(adapterParams, \"Search\");\n baseParams = {\n search_string: searchEvent.searchTerm,\n content_category: \"product\",\n content_ids: searchEvent.content_ids || [],\n };\n break;\n\n default:\n // Skip tracking for unknown event types\n this.logger.debug(`Skipping unknown event type: ${event.type}`);\n return { eventName: \"\", enhancedParams: {} };\n }\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n return { eventName, enhancedParams };\n }\n}\n","import { BaseAdapter } from \"./base-adapter\";\nimport {\n type TEvent,\n EventType,\n type IPageViewEvent,\n type IFormSubmissionEvent,\n type IViewItemEvent,\n type IViewedProductEvent,\n type IAddToCartEvent,\n type ICheckoutStartedEvent,\n type ICheckoutCompletedEvent,\n type ISearchEvent,\n type ICustomEvent,\n type IAddPaymentInfoEvent,\n type ICheckoutPaymentEvent,\n type IViewSearchResultsEvent,\n type IOrderPlacedEvent,\n type IBeginCheckoutEvent,\n type IViewContentEvent,\n type IPurchaseEvent,\n type ISpecificEvent,\n} from \"../types\";\nimport type { TAdapterParams } from \"../adapter-params\";\n\ndeclare global {\n interface Window {\n gtag?: (...args: any[]) => void;\n }\n}\n\n/**\n * Configuration for the Google Analytics adapter\n */\nexport interface GoogleAdapterConfig {\n /**\n * Google Analytics measurement ID (e.g., G-XXXXXXXXXX)\n */\n measurementId: string;\n}\n\n/**\n * Google Analytics tracking adapter\n * Assumes that the Google Analytics script is already loaded in the page\n */\nexport class GoogleAdapter extends BaseAdapter {\n public readonly name = \"GoogleAnalytics\";\n\n constructor(config: GoogleAdapterConfig) {\n super(config);\n // Initialize logger with proper adapter name\n this.initializeLogger();\n }\n\n /**\n * Initialize Google Analytics adapter\n * This just checks if gtag is available\n */\n public isEnabled(): boolean {\n const enabled =\n this.initialized && typeof window !== \"undefined\" && !!window.gtag;\n\n this.logger.debug(\"Checking if enabled\", {\n initialized: this.initialized,\n hasWindow: typeof window !== \"undefined\",\n hasGtag: !!window?.gtag,\n enabled,\n });\n\n return enabled;\n }\n\n public async initialize(): Promise<void> {\n this.logger.info(\"Starting initialization\");\n\n if (this.initialized) {\n this.logger.debug(\"Already initialized\");\n return;\n }\n\n // Skip initialization if running on server\n if (typeof window === \"undefined\") {\n this.logger.debug(\"Skipping initialization on server\");\n return;\n }\n\n const measurementId = this.getConfig<string>(\"measurementId\");\n this.logger.info(\"Using measurement ID\", { measurementId });\n\n // Check if gtag is available\n if (!window.gtag) {\n this.logger.warn(\n \"Google Analytics gtag not found. Make sure the script is loaded.\"\n );\n return;\n }\n\n this.initialized = true;\n this.logger.info(\"Successfully initialized\", { measurementId });\n }\n\n /**\n * Track an event with Google Analytics\n * @param event The event to track\n * @param adapterParams Optional adapter-specific parameters\n */\n public async trackEvent(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n this.logger.debug(\"Attempting to track event\", {\n initialized: this.initialized,\n hasWindow: typeof window !== \"undefined\",\n hasGtag: !!window?.gtag,\n eventType: event.type,\n });\n\n if (!this.initialized || typeof window === \"undefined\" || !window.gtag) {\n this.logger.warn(\n \"Cannot track event, adapter not initialized or gtag not available\",\n {\n eventType: event.type,\n initialized: this.initialized,\n hasWindow: typeof window !== \"undefined\",\n hasGtag: !!window?.gtag,\n }\n );\n return;\n }\n\n const measurementId = this.getConfig<string>(\"measurementId\");\n this.logger.info(\"Tracking event\", {\n eventType: event.type,\n measurementId,\n });\n\n switch (event.type) {\n case EventType.PAGE_VIEW:\n this.trackPageView(\n event as IPageViewEvent,\n measurementId,\n adapterParams\n );\n break;\n case EventType.USER_ENGAGEMENT:\n this.trackUserEngagement(adapterParams);\n break;\n case EventType.SESSION_START:\n this.trackSessionStart(adapterParams);\n break;\n case EventType.FIRST_VISIT:\n this.trackFirstVisit(adapterParams);\n break;\n case EventType.VIEW_ITEM:\n this.trackViewItem(event as IViewItemEvent, adapterParams);\n break;\n case EventType.VIEWED_PRODUCT:\n this.trackViewedProduct(event as IViewedProductEvent, adapterParams);\n break;\n case EventType.CHECKOUT_PAYMENT:\n this.trackCheckoutPayment(\n event as ICheckoutPaymentEvent,\n adapterParams\n );\n break;\n case EventType.SCROLL:\n this.trackScroll(adapterParams);\n break;\n case EventType.ADD_TO_CART:\n this.trackAddToCart(event as IAddToCartEvent, adapterParams);\n break;\n case EventType.FORM_START:\n this.trackFormStart(adapterParams);\n break;\n case EventType.CHECKOUT_STARTED:\n this.trackCheckoutStarted(\n event as ICheckoutStartedEvent,\n adapterParams\n );\n break;\n case EventType.FORM_SUBMISSION:\n this.trackFormSubmission(event as IFormSubmissionEvent, adapterParams);\n break;\n case EventType.SEARCH:\n this.trackSearch(event as ISearchEvent, adapterParams);\n break;\n case EventType.CHECKOUT_COMPLETED:\n this.trackCheckoutCompleted(\n event as ICheckoutCompletedEvent,\n adapterParams\n );\n break;\n case EventType.ADD_PAYMENT_INFO:\n this.trackAddPaymentInfo(event as IAddPaymentInfoEvent, adapterParams);\n break;\n case EventType.VIEW_CONTENT:\n this.trackViewContent(event as IViewContentEvent, adapterParams);\n break;\n case EventType.PURCHASE:\n this.trackPurchase(event as IPurchaseEvent, adapterParams);\n break;\n case EventType.VIEW_SEARCH_RESULTS:\n this.trackViewSearchResults(\n event as IViewSearchResultsEvent,\n adapterParams\n );\n break;\n case EventType.ORDER_PLACED:\n this.trackOrderPlaced(event as IOrderPlacedEvent, adapterParams);\n break;\n case EventType.BEGIN_CHECKOUT:\n this.trackBeginCheckout(event as IBeginCheckoutEvent, adapterParams);\n break;\n case EventType.CUSTOM:\n this.trackCustomEvent(event as ICustomEvent, adapterParams);\n break;\n case EventType.SPECIFIC:\n this.trackSpecificEvent(event as ISpecificEvent, adapterParams);\n break;\n default: {\n this.logger.debug(\"Unhandled event type\", { eventType: event.type });\n break;\n }\n }\n }\n\n /**\n * Track a page view event\n * @param event The page view event\n * @param measurementId The Google Analytics measurement ID\n */\n private trackPageView(\n event: IPageViewEvent,\n measurementId: string,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"page_view\");\n const baseParams = {\n page_path: event.path,\n page_title: event.title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer || undefined,\n ga_session_id: event.ga_session_id || \"\",\n ga_session_number: event.ga_session_number || \"\",\n gclid: event.gclid || \"\",\n engagement_time_msec: event.engagement_time_msec,\n campaign: event.campaign || \"\",\n campaign_id: event.campaign_id || \"\",\n content: event.content || \"\",\n debug_mode: event.debug_mode,\n medium: event.medium || \"\",\n source: event.page_source || \"\",\n page_term: event.page_term || \"\",\n ignore_referrer: event.ignore_referrer,\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n user_id: event.user_id,\n user_properties: event.user_properties,\n session_engaged: event.session_engaged,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackUserEngagement(adapterParams?: TAdapterParams): void {\n const eventName = this.getEventName(adapterParams, \"user_engagement\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n campaign: \"\",\n ga_session_id: \"\",\n ga_session_number: \"\",\n gclid: \"\",\n ignore_referrer: \"\",\n medium: \"\",\n page_location: window.location.href,\n page_path: window.location.pathname,\n page_referrer: document.referrer,\n page_title: document.title,\n source: \"\",\n srsltid: \"\",\n term: \"\",\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackSessionStart(adapterParams?: TAdapterParams): void {\n const eventName = this.getEventName(adapterParams, \"session_start\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n campaign: \"\",\n campaign_id: \"\",\n content: \"\",\n ga_session_id: \"\",\n ga_session_number: \"\",\n gclid: \"\",\n ignore_referrer: \"\",\n medium: \"\",\n page_location: window.location.href,\n page_path: window.location.pathname,\n page_referrer: document.referrer,\n page_title: document.title,\n source: \"\",\n srsltid: \"\",\n term: \"\",\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackFirstVisit(adapterParams?: TAdapterParams): void {\n const eventName = this.getEventName(adapterParams, \"first_visit\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n campaign: \"\",\n campaign_id: \"\",\n content: \"\",\n debug_mode: \"\",\n engagement_time_msec: 0,\n entrances: 1,\n ga_session_id: \"\",\n ga_session_number: \"\",\n gclid: \"\",\n ignore_referrer: \"\",\n medium: \"\",\n page_location: window.location.href,\n page_path: window.location.pathname,\n page_referrer: document.referrer,\n page_title: document.title,\n page_source: \"\",\n page_term: \"\",\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackViewItem(\n event: IViewItemEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"view_item\");\n const baseParams = {\n currency: event.currency,\n value: event.value,\n items: event.items,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackScroll(adapterParams?: TAdapterParams): void {\n const eventName = this.getEventName(adapterParams, \"scroll\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n engagement_time_msec: undefined,\n ga_session_id: \"\",\n ga_session_number: \"\",\n ignore_referrer: undefined,\n page_location: window.location.href,\n page_path: window.location.pathname,\n page_referrer: document.referrer,\n page_title: document.title,\n percent_scrolled: 90, // or dynamically captured\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackCheckoutPayment(\n event: ICheckoutPaymentEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"checkout_payment\");\n const baseParams = {\n currency: event.currency,\n value: event.cartValue,\n items: event.items.map((item) => ({\n item_id: item.productId,\n item_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackViewedProduct(\n event: IViewedProductEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"viewed_product\");\n const baseParams = {\n currency: event.currency,\n value: event.price,\n items: [\n {\n item_id: event.productId,\n item_name: event.productName,\n price: event.price,\n currency: event.currency,\n },\n ],\n view_duration: event.viewDuration,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackAddToCart(\n event: IAddToCartEvent,\n adapterParams?: TAdapterParams\n ): void {\n this.logger.debug(\"Processing add_to_cart event\", { event });\n\n const eventName = this.getEventName(adapterParams, \"add_to_cart\");\n const baseParams = {\n currency: event.currency || \"INR\",\n value: event.price * (event.quantity || 1), // Calculate total value\n items: [\n {\n item_id: event.productId,\n item_name: event.productName,\n price: event.price,\n quantity: event.quantity || 1,\n item_variant: event.variant,\n },\n ],\n // GA4-specific parameters\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n campaign: event.campaign || \"\",\n ga_session_id: event.ga_session_id || \"\",\n engagement_time_msec: event.engagement_time_msec,\n session_engaged: event.session_engaged,\n page_title: event.page_title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer,\n user_id: event.user_id,\n user_properties: event.user_properties,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n this.logger.debug(\"Sending add_to_cart event to gtag\", {\n eventName,\n params: enhancedParams,\n });\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackFormStart(adapterParams?: TAdapterParams): void {\n const eventName = this.getEventName(adapterParams, \"form_start\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n engagement_time_msec: undefined,\n first_field_id: \"\",\n first_field_name: \"\",\n first_field_position: \"\",\n first_field_type: \"\",\n form_destination: \"\",\n form_length: \"\",\n form_id: \"\",\n ga_session_id: \"\",\n ga_session_number: \"\",\n ignore_referrer: undefined,\n page_location: window.location.href,\n page_referrer: document.referrer,\n page_title: document.title,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackCheckoutStarted(\n event: ICheckoutStartedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"begin_checkout\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n currency: event.currency || \"INR\",\n ecomm_pagetype: \"\",\n ecomm_prodid: \"\",\n ecomm_totalvalue: event.cartValue,\n value: event.cartValue,\n engagement_time_msec: undefined,\n ga_session_id: \"\",\n ga_session_number: \"\",\n ignore_referrer: undefined,\n page_location: window.location.href,\n page_referrer: document.referrer,\n page_title: document.title,\n items: event.items.map((item) => ({\n item_id: item.productId,\n item_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n item_variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackFormSubmission(\n event: IFormSubmissionEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"form_submit\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n engagement_time_msec: undefined,\n form_destination: \"\",\n form_length: \"\",\n form_id: event.formId,\n ga_session_id: \"\",\n ga_session_number: \"\",\n ignore_referrer: undefined,\n page_location: window.location.href,\n page_referrer: document.referrer,\n page_title: document.title,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackSearch(\n event: ISearchEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"search\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n engagement_time_msec: undefined,\n ga_session_id: \"\",\n ga_session_number: \"\",\n ignore_referrer: undefined,\n page_location: window.location.href,\n page_referrer: document.referrer,\n page_title: document.title,\n search_term: event.searchTerm,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a checkout completed event\n * @param event The checkout completed event\n */\n private trackCheckoutCompleted(\n event: ICheckoutCompletedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"purchase\");\n const baseParams = {\n transaction_id: event.orderId,\n currency: event.currency || \"INR\",\n value: event.cartValue,\n items: event.items.map((item) => ({\n item_id: item.productId,\n item_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n item_variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a custom event\n * @param event The custom event\n */\n private trackCustomEvent(\n event: ICustomEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"custom\"); // this will support all custom event\n const affiliateParams = this.enhanceWithAffiliateParams({});\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a specific event\n * @param event The specific event\n */\n private trackSpecificEvent(\n event: ISpecificEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"\"); // this will support all custom event\n if (!eventName) {\n return;\n }\n const affiliateParams = this.enhanceWithAffiliateParams({});\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track an add payment info event\n * @param event The add payment info event\n */\n private trackAddPaymentInfo(\n event: IAddPaymentInfoEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"add_payment_info\");\n const baseParams = {\n currency: event.currency || \"INR\",\n value: event.cartValue,\n payment_type: event.paymentType || \"\",\n items: event.items.map((item) => ({\n item_id: item.productId,\n item_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n item_variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a view content event\n * @param event The view content event\n */\n private trackViewContent(\n event: IViewContentEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"view_item\");\n const baseParams = {\n currency: event.currency || \"USD\",\n value: event.value || 0,\n items:\n event.items?.map((item: any) => ({\n item_id: item.item_id,\n item_name: item.item_name,\n item_category: item.item_category,\n item_category2: item.item_category2,\n item_category3: item.item_category3,\n item_category4: item.item_category4,\n item_category5: item.item_category5,\n item_brand: item.item_brand,\n item_variant: item.item_variant,\n price: item.price,\n quantity: item.quantity,\n })) || [],\n // GA4-specific parameters\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n campaign: event.campaign || \"\",\n ga_session_id: event.ga_session_id || \"\",\n engagement_time_msec: event.engagement_time_msec,\n session_engaged: event.session_engaged,\n page_title: event.page_title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer,\n user_id: event.user_id,\n user_properties: event.user_properties,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a purchase event\n * @param event The purchase event\n */\n private trackPurchase(\n event: IPurchaseEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"purchase\");\n const baseParams = {\n transaction_id: event.transaction_id,\n value: event.value,\n currency: event.currency || \"USD\",\n coupon: event.coupon,\n shipping: event.shipping,\n tax: event.tax,\n items:\n event.items?.map((item: any) => ({\n item_id: item.item_id,\n item_name: item.item_name,\n item_category: item.item_category,\n item_category2: item.item_category2,\n item_category3: item.item_category3,\n item_category4: item.item_category4,\n item_category5: item.item_category5,\n item_brand: item.item_brand,\n item_variant: item.item_variant,\n price: item.price,\n quantity: item.quantity,\n })) || [],\n // GA4-specific parameters\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n campaign: event.campaign || \"\",\n ga_session_id: event.ga_session_id || \"\",\n engagement_time_msec: event.engagement_time_msec,\n session_engaged: event.session_engaged,\n page_title: event.page_title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer,\n user_id: event.user_id,\n user_properties: event.user_properties,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a view search results event\n * @param event The view search results event\n */\n private trackViewSearchResults(\n event: IViewSearchResultsEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"view_search_results\");\n const baseParams = {\n search_term: event.search_term,\n // GA4-specific parameters\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n campaign: event.campaign || \"\",\n ga_session_id: event.ga_session_id || \"\",\n engagement_time_msec: event.engagement_time_msec,\n session_engaged: event.session_engaged,\n page_title: event.page_title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer,\n user_id: event.user_id,\n user_properties: event.user_properties,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track an order placed event\n * @param event The order placed event\n */\n private trackOrderPlaced(\n event: IOrderPlacedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"purchase\");\n const baseParams = {\n transaction_id: event.order_id,\n value: event.total_amount,\n currency: event.currency || \"USD\",\n items:\n event.items?.map((item: any) => ({\n item_id: item.product_id,\n item_name: item.product_name,\n item_category: item.category,\n item_brand: item.brand,\n item_variant: item.variant,\n price: item.price,\n quantity: item.quantity,\n })) || [],\n // GA4-specific parameters\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n campaign: event.campaign || \"\",\n ga_session_id: event.ga_session_id || \"\",\n engagement_time_msec: event.engagement_time_msec,\n session_engaged: event.session_engaged,\n page_title: event.page_title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer,\n user_id: event.user_id,\n user_properties: event.user_properties,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a begin checkout event\n * @param event The begin checkout event\n */\n private trackBeginCheckout(\n event: IBeginCheckoutEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"begin_checkout\");\n const baseParams = {\n currency: event.currency || \"USD\",\n value: event.value || 0,\n coupon: event.coupon,\n items:\n event.items?.map((item: any) => ({\n item_id: item.item_id,\n item_name: item.item_name,\n item_category: item.item_category,\n item_category2: item.item_category2,\n item_category3: item.item_category3,\n item_category4: item.item_category4,\n item_category5: item.item_category5,\n item_brand: item.item_brand,\n item_variant: item.item_variant,\n price: item.price,\n quantity: item.quantity,\n })) || [],\n // GA4-specific parameters\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n campaign: event.campaign || \"\",\n ga_session_id: event.ga_session_id || \"\",\n engagement_time_msec: event.engagement_time_msec,\n session_engaged: event.session_engaged,\n page_title: event.page_title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer,\n user_id: event.user_id,\n user_properties: event.user_properties,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Convert an event to Google Analytics parameters\n * @param event The event to convert\n */\n private eventToParams(event: TEvent): Record<string, any> {\n // Remove type and timestamp from the event\n const { type, timestamp, ...params } = event;\n const affiliateParams = this.enhanceWithAffiliateParams(params);\n return this.enhanceWithExperimentParams(affiliateParams);\n }\n}\n","/**\n * Event name constants for tracking platforms\n */\nexport const EventNames = {\n // Page events\n PAGE_VIEWED: \"Page Viewed\",\n\n // Interaction events\n BUTTON_CLICKED: \"Button Clicked\",\n FORM_SUBMITTED: \"Form Submitted\",\n\n // Product events\n PRODUCT_VIEWED: \"Product Viewed\",\n ADDED_TO_CART: \"Added To Cart\",\n REMOVED_FROM_CART: \"Removed From Cart\",\n\n // Checkout events\n CHECKOUT_STARTED: \"Checkout Started\",\n PURCHASE_COMPLETED: \"Purchase Completed\",\n\n // Search events\n SEARCH: \"Search\",\n\n // User events\n USER_SIGNED_UP: \"User Signed Up\",\n USER_LOGGED_IN: \"User Logged In\",\n};\n\n/**\n * Default currency code\n */\nexport const DEFAULT_CURRENCY = \"INR\";\n","import { BaseAdapter } from \"./base-adapter\";\nimport {\n type TEvent,\n EventType,\n type IPageViewEvent,\n type IButtonClickEvent,\n type IFormSubmissionEvent,\n type IProductViewEvent,\n type IAddToCartEvent,\n type IRemoveFromCartEvent,\n type ICheckoutStartedEvent,\n type ICheckoutCompletedEvent,\n type ISearchEvent,\n type IUserSignupEvent,\n type IUserLoginEvent,\n type ICustomEvent,\n ISpecificEvent,\n} from \"../types\";\nimport { EventNames, DEFAULT_CURRENCY } from \"../constants\";\nimport type { TAdapterParams } from \"../adapter-params\";\n\nimport moengage from \"@moengage/web-sdk\";\n\n/**\n * Configuration for the MoenGage adapter\n */\nexport interface MoengageAdapterConfig {\n /**\n * MoenGage App ID (required for SDK initialization)\n */\n appId: string;\n /**\n * Whether to enable user identification\n */\n enableUserIdentification?: boolean;\n /**\n * Debug mode for MoenGage SDK\n */\n debug?: boolean;\n /**\n * Data center region (default: 'dc_01')\n */\n region?: string;\n}\n\n/**\n * MoenGage tracking adapter using the official Web SDK\n */\nexport class MoengageAdapter extends BaseAdapter {\n public readonly name = \"MoEngage\";\n\n constructor(config: MoengageAdapterConfig) {\n super(config);\n // Initialize logger with proper adapter name\n this.initializeLogger();\n }\n\n /**\n * Get MoEngage SDK instance from window\n */\n private getMoengageSDK(): any {\n return typeof window !== \"undefined\" ? (window as any).Moengage : null;\n }\n\n /**\n * Initialize the MoenGage adapter using the Web SDK\n */\n public async initialize(): Promise<void> {\n if (this.initialized) {\n this.logger.debug(\"Already initialized\");\n return;\n }\n\n // Skip initialization if running on server\n if (typeof window === \"undefined\") {\n this.logger.debug(\"Skipping initialization on server\");\n return;\n }\n\n try {\n const appId = this.getConfig<string>(\"appId\");\n if (!appId) {\n this.logger.warn(\"App ID is required for initialization\");\n return;\n }\n\n // Initialize MoenGage SDK\n const region = this.getConfig<string>(\"region\", \"dc_01\");\n const config = {\n app_id: appId,\n debug_logs: this.getConfig<boolean>(\"debug\", false) ? 1 : 0,\n cluster: region,\n disableCookies: true,\n };\n\n this.logger.info(\"Initializing with config\", {\n region,\n debug: this.getConfig<boolean>(\"debug\", false),\n appId: \"***\", // Hide sensitive data\n });\n\n // Initialize MoEngage SDK\n moengage.initialize(config);\n\n this.logger.info(\"Successfully initialized with SDK\");\n this.initialized = true;\n } catch (error) {\n this.logger.error(\"Failed to initialize\", error as Error);\n }\n }\n\n /**\n * Track an event with MoenGage SDK\n * @param event The event to track\n * @param adapterParams Optional adapter-specific parameters\n */\n public async trackEvent(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n const sdk = this.getMoengageSDK();\n if (!this.initialized || typeof window === \"undefined\" || !sdk) {\n this.logger.warn(\n \"Cannot track event, adapter not initialized or SDK not available\",\n {\n eventType: event.type,\n initialized: this.initialized,\n hasWindow: typeof window !== \"undefined\",\n hasSdk: !!sdk,\n }\n );\n return;\n }\n\n this.logger.debug(\"Tracking event\", { eventType: event.type });\n\n switch (event.type) {\n case EventType.PAGE_VIEW:\n this.trackPageView(event as IPageViewEvent, adapterParams);\n break;\n case EventType.BUTTON_CLICK:\n this.trackButtonClick(event as IButtonClickEvent, adapterParams);\n break;\n case EventType.FORM_SUBMISSION:\n this.trackFormSubmission(event as IFormSubmissionEvent, adapterParams);\n break;\n case EventType.PRODUCT_VIEW:\n this.trackProductView(event as IProductViewEvent, adapterParams);\n break;\n case EventType.ADD_TO_CART:\n this.trackAddToCart(event as IAddToCartEvent, adapterParams);\n break;\n case EventType.REMOVE_FROM_CART:\n this.trackRemoveFromCart(event as IRemoveFromCartEvent, adapterParams);\n break;\n case EventType.CHECKOUT_STARTED:\n this.trackCheckoutStarted(\n event as ICheckoutStartedEvent,\n adapterParams\n );\n break;\n case EventType.CHECKOUT_COMPLETED:\n this.trackCheckoutCompleted(\n event as ICheckoutCompletedEvent,\n adapterParams\n );\n break;\n case EventType.SEARCH:\n this.trackSearch(event as ISearchEvent, adapterParams);\n break;\n case EventType.USER_SIGNUP:\n this.trackUserSignup(event as IUserSignupEvent, adapterParams);\n break;\n case EventType.USER_LOGIN:\n this.trackUserLogin(event as IUserLoginEvent, adapterParams);\n break;\n case EventType.CUSTOM:\n this.trackCustomEvent(event as ICustomEvent, adapterParams);\n break;\n case EventType.SPECIFIC:\n this.trackSpecificEvent(event as ISpecificEvent, adapterParams);\n break;\n default:\n this.logger.debug(\"Unhandled event type\", { eventType: event.type });\n break;\n }\n }\n\n /**\n * Track a page view event\n * @param event The page view event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackPageView(\n event: IPageViewEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n if (!sdk?.track_event) {\n this.logger.warn(\"track_event not available for PageView event\");\n return;\n }\n\n const eventName = this.getEventName(adapterParams, EventNames.PAGE_VIEWED);\n const baseEventData = {\n page_path: event.path,\n page_title: event.title,\n page_url: window.location.href,\n referrer: event.referrer || document.referrer,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n\n sdk.track_event(eventName, enhancedEventData);\n this.logger.debug(\"PageView event tracked\", {\n eventName,\n eventData: enhancedEventData,\n });\n }\n\n /**\n * Track a button click event\n * @param event The button click event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackButtonClick(\n event: IButtonClickEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.BUTTON_CLICKED\n );\n const baseEventData = {\n button_id: event.buttonId,\n button_text: event.buttonText,\n location: event.location,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a form submission event\n * @param event The form submission event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackFormSubmission(\n event: IFormSubmissionEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.FORM_SUBMITTED\n );\n const baseEventData = {\n form_id: event.formId,\n form_name: event.formName,\n success: event.success,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a product view event\n * @param event The product view event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackProductView(\n event: IProductViewEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.PRODUCT_VIEWED\n );\n const baseEventData = {\n product_id: event.productId,\n product_name: event.productName,\n price: event.price,\n currency: event.currency || DEFAULT_CURRENCY,\n category: event.category,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track an add to cart event\n * @param event The add to cart event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackAddToCart(\n event: IAddToCartEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const quantity = event.quantity ?? 1;\n const eventName = this.getEventName(\n adapterParams,\n EventNames.ADDED_TO_CART\n );\n const baseEventData = {\n product_id: event.productId,\n product_name: event.productName,\n price: event.price,\n currency: event.currency || DEFAULT_CURRENCY,\n quantity: quantity,\n variant: event.variant,\n total_value: event.price * quantity,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a remove from cart event\n * @param event The remove from cart event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackRemoveFromCart(\n event: IRemoveFromCartEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.REMOVED_FROM_CART\n );\n const baseEventData = {\n product_id: event.productId,\n product_name: event.productName,\n price: event.price,\n currency: event.currency || DEFAULT_CURRENCY,\n quantity: event.quantity,\n variant: event.variant,\n total_value: event.price * event.quantity,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a checkout started event\n * @param event The checkout started event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackCheckoutStarted(\n event: ICheckoutStartedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.CHECKOUT_STARTED\n );\n const baseEventData = {\n cart_value: event.cartValue,\n currency: event.currency || DEFAULT_CURRENCY,\n item_count: event.itemCount,\n items: event.items.map((item) => ({\n product_id: item.productId,\n product_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a checkout completed event\n * @param event The checkout completed event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackCheckoutCompleted(\n event: ICheckoutCompletedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.PURCHASE_COMPLETED\n );\n const baseEventData = {\n order_id: event.orderId,\n cart_value: event.cartValue,\n currency: event.currency || DEFAULT_CURRENCY,\n item_count: event.itemCount,\n items: event.items.map((item) => ({\n product_id: item.productId,\n product_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a search event\n * @param event The search event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackSearch(\n event: ISearchEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(adapterParams, EventNames.SEARCH);\n const baseEventData = {\n search_term: event.searchTerm,\n results_count: event.resultsCount,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a user signup event\n * @param event The user signup event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackUserSignup(\n event: IUserSignupEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.USER_SIGNED_UP\n );\n const baseEventData = {\n user_id: event.userId,\n method: event.method,\n success: event.success,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n\n // Add user attributes if user ID is available and user identification is enabled\n if (\n event.userId &&\n this.getConfig<boolean>(\"enableUserIdentification\", true)\n ) {\n sdk?.add_unique_user_id(event.userId);\n }\n }\n\n /**\n * Track a user login event\n * @param event The user login event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackUserLogin(\n event: IUserLoginEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.USER_LOGGED_IN\n );\n const baseEventData = {\n user_id: event.userId,\n method: event.method,\n success: event.success,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n\n // Add user attributes if user ID is available and user identification is enabled\n if (\n event.userId &&\n this.getConfig<boolean>(\"enableUserIdentification\", true)\n ) {\n sdk?.add_unique_user_id(event.userId);\n }\n }\n\n /**\n * Track a custom event\n * @param event The custom event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackCustomEvent(\n event: ICustomEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(adapterParams, \"custom\");\n const affiliateParams = this.enhanceWithAffiliateParams({});\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a specific event\n * @param event The specific event\n */\n private trackSpecificEvent(\n event: ISpecificEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(adapterParams, \"\"); // this will support all custom event\n if (!eventName) {\n return;\n }\n const affiliateParams = this.enhanceWithAffiliateParams({});\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n sdk?.track_event(eventName, enhancedParams);\n }\n\n /**\n * Set user attributes in MoenGage\n * @param attributes User attributes to set\n */\n public setUserAttributes(attributes: Record<string, any>): void {\n const sdk = this.getMoengageSDK();\n if (!this.initialized || typeof window === \"undefined\" || !sdk) {\n this.logger.warn(\"Cannot set user attributes, adapter not initialized\");\n return;\n }\n\n this.logger.debug(\"Setting user attributes\", {\n attributeKeys: Object.keys(attributes),\n });\n\n // Set user attributes\n for (const [key, value] of Object.entries(attributes)) {\n switch (key) {\n case \"firstName\":\n sdk?.add_first_name(value as string);\n break;\n case \"lastName\":\n sdk?.add_last_name(value as string);\n break;\n case \"email\":\n sdk?.add_email(value as string);\n break;\n case \"mobile\":\n sdk?.add_mobile(value as string);\n break;\n case \"username\":\n sdk?.add_user_name(value as string);\n break;\n case \"gender\":\n sdk?.add_gender(value as string);\n break;\n case \"birthday\":\n sdk?.add_birthday(value as Date);\n break;\n case \"userId\":\n sdk?.add_unique_user_id(value as string);\n break;\n default:\n sdk?.add_user_attribute(key, value);\n break;\n }\n }\n }\n}\n","import { BaseAdapter } from \"./base-adapter\";\nimport {\n type TEvent,\n EventType,\n type IPageViewEvent,\n type IAddToCartEvent,\n type ICheckoutStartedEvent,\n type ICheckoutCompletedEvent,\n type IProductViewEvent,\n type ISearchEvent,\n type IAddPaymentInfoEvent,\n ICustomEvent,\n ISpecificEvent,\n} from \"../types\";\nimport type { TAdapterParams } from \"../adapter-params\";\n\ndeclare global {\n interface Window {\n posthog?: {\n capture: (eventName: string, properties?: Record<string, any>) => void;\n identify: (distinctId: string, properties?: Record<string, any>) => void;\n };\n }\n}\n\n/**\n * Configuration for the PostHog adapter\n */\nexport interface PostHogAdapterConfig {\n /**\n * Enable debug logs\n */\n enableDebugLogs?: boolean;\n}\n\n/**\n * PostHog tracking adapter\n * Assumes that the PostHog script is already loaded in the page\n */\nexport class PostHogAdapter extends BaseAdapter {\n public readonly name = \"PostHog\";\n\n constructor(config: PostHogAdapterConfig = {}) {\n super(config);\n // Initialize logger with proper adapter name\n this.initializeLogger();\n }\n\n /**\n * Initialize PostHog\n */\n public async initialize(): Promise<void> {\n if (this.initialized) {\n this.logger.debug(\"Already initialized\");\n return;\n }\n\n // Skip initialization if running on server\n if (typeof window === \"undefined\") {\n this.logger.debug(\"Skipping initialization on server\");\n return;\n }\n\n // Check if posthog is available\n if (!window.posthog) {\n this.logger.warn(\n \"PostHog object not found. Make sure the script is loaded.\"\n );\n return;\n }\n\n this.logger.info(\"Successfully initialized\");\n this.initialized = true;\n }\n\n /**\n * Track an event with PostHog\n * @param event The event to track\n * @param adapterParams Optional adapter-specific parameters\n */\n public async trackEvent(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n if (!this.initialized || typeof window === \"undefined\" || !window.posthog) {\n this.logger.warn(\n \"Cannot track event, adapter not initialized or posthog not available\",\n {\n eventType: event.type,\n initialized: this.initialized,\n hasWindow: typeof window !== \"undefined\",\n hasPosthog: !!window?.posthog,\n }\n );\n return;\n }\n\n this.logger.debug(\"Tracking event\", { eventType: event.type });\n\n switch (event.type) {\n case EventType.PAGE_VIEW:\n this.trackPageView(event, adapterParams);\n break;\n case EventType.PRODUCT_VIEW:\n this.trackProductView(event, adapterParams);\n break;\n case EventType.ADD_TO_CART:\n this.trackAddToCart(event, adapterParams);\n break;\n case EventType.CHECKOUT_STARTED:\n this.trackCheckoutStarted(event, adapterParams);\n break;\n case EventType.CHECKOUT_COMPLETED:\n this.trackCheckoutCompleted(event, adapterParams);\n break;\n case EventType.ADD_PAYMENT_INFO:\n this.trackAddPaymentInfo(event, adapterParams);\n break;\n case EventType.SEARCH:\n this.trackSearch(event, adapterParams);\n break;\n case EventType.CUSTOM:\n this.trackCustomEvent(event, adapterParams);\n break;\n case EventType.SPECIFIC:\n this.trackSpecificEvent(event, adapterParams);\n break;\n default:\n this.logger.debug(\"Unhandled event type\", { eventType: event.type });\n break;\n }\n }\n\n /**\n * Track a custom event\n * @param event The custom event\n */\n private trackCustomEvent(\n event: ICustomEvent,\n adapterParams?: TAdapterParams\n ): void {\n // For other events, track as custom events\n const eventName = this.getEventName(adapterParams, \"custom\");\n const affiliateParams = this.enhanceWithAffiliateParams({});\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n private trackSpecificEvent(\n event: ISpecificEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"\"); // this will support all custom event\n if (!eventName) {\n return;\n }\n const affiliateParams = this.enhanceWithAffiliateParams({});\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n /**\n * Track a page view event\n * @param event The page view event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackPageView(\n event: IPageViewEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"$pageview\");\n const baseParams = {\n path: event.path,\n title: event.title,\n referrer: event.referrer,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n /**\n * Track a product view event\n * @param event The product view event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackProductView(\n event: IProductViewEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"product_viewed\");\n const baseParams = {\n product_id: event.productId,\n product_name: event.productName,\n category: event.category,\n price: event.price,\n currency: event.currency || \"INR\",\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n /**\n * Track an add to cart event\n * @param event The add to cart event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackAddToCart(\n event: IAddToCartEvent,\n adapterParams?: TAdapterParams\n ): void {\n this.logger.debug(\"Handling ADD_TO_CART event\", { event });\n\n const eventName = this.getEventName(adapterParams, \"add_to_cart\");\n const baseParams = {\n product_id: event.productId,\n product_name: event.productName,\n price: event.price,\n currency: event.currency || \"INR\",\n quantity: event.quantity,\n variant: event.variant,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n this.logger.debug(\"Sending to PostHog with params\", {\n eventName,\n params: enhancedParams,\n });\n\n if (!window.posthog) {\n this.logger.error(\"window.posthog is not available\");\n return;\n }\n\n window.posthog.capture(eventName, enhancedParams);\n this.logger.debug(\"Event sent to PostHog successfully\");\n }\n\n /**\n * Track a checkout started event\n * @param event The checkout started event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackCheckoutStarted(\n event: ICheckoutStartedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"checkout_started\");\n const baseParams = {\n cart_value: event.cartValue,\n currency: event.currency || \"INR\",\n item_count: event.itemCount,\n items: event.items.map((item) => ({\n product_id: item.productId,\n product_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n /**\n * Track a checkout completed event\n * @param event The checkout completed event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackCheckoutCompleted(\n event: ICheckoutCompletedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"purchase\");\n const baseParams = {\n order_id: event.orderId,\n cart_value: event.cartValue,\n currency: event.currency || \"INR\",\n item_count: event.itemCount,\n items: event.items.map((item) => ({\n product_id: item.productId,\n product_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n /**\n * Track a search event\n * @param event The search event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackSearch(\n event: ISearchEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"search\");\n const baseParams = {\n search_term: event.searchTerm,\n results_count: event.resultsCount,\n content_ids: event.content_ids,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n /**\n * Track an add payment info event\n * @param event The add payment info event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackAddPaymentInfo(\n event: IAddPaymentInfoEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"add_payment_info\");\n const baseParams = {\n cart_value: event.cartValue,\n currency: event.currency || \"INR\",\n payment_type: event.paymentType,\n item_count: event.itemCount,\n items: event.items.map((item) => ({\n product_id: item.productId,\n product_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n}\n","import {\n sendShopifyAnalytics,\n getClientBrowserParameters,\n AnalyticsEventName,\n type ShopifyAnalyticsProduct,\n type ShopifyPageViewPayload,\n ShopifySalesChannel,\n} from \"@shopify/hydrogen-react\";\nimport { BaseAdapter } from \"./base-adapter\";\nimport type { TEvent } from \"../types\";\nimport { EventType } from \"../types\";\nimport type { TAdapterParams } from \"../adapter-params\";\n\n// Constants based on environment variables\nconst DEFAULT_CURRENCY = \"INR\";\nconst DEFAULT_LANGUAGE = \"en\";\n\n/**\n * Type definitions for analytics payloads\n */\ntype SendPageViewPayload = {\n pageType?: string;\n products?: ShopifyAnalyticsProduct[];\n collectionHandle?: string;\n searchString?: string;\n totalValue?: number;\n cartId?: string;\n};\n\ntype SendAddToCartPayload = {\n cartId: string;\n products?: ShopifyAnalyticsProduct[];\n totalValue?: ShopifyPageViewPayload[\"totalValue\"];\n};\n\n/**\n * Configuration interface for Shopify Analytics Adapter\n */\nexport interface ShopifyAdapterConfig {\n shopId: string;\n domain: string;\n currency?: string;\n}\n\n/**\n * Shopify Analytics Adapter for tracking events to Shopify Admin Analytics\n */\nexport class ShopifyAdapter extends BaseAdapter {\n public readonly name = \"ShopifyAnalytics\";\n private shopId: string;\n private domain: string;\n private currency: string;\n\n constructor(config: ShopifyAdapterConfig) {\n super(config);\n this.shopId = config.shopId;\n this.domain = config.domain;\n this.currency = config.currency || DEFAULT_CURRENCY;\n }\n\n /**\n * Initialize the Shopify Analytics adapter\n */\n public async initialize(): Promise<void> {\n this.initializeLogger();\n\n try {\n // Enable Shopify cookies for attribution tracking\n if (typeof window !== \"undefined\") {\n this.logger.info(\"Shopify Analytics adapter initialized successfully\", {\n shopId: this.shopId,\n domain: this.domain,\n currency: this.currency,\n });\n\n // Note: useShopifyCookies should be called at the React component level\n // This will be handled in the ShopkitAnalytics component\n }\n this.initialized = true;\n } catch (error) {\n this.logger.error(\n \"Failed to initialize Shopify Analytics\",\n error instanceof Error ? error : new Error(String(error))\n );\n throw error;\n }\n }\n\n /**\n * Convert product ID to Shopify GID format if it's not already\n */\n private formatProductGid(productId: string): string {\n return productId.startsWith(\"gid://shopify/Product/\")\n ? productId\n : `gid://shopify/Product/${productId}`;\n }\n\n /**\n * Get user consent status from your consent management system\n */\n private getUserConsent() {\n // Check if user has given consent (implement based on your consent system)\n const hasConsent = this.checkUserConsent();\n\n return {\n hasUserConsent: hasConsent,\n analyticsAllowed: hasConsent && this.checkAnalyticsConsent(),\n marketingAllowed: hasConsent && this.checkMarketingConsent(),\n saleOfDataAllowed: hasConsent && this.checkSaleOfDataConsent(),\n };\n }\n\n /**\n * Check user consent (implement based on your consent management)\n */\n private checkUserConsent(): boolean {\n // Example implementations:\n // return localStorage.getItem('user_consent') === 'true';\n // return document.cookie.includes('consent=granted');\n // return window.gtag?.('consent', 'query') === 'granted';\n\n // For now, return true but implement your actual consent logic\n return true;\n }\n\n private checkAnalyticsConsent(): boolean {\n // Implement your analytics consent check\n return true;\n }\n\n private checkMarketingConsent(): boolean {\n // Implement your marketing consent check\n return true;\n }\n\n private checkSaleOfDataConsent(): boolean {\n // Implement your sale of data consent check\n return true;\n }\n\n /**\n * Create base payload for all events\n */\n private createBasePayload(): any {\n const consent = this.getUserConsent();\n\n return {\n ...getClientBrowserParameters(),\n ...consent,\n storefrontId: this.shopId,\n shopifySalesChannel: ShopifySalesChannel.headless,\n shopId: `gid://shopify/Shop/${this.shopId}`,\n currency: this.currency,\n acceptedLanguage: DEFAULT_LANGUAGE,\n ...this.enhanceWithExperimentParams(this.enhanceWithAffiliateParams({})),\n };\n }\n\n /**\n * Send page view event using the enhanced functionality\n */\n public sendPageView = (\n eventName: keyof typeof AnalyticsEventName,\n payload?: SendPageViewPayload\n ) => {\n const enhancedPayload = {\n ...this.createBasePayload(),\n ...payload,\n };\n\n return sendShopifyAnalytics({\n eventName,\n payload: enhancedPayload,\n });\n };\n\n /**\n * Send add to cart event using the enhanced functionality\n */\n public sendAddToCart = ({\n cartId,\n totalValue,\n products,\n }: SendAddToCartPayload) => {\n return this.sendPageView(AnalyticsEventName.ADD_TO_CART, {\n cartId,\n totalValue,\n products,\n });\n };\n\n /**\n * Track an event using Shopify Analytics\n */\n public async trackEvent(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n if (!this.initialized || typeof window === \"undefined\") {\n this.logger.warn(\n \"Shopify Analytics not initialized or not in browser environment\"\n );\n return;\n }\n\n // Get adapter-specific parameters\n const params = this.getAdapterParams(adapterParams);\n const customEventName = this.getEventName(params, event.type);\n\n this.logger.debug(\"Processing Shopify Analytics event\", {\n eventType: event.type,\n customEventName,\n eventData: event,\n adapterParams: params,\n });\n\n // Enhance event with affiliate and experiment data\n const affiliateParams = this.enhanceWithAffiliateParams(event);\n const enhancedEvent = this.enhanceWithExperimentParams(affiliateParams);\n\n try {\n switch (event.type) {\n case EventType.SHOPIFY_PAGE_VIEW:\n case EventType.PAGE_VIEW:\n const pageViewPayload = this.mergeEventData(enhancedEvent, params);\n this.sendPageView(AnalyticsEventName.PAGE_VIEW, pageViewPayload);\n this.logger.debug(\"Sent PAGE_VIEW event to Shopify Analytics\", {\n payload: pageViewPayload,\n });\n break;\n\n case EventType.VIEW_CONTENT:\n const viewContentPayload = this.mergeEventData(enhancedEvent, params);\n this.sendPageView(\n AnalyticsEventName.PRODUCT_VIEW,\n viewContentPayload\n );\n this.logger.debug(\"Sent PRODUCT_VIEW event to Shopify Analytics\", {\n payload: viewContentPayload,\n });\n break;\n\n case EventType.ADD_TO_CART:\n const addToCartPayload = this.mergeEventData(enhancedEvent, params);\n this.trackAddToCart(addToCartPayload);\n break;\n\n case EventType.CHECKOUT_STARTED:\n case EventType.BEGIN_CHECKOUT:\n // For checkout events, send as PAGE_VIEW with checkout page context\n const checkoutPayload = this.mergeEventData(\n { ...enhancedEvent, pageType: \"checkout\" },\n params\n );\n this.sendPageView(AnalyticsEventName.PAGE_VIEW, checkoutPayload);\n this.logger.debug(\n \"Sent checkout PAGE_VIEW event to Shopify Analytics\",\n { payload: checkoutPayload }\n );\n break;\n\n default:\n this.logger.debug(\"Event not supported by Shopify Analytics\", {\n eventType: event.type,\n });\n break;\n }\n } catch (error) {\n this.logger.error(\n \"Error tracking Shopify Analytics event\",\n error instanceof Error ? error : new Error(String(error)),\n { eventType: event.type }\n );\n }\n }\n\n /**\n * Check if Shopify session is properly established\n */\n public isSessionValid(): boolean {\n if (typeof window === \"undefined\") return false;\n\n const shopifyY = document.cookie.includes(\"_shopify_y=\");\n const shopifyS = document.cookie.includes(\"_shopify_s=\");\n\n return shopifyY && shopifyS;\n }\n\n /**\n * Get session debug information\n */\n public getSessionDebugInfo(): object {\n if (typeof window === \"undefined\")\n return { error: \"Not in browser environment\" };\n\n return {\n sessionValid: this.isSessionValid(),\n cookies: {\n shopify_y: document.cookie.includes(\"_shopify_y=\"),\n shopify_s: document.cookie.includes(\"_shopify_s=\"),\n all: document.cookie,\n },\n shopId: this.shopId,\n initialized: this.initialized,\n };\n }\n\n /**\n * Track add to cart event with enhanced error handling\n */\n private trackAddToCart(event: any): void {\n if (!event.productId) {\n this.logger.warn(\"ADD_TO_CART event missing productId\");\n return;\n }\n\n try {\n const productGid = this.formatProductGid(event.productId);\n const cartKey = `merchant_${process.env.NEXT_PUBLIC_MERCHANT_NAME}_cartId`;\n const cartId = localStorage.getItem(cartKey) || \"\";\n\n if (!cartId) {\n this.logger.warn(\"No cart ID found, creating anonymous session\");\n // You might want to generate a temporary cart ID or handle this case\n }\n\n // Validate product data\n if (!event.price || event.price <= 0) {\n this.logger.warn(\"Invalid or missing price for ADD_TO_CART\", {\n price: event.price,\n });\n }\n\n // Create products array for analytics\n const productPrice = Math.max(0, event.price || 0);\n const productQuantity = Math.max(1, event.quantity || 1);\n\n const products: ShopifyAnalyticsProduct[] = [\n {\n productGid: productGid,\n variantGid:\n event.variantId || productGid.replace(\"Product\", \"ProductVariant\"),\n quantity: productQuantity,\n price: productPrice.toString(), // Convert to string as required by ShopifyAnalyticsProduct\n name: event.productName || \"Unknown Product\",\n brand: event.brand || \"Unknown Brand\",\n },\n ];\n\n const totalValue = productPrice * productQuantity;\n\n this.logger.debug(\"ADD_TO_CART event tracked successfully\", {\n productGid,\n quantity: productQuantity,\n price: productPrice,\n totalValue,\n });\n\n // Use the enhanced sendAddToCart method\n this.sendAddToCart({\n cartId,\n products,\n totalValue,\n });\n } catch (error) {\n this.logger.error(\n \"Error in trackAddToCart\",\n error instanceof Error ? error : new Error(String(error))\n );\n // Don't throw - just log the error to prevent breaking the user experience\n }\n }\n}\n","import {\n PixelAdapter,\n GoogleAdapter,\n MoengageAdapter,\n PostHogAdapter,\n} from \"../adapters\";\n\n/**\n * Initialize the event tracking system with the provided adapters\n * @param adapters Array of tracking adapters to register\n */\nexport async function initializeEventTracking(\n adapters: import(\"./subscriber\").TrackingAdapter[]\n): Promise<void> {\n // Import the subscriber\n const { eventSubscriber } = await import(\"./subscriber\");\n\n // Register each adapter using for...of instead of forEach\n for (const adapter of adapters) {\n eventSubscriber.registerAdapter(adapter);\n }\n\n // Initialize the subscriber\n await eventSubscriber.initialize();\n}\n\n/**\n * Example initialization function with common adapters\n * This is a convenience function that shows how to set up common tracking adapters\n * Users can customize this based on their needs\n */\nexport async function initTracking(config: {\n pixelId?: string;\n gaId?: string;\n moengageAppId?: string;\n moengageRegion?: string;\n enableDebugLogs?: boolean;\n}) {\n console.log(\"Initializing event tracking system...\");\n\n const adapters = [];\n\n // Facebook Pixel\n if (config.pixelId) {\n const pixelAdapter = new PixelAdapter({\n pixelId: config.pixelId,\n });\n adapters.push(pixelAdapter);\n }\n\n // Google Analytics\n if (config.gaId) {\n const gaAdapter = new GoogleAdapter({\n measurementId: config.gaId,\n });\n adapters.push(gaAdapter);\n }\n\n // MoEngage\n if (config.moengageAppId) {\n const moengageAdapter = new MoengageAdapter({\n appId: config.moengageAppId,\n enableUserIdentification: true,\n region: config.moengageRegion || \"dc_01\",\n debug: config.enableDebugLogs || false,\n });\n adapters.push(moengageAdapter);\n }\n\n // PostHog\n const posthogAdapter = new PostHogAdapter({\n enableDebugLogs: config.enableDebugLogs || false,\n });\n adapters.push(posthogAdapter);\n\n console.log(\n \"Configured adapters:\",\n adapters.map((a) => a.name)\n );\n\n await initializeEventTracking(adapters);\n console.log(\"Event tracking system initialized successfully\");\n}\n"],"mappings":"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","names":["EventType","moengage","eventSubscriber"]}
|
|
1
|
+
{"version":3,"sources":["../../src/events/publisher.ts","../../src/events/subscriber.ts","../../src/events/index.ts","../../src/types.ts","../../src/logger/index.ts","../../src/experiment/constants.ts","../../src/experiment/experiment-tracker.ts","../../src/adapters/base-adapter.ts","../../src/utils/event-id.ts","../../src/adapters/pixel-adapter.ts","../../src/adapters/google-adapter.ts","../../src/constants.ts","../../src/adapters/moengage-adapter.ts","../../src/adapters/posthog-adapter.ts","../../src/adapters/shopify-adapter.ts","../../src/events/init.ts"],"sourcesContent":["import type { TEvent } from \"../types\";\nimport type { TAdapterParams } from \"../adapter-params\";\n\n/**\n * EventPublisher class responsible for publishing events to subscribers\n */\nclass EventPublisher {\n private static instance: EventPublisher;\n private subscribers: Array<(event: TEvent, params?: TAdapterParams) => void> =\n [];\n\n /**\n * Private constructor to enforce singleton pattern\n */\n private constructor() {}\n\n /**\n * Get the singleton instance of EventPublisher\n */\n public static getInstance(): EventPublisher {\n if (!EventPublisher.instance) {\n EventPublisher.instance = new EventPublisher();\n }\n return EventPublisher.instance;\n }\n\n /**\n * Subscribe to events\n * @param callback Function to be called when an event is published\n * @returns Unsubscribe function\n */\n public subscribe(\n callback: (event: TEvent, params?: TAdapterParams) => void\n ): () => void {\n this.subscribers.push(callback);\n\n // Return unsubscribe function\n return () => {\n this.subscribers = this.subscribers.filter(\n (subscriber) => subscriber !== callback\n );\n };\n }\n\n /**\n * Publish an event to all subscribers\n * @param event The event to publish\n * @param params Optional adapter-specific parameters\n */\n public publish(event: TEvent, params?: TAdapterParams): void {\n try {\n console.log(\"EventPublisher: Received event\", {\n type: event.type,\n data: event,\n params: params,\n });\n\n // Add timestamp if not already present\n const eventWithTimestamp = !event.timestamp\n ? { ...event, timestamp: Date.now() }\n : event;\n\n console.log(\n \"EventPublisher: Number of subscribers:\",\n this.subscribers.length\n );\n\n // Notify all subscribers\n for (const subscriber of this.subscribers) {\n try {\n console.log(\"EventPublisher: Notifying subscriber\");\n subscriber(eventWithTimestamp, params);\n console.log(\"EventPublisher: Subscriber notified successfully\");\n } catch (error) {\n console.error(\"EventPublisher: Error in subscriber:\", error);\n // Continue with other subscribers even if one fails\n }\n }\n\n console.log(\"EventPublisher: Event processing completed\");\n } catch (error) {\n console.error(\"Error publishing event:\", error, event);\n // Silently fail to prevent app crashes due to event publishing issues\n }\n }\n}\n\n// Export singleton instance\nexport const eventPublisher = EventPublisher.getInstance();\n\n/**\n * Helper function to publish events\n * @param event The event to publish\n * @param params Optional adapter-specific parameters\n */\nexport function publishEvent(event: TEvent, params?: TAdapterParams): void {\n try {\n eventPublisher.publish(event, params);\n } catch (error) {\n console.error(\"Error in publishEvent helper:\", error);\n // Silently fail to prevent app crashes due to event publishing issues\n }\n}\n","import type { TEvent } from \"../types\";\nimport type { TAdapterParams } from \"../adapter-params\";\nimport { eventPublisher } from \"./publisher\";\n\n/**\n * Interface for tracking adapters\n */\nexport interface TrackingAdapter {\n /**\n * Name of the tracking adapter\n */\n name: string;\n\n /**\n * Whether the adapter is enabled\n */\n isEnabled: () => boolean;\n\n /**\n * Initialize the adapter\n */\n initialize: () => Promise<void>;\n\n /**\n * Track an event\n */\n trackEvent: (event: TEvent, params?: TAdapterParams) => Promise<void>;\n}\n\n/**\n * EventSubscriber class responsible for managing tracking adapters and forwarding events\n */\nclass EventSubscriber {\n private static instance: EventSubscriber;\n private adapters: TrackingAdapter[] = [];\n private initialized = false;\n private eventQueue: Array<{ event: TEvent; params?: TAdapterParams }> = [];\n private unsubscribe: (() => void) | null = null;\n\n /**\n * Private constructor to enforce singleton pattern\n */\n private constructor() {}\n\n /**\n * Get the singleton instance of EventSubscriber\n */\n public static getInstance(): EventSubscriber {\n if (!EventSubscriber.instance) {\n EventSubscriber.instance = new EventSubscriber();\n }\n return EventSubscriber.instance;\n }\n\n /**\n * Register a tracking adapter\n * @param adapter The tracking adapter to register\n */\n public registerAdapter(adapter: TrackingAdapter): void {\n // Check if adapter with the same name already exists\n const existingAdapter = this.adapters.find((a) => a.name === adapter.name);\n if (existingAdapter) {\n console.warn(\n `Tracking adapter with name \"${adapter.name}\" is already registered. Skipping.`\n );\n return;\n }\n\n this.adapters.push(adapter);\n\n // If already initialized, initialize the new adapter\n if (this.initialized) {\n this.initializeAdapter(adapter).catch((error) => {\n console.error(`Failed to initialize adapter \"${adapter.name}\":`, error);\n });\n }\n }\n\n /**\n * Unregister a tracking adapter by name\n * @param adapterName The name of the adapter to unregister\n */\n public unregisterAdapter(adapterName: string): void {\n this.adapters = this.adapters.filter(\n (adapter) => adapter.name !== adapterName\n );\n }\n\n /**\n * Initialize all registered adapters\n */\n public async initialize(): Promise<void> {\n console.log(\"Initializing EventSubscriber...\");\n\n if (this.initialized) {\n console.log(\"EventSubscriber already initialized\");\n return;\n }\n\n // Subscribe to events\n this.unsubscribe = eventPublisher.subscribe(this.handleEvent.bind(this));\n console.log(\"Subscribed to event publisher\");\n\n // Initialize all adapters\n console.log(`Initializing ${this.adapters.length} adapters...`);\n await Promise.all(\n this.adapters.map((adapter) => this.initializeAdapter(adapter))\n );\n\n this.initialized = true;\n console.log(\"EventSubscriber initialization complete\");\n\n // Process any queued events\n this.processEventQueue();\n }\n\n /**\n * Initialize a single adapter\n * @param adapter The adapter to initialize\n */\n private async initializeAdapter(adapter: TrackingAdapter): Promise<void> {\n try {\n await adapter.initialize();\n console.log(\n `Tracking adapter \"${adapter.name}\" initialized successfully.`\n );\n } catch (error) {\n console.error(`Failed to initialize adapter \"${adapter.name}\":`, error);\n }\n }\n\n /**\n * Handle an event by forwarding it to all enabled adapters\n * @param event The event to handle\n * @param params Optional adapter-specific parameters\n */\n private handleEvent(event: TEvent, params?: TAdapterParams): void {\n console.log(\"Handling event:\", event.type, \"with params:\", params);\n\n if (!this.initialized) {\n console.log(\"Subscriber not initialized, queueing event\");\n this.eventQueue.push({ event, params });\n return;\n }\n\n // Forward the event to all enabled adapters\n for (const adapter of this.adapters) {\n if (adapter.isEnabled()) {\n console.log(`Forwarding event to adapter: ${adapter.name}`);\n adapter.trackEvent(event, params).catch((error) => {\n console.error(\n `Error tracking event in adapter \"${adapter.name}\":`,\n error\n );\n });\n } else {\n console.log(`Adapter ${adapter.name} is disabled, skipping`);\n }\n }\n }\n\n /**\n * Process any events that were queued before initialization\n */\n private processEventQueue(): void {\n if (this.eventQueue.length === 0) {\n return;\n }\n\n console.log(`Processing ${this.eventQueue.length} queued events.`);\n\n // Process each queued event\n for (const { event, params } of this.eventQueue) {\n this.handleEvent(event, params);\n }\n\n // Clear the queue\n this.eventQueue = [];\n }\n\n /**\n * Shutdown the subscriber and all adapters\n */\n public shutdown(): void {\n if (this.unsubscribe) {\n this.unsubscribe();\n this.unsubscribe = null;\n }\n\n this.initialized = false;\n this.eventQueue = [];\n }\n\n /**\n * Get all registered adapters\n */\n public getAdapters(): TrackingAdapter[] {\n return [...this.adapters];\n }\n}\n\n// Export singleton instance\nexport const eventSubscriber = EventSubscriber.getInstance();\n","// Export event types\nexport * from \"../types\";\n\n// Export publisher\nexport { eventPublisher, publishEvent } from \"./publisher\";\n\n// Export subscriber\nexport { eventSubscriber } from \"./subscriber\";\nexport type { TrackingAdapter } from \"./subscriber\";\n\n// Export initialization functions\nexport { initializeEventTracking, initTracking } from \"./init\";\n\n// Export constants\nexport * from \"../constants\";\n","/**\n * Event types that can be tracked across the application\n */\nexport enum EventType {\n PRODUCT_VIEW = \"product_view\",\n PAGE_VIEW = \"page_view\",\n BUTTON_CLICK = \"button_click\",\n FORM_SUBMISSION = \"form_submission\",\n VIEW_ITEM = \"view_item\",\n VIEWED_PRODUCT = \"viewed_product\",\n CHECKOUT_PAYMENT = \"checkout_payment\",\n COLLECTION_VIEW = \"collection_view\",\n ADD_TO_CART = \"add_to_cart\",\n REMOVE_FROM_CART = \"remove_from_cart\",\n CHECKOUT_STARTED = \"checkout_started\",\n CHECKOUT_COMPLETED = \"checkout_completed\",\n SEARCH = \"search\",\n USER_SIGNUP = \"user_signup\",\n USER_LOGIN = \"user_login\",\n CUSTOM = \"custom\",\n USER_ENGAGEMENT = \"user_engagement\",\n SESSION_START = \"session_start\",\n FIRST_VISIT = \"first_visit\",\n SCROLL = \"scroll\",\n FORM_START = \"form_start\",\n ADD_PAYMENT_INFO = \"add_payment_info\",\n CART_VIEWED = \"cart_viewed\",\n VIEW_CONTENT = \"view_content\",\n PURCHASE = \"purchase\",\n VIEW_SEARCH_RESULTS = \"view_search_results\",\n ORDER_PLACED = \"order_placed\",\n INITIATE_CHECKOUT = \"initiate_checkout\",\n\n // GoKwik Checkout events (actually used)\n STARTED_CHECKOUT_GK = \"started_checkout_gk\",\n MOBILE_ADDED_GK = \"mobile_added_gk\",\n ADDRESS_SELECTED_GK = \"address_selected_gk\",\n ADDRESS_COMPLETED_GK = \"address_completed_gk\",\n PAYMENT_METHOD_SELECTED_GK = \"payment_method_selected_gk\",\n PAYMENT_COMPLETED_GK = \"payment_completed_gk\",\n ORDER_SUCCESS = \"order_success\",\n ORDER_COMPLETED = \"order_completed\",\n PIN_CODE_ADDED_GK = \"pin_code_added_gk\",\n ADDRESS_ADDED_GK = \"address_added_gk\",\n\n // Track Specific Event\n SPECIFIC = \"specific\",\n\n // Shopify specific\n SHOPIFY_PAGE_VIEW = \"shopify_page_view\",\n}\n\n/**\n * Base event interface that all events should implement\n */\nexport interface BaseEvent {\n type: EventType;\n timestamp?: number;\n eventId?: string;\n event_category?: string;\n description?: string;\n [key: string]: any;\n}\n\n/**\n * Page view event\n */\nexport interface IPageViewEvent extends BaseEvent {\n type: EventType.PAGE_VIEW;\n path: string;\n title: string;\n referrer?: string;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n campaign?: string;\n campaign_id?: string;\n content?: string;\n debug_mode?: boolean;\n engagement_time_msec?: number;\n entrances?: number;\n ga_session_id?: string;\n ga_session_number?: number;\n gclid?: string;\n ignore_referrer?: boolean;\n medium?: string;\n page_location?: string;\n page_path?: string;\n page_referrer?: string;\n page_title?: string;\n page_source?: string;\n page_term?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n}\n\n/**\n * User engagement event\n */\nexport interface IUserEngagementEvent extends BaseEvent {\n type: EventType.USER_ENGAGEMENT;\n engagementTime?: number; // in seconds\n pagePath?: string;\n pageTitle?: string;\n scrollDepth?: number; // in percentage\n formId?: string;\n formName?: string;\n formFields?: Array<{\n fieldName: string;\n fieldValue: string;\n }>;\n buttonId?: string;\n buttonText?: string;\n location?: string;\n customProperties?: Record<string, any>;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n campaign?: string;\n ga_session_id?: string;\n ga_session_number?: number;\n gclid?: string;\n ignore_referrer?: boolean;\n medium?: string;\n page_location?: string;\n page_path?: string;\n page_referrer?: string;\n page_title?: string;\n source?: string;\n srsltid?: string;\n term?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n}\n\n/**\n * Session start event\n */\nexport interface ISessionStartEvent extends BaseEvent {\n type: EventType.SESSION_START;\n sessionId: string;\n userId?: string;\n userAgent?: string;\n referrer?: string;\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n}\n\n/**\n * First visit event\n */\nexport interface IFirstVisitEvent extends BaseEvent {\n type: EventType.FIRST_VISIT;\n userId?: string;\n userAgent?: string;\n referrer?: string;\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n}\n\n/**\n * Scroll event\n */\nexport interface IScrollEvent extends BaseEvent {\n type: EventType.SCROLL;\n scrollDepth: number; // in percentage\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n engagement_time_msec?: number;\n ga_session_id?: string;\n ga_session_number?: number;\n ignore_referrer?: boolean;\n page_location?: string;\n page_referrer?: string;\n page_title?: string;\n percent_scrolled?: number;\n}\n\n/**\n * Form start event\n */\nexport interface IFormStartEvent extends BaseEvent {\n type: EventType.FORM_START;\n formId: string;\n formName?: string;\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n}\n\n/**\n * Form submission event\n */\nexport interface IFormSubmissionEvent extends BaseEvent {\n type: EventType.FORM_SUBMISSION;\n formId: string;\n formName?: string;\n success: boolean;\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n}\n\n/**\n * User signup event\n */\nexport interface IUserSignupEvent extends BaseEvent {\n type: EventType.USER_SIGNUP;\n userId?: string;\n method?: string;\n success: boolean;\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n}\n\n/**\n * User login event\n */\nexport interface IUserLoginEvent extends BaseEvent {\n type: EventType.USER_LOGIN;\n userId?: string;\n method?: string;\n success: boolean;\n pagePath: string;\n pageTitle: string;\n timestamp: number;\n customProperties?: Record<string, any>;\n}\n\n/**\n * Button click event\n */\nexport interface IButtonClickEvent extends BaseEvent {\n type: EventType.BUTTON_CLICK;\n buttonId?: string;\n buttonText?: string;\n location: string;\n}\n\n/**\n * View item event for e-commerce tracking\n */\nexport interface IViewItemEvent extends BaseEvent {\n type: EventType.VIEW_ITEM;\n currency: string;\n value: number;\n items: Array<{\n item_id: string;\n item_name: string;\n price: number;\n currency: string;\n }>;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n campaign?: string;\n ecomm_pagetype?: string;\n ecomm_prodid?: string;\n ecomm_totalvalue?: number;\n engagement_time_msec?: number;\n ga_session_id?: string;\n ga_session_number?: number;\n gclid?: string;\n ignore_referrer?: boolean;\n medium?: string;\n page_location?: string;\n page_path?: string;\n page_referrer?: string;\n page_title?: string;\n source?: string;\n srsltid?: string;\n term?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n}\n\n/**\n * Checkout payment event\n */\nexport interface ICheckoutPaymentEvent extends BaseEvent {\n type: EventType.CHECKOUT_PAYMENT;\n cartValue: number;\n currency: string;\n itemCount: number;\n items: Array<{\n productId: string;\n productName: string;\n price: number;\n quantity: number;\n }>;\n}\n\n/**\n * Viewed product event after 20 seconds\n */\nexport interface IViewedProductEvent extends BaseEvent {\n type: EventType.VIEWED_PRODUCT;\n productId: string;\n productName: string;\n price: number;\n currency: string;\n viewDuration?: number;\n}\n\nexport interface IPCollectionViewEvent extends BaseEvent {\n type: EventType.COLLECTION_VIEW;\n productId: string;\n productName: string;\n price?: number;\n currency?: string;\n category?: string;\n}\n\n/**\n * Add to cart event\n */\nexport interface IAddToCartEvent extends BaseEvent {\n type: EventType.ADD_TO_CART;\n productId: string;\n productName: string;\n price: number;\n currency?: string;\n quantity?: number;\n variant?: string;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n ecomm_pagetype?: string;\n ecomm_prodid?: string;\n ecomm_totalvalue?: number;\n value?: number;\n engagement_time_msec?: number;\n ga_session_id?: string;\n ga_session_number?: number;\n ignore_referrer?: boolean;\n page_location?: string;\n page_referrer?: string;\n page_title?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n}\n\n/**\n * Remove from cart event\n */\nexport interface IRemoveFromCartEvent extends BaseEvent {\n type: EventType.REMOVE_FROM_CART;\n productId: string;\n productName: string;\n price: number;\n currency?: string;\n quantity: number;\n variant?: string;\n}\n\n/**\n * Checkout started event\n */\nexport interface ICheckoutStartedEvent extends BaseEvent {\n type: EventType.CHECKOUT_STARTED;\n cartValue: number;\n currency?: string;\n itemCount: number;\n items: Array<{\n productId: string;\n productName: string;\n price: number;\n quantity: number;\n variant?: string;\n }>;\n coupon?: string;\n discount?: number | string;\n}\n\n/**\n * Checkout completed event\n */\nexport interface ICheckoutCompletedEvent extends BaseEvent {\n type: EventType.CHECKOUT_COMPLETED;\n orderId: string;\n cartValue: number;\n currency?: string;\n itemCount: number;\n items: Array<{\n productId: string;\n productName: string;\n price: number;\n quantity: number;\n variant?: string;\n }>;\n}\n\n/**\n * Cart viewed event\n */\nexport interface ICartViewedEvent extends BaseEvent {\n type: EventType.CART_VIEWED;\n cartId: string;\n products: Array<{\n variantId: string;\n price: number;\n quantity: number;\n }>;\n}\n\n/**\n * Search event\n */\nexport interface ISearchEvent extends BaseEvent {\n type: EventType.SEARCH;\n searchTerm: string;\n content_ids: string[];\n resultsCount?: number;\n}\n\n/**\n * View content event (for Facebook Pixel ViewContent)\n */\nexport interface IViewContentEvent extends BaseEvent {\n type: EventType.VIEW_CONTENT;\n content_type?: string;\n content_ids?: string[];\n content_name?: string;\n content_category?: string;\n value?: number;\n currency?: string;\n items?: Array<any>;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n campaign?: string;\n ga_session_id?: string;\n engagement_time_msec?: number;\n session_engaged?: boolean;\n page_title?: string;\n page_location?: string;\n page_referrer?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n}\n\n/**\n * Purchase event (for Facebook Pixel Purchase)\n */\nexport interface IPurchaseEvent extends BaseEvent {\n type: EventType.PURCHASE;\n value: number;\n currency?: string;\n content_type?: string;\n content_ids?: string[];\n num_items?: number;\n contents?: Array<{\n id: string;\n quantity: number;\n item_price?: number;\n }>;\n transaction_id?: string;\n coupon?: string;\n shipping?: number;\n tax?: number;\n items?: Array<any>;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n campaign?: string;\n ga_session_id?: string;\n engagement_time_msec?: number;\n session_engaged?: boolean;\n page_title?: string;\n page_location?: string;\n page_referrer?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n}\n\n/**\n * View search results event (for GA4)\n */\nexport interface IViewSearchResultsEvent extends BaseEvent {\n type: EventType.VIEW_SEARCH_RESULTS;\n search_term?: string;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n campaign?: string;\n campaign_id?: string;\n content?: string;\n ga_session_id?: string;\n ga_session_number?: number;\n gclid?: string;\n medium?: string;\n page_location?: string;\n page_path?: string;\n page_referrer?: string;\n page_title?: string;\n source?: string;\n srsltid?: string;\n term?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n engagement_time_msec?: number;\n}\n\n/**\n * Order placed event (for GA4)\n */\nexport interface IOrderPlacedEvent extends BaseEvent {\n type: EventType.ORDER_PLACED;\n transaction_id?: string;\n order_id?: string;\n total_amount?: number;\n value: number;\n currency?: string;\n items?: Array<any>;\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n coupon?: string;\n discount?: number;\n engagement_time_msec?: number;\n ga_session_id?: string;\n ga_session_number?: number;\n ignore_referrer?: boolean;\n page_location?: string;\n page_referrer?: string;\n page_title?: string;\n payment_type?: string;\n shipping?: number;\n shipping_name?: string;\n tax?: number;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n}\n\n/**\n * Begin checkout event (for GA4)\n */\nexport interface IBeginCheckoutEvent extends BaseEvent {\n type: EventType.INITIATE_CHECKOUT;\n orderId?: string;\n cartValue: number;\n currency?: string;\n itemCount: number;\n items: Array<{\n productId: string;\n productName: string;\n price: number;\n quantity: number;\n variant?: string;\n }>;\n coupon?: string;\n discount?: number | string;\n\n // GA4 specific parameters\n batch_ordering_id?: string;\n batch_page_id?: string;\n ecomm_pagetype?: string;\n ecomm_prodid?: string;\n ecomm_totalvalue?: number;\n engagement_time_msec?: number;\n ga_session_id?: string;\n ga_session_number?: number;\n ignore_referrer?: boolean;\n page_location?: string;\n page_referrer?: string;\n page_title?: string;\n user_id?: string;\n user_properties?: Record<string, any>;\n session_engaged?: boolean;\n}\n\n/**\n * Custom event for any other tracking needs\n */\nexport interface ICustomEvent extends BaseEvent {\n type: EventType.CUSTOM;\n}\n\n/**\n * Custom event for any other tracking needs\n */\nexport interface ISpecificEvent extends BaseEvent {\n type: EventType.SPECIFIC;\n}\n\n/**\n * Add Payment Info event\n */\nexport interface IAddPaymentInfoEvent extends BaseEvent {\n type: EventType.ADD_PAYMENT_INFO;\n cartValue: number;\n currency?: string;\n itemCount: number;\n paymentType?: string;\n items: Array<{\n productId: string;\n productName: string;\n price: number;\n quantity: number;\n variant?: string;\n }>;\n}\n\n/**\n * Started checkout GK event\n */\nexport interface IStartedCheckoutGKEvent extends BaseEvent {\n type: EventType.STARTED_CHECKOUT_GK;\n cartValue?: number;\n currency?: string;\n}\n\n/**\n * Mobile added GK event\n */\nexport interface IMobileAddedGKEvent extends BaseEvent {\n type: EventType.MOBILE_ADDED_GK;\n mobile?: string;\n}\n\n/**\n * Address selected GK event\n */\nexport interface IAddressSelectedGKEvent extends BaseEvent {\n type: EventType.ADDRESS_SELECTED_GK;\n addressId?: string;\n}\n\n/**\n * Address completed GK event\n */\nexport interface IAddressCompletedGKEvent extends BaseEvent {\n type: EventType.ADDRESS_COMPLETED_GK;\n addressId?: string;\n}\n\n/**\n * Payment method selected GK event\n */\nexport interface IPaymentMethodSelectedGKEvent extends BaseEvent {\n type: EventType.PAYMENT_METHOD_SELECTED_GK;\n paymentMethod?: string;\n}\n\n/**\n * Payment completed GK event\n */\nexport interface IPaymentCompletedGKEvent extends BaseEvent {\n type: EventType.PAYMENT_COMPLETED_GK;\n amount?: number;\n currency?: string;\n}\n\n/**\n * Order success event\n */\nexport interface IOrderSuccessEvent extends BaseEvent {\n type: EventType.ORDER_SUCCESS;\n orderId?: string;\n amount?: number;\n currency?: string;\n}\n\n/**\n * Order completed event\n */\nexport interface IOrderCompletedEvent extends BaseEvent {\n type: EventType.ORDER_COMPLETED;\n orderId?: string;\n amount?: number;\n currency?: string;\n}\n\n/**\n * PIN code added GK event\n */\nexport interface IPinCodeAddedGKEvent extends BaseEvent {\n type: EventType.PIN_CODE_ADDED_GK;\n pinCode?: string;\n}\n\n/**\n * Address added GK event\n */\nexport interface IAddressAddedGKEvent extends BaseEvent {\n type: EventType.ADDRESS_ADDED_GK;\n addressId?: string;\n}\n\n/**\n * Product view event\n */\nexport interface IProductViewEvent extends BaseEvent {\n type: EventType.PRODUCT_VIEW;\n event_category: string;\n description: string;\n productId: string;\n productName: string;\n price: number;\n currency?: string;\n category?: string;\n}\n\nexport interface IShopifyPageViewEvent extends BaseEvent {\n type: EventType.SHOPIFY_PAGE_VIEW;\n path?: string;\n}\n\n/**\n * Union type of all possible events\n */\nexport type TEvent =\n | IProductViewEvent\n | IPageViewEvent\n | IButtonClickEvent\n | IFormSubmissionEvent\n | IViewItemEvent\n | IPCollectionViewEvent\n | IAddToCartEvent\n | IRemoveFromCartEvent\n | ICheckoutStartedEvent\n | ICheckoutCompletedEvent\n | ICartViewedEvent\n | ISearchEvent\n | IUserSignupEvent\n | IUserLoginEvent\n | IViewContentEvent\n | IPurchaseEvent\n | IViewSearchResultsEvent\n | IOrderPlacedEvent\n | IBeginCheckoutEvent\n | ICustomEvent\n | ISpecificEvent\n | IUserEngagementEvent\n | ISessionStartEvent\n | IFirstVisitEvent\n | IScrollEvent\n | IFormStartEvent\n | IAddPaymentInfoEvent\n | IStartedCheckoutGKEvent\n | IMobileAddedGKEvent\n | IAddressSelectedGKEvent\n | IAddressCompletedGKEvent\n | IPaymentMethodSelectedGKEvent\n | IPaymentCompletedGKEvent\n | IOrderSuccessEvent\n | IOrderCompletedEvent\n | IPinCodeAddedGKEvent\n | IAddressAddedGKEvent\n | IViewedProductEvent\n | ICheckoutPaymentEvent\n | IShopifyPageViewEvent;\n","/**\n * Logger configuration options\n */\nexport interface LoggerConfig {\n level?: \"trace\" | \"debug\" | \"info\" | \"warn\" | \"error\" | \"fatal\";\n enabled?: boolean;\n name?: string;\n prettyPrint?: boolean;\n redact?: string[];\n}\n\n/**\n * Logger interface for consistent logging across adapters\n */\nexport interface Logger {\n trace: (message: string, ...args: any[]) => void;\n debug: (message: string, ...args: any[]) => void;\n info: (message: string, ...args: any[]) => void;\n warn: (message: string, ...args: any[]) => void;\n error: (message: string, error?: Error, ...args: any[]) => void;\n fatal: (message: string, error?: Error, ...args: any[]) => void;\n child: (bindings: Record<string, any>) => Logger;\n}\n\n/**\n * Log levels with numeric values for comparison\n */\nconst LOG_LEVELS = {\n trace: 10,\n debug: 20,\n info: 30,\n warn: 40,\n error: 50,\n fatal: 60,\n} as const;\n\n/**\n * No-op logger for when logging is disabled\n */\nclass NoOpLogger implements Logger {\n trace() {}\n debug() {}\n info() {}\n warn() {}\n error() {}\n fatal() {}\n child(): Logger {\n return new NoOpLogger();\n }\n}\n\n/**\n * Simple universal logger that works in all environments\n */\nclass UniversalLogger implements Logger {\n private config: Required<LoggerConfig>;\n private bindings: Record<string, any>;\n\n constructor(\n config: Required<LoggerConfig>,\n bindings: Record<string, any> = {}\n ) {\n this.config = config;\n this.bindings = bindings;\n }\n\n private shouldLog(level: keyof typeof LOG_LEVELS): boolean {\n return LOG_LEVELS[level] >= LOG_LEVELS[this.config.level];\n }\n\n private formatMessage(\n level: string,\n message: string,\n args: any[],\n error?: Error\n ): string {\n const timestamp = new Date().toISOString();\n const name = this.config.name;\n\n let logMessage = this.config.prettyPrint\n ? `[${timestamp}] ${level.toUpperCase()} [${name}]: ${message}`\n : JSON.stringify({\n timestamp,\n level,\n name,\n message,\n ...this.bindings,\n ...(args.length > 0 && { args }),\n ...(error && {\n error: { message: error.message, stack: error.stack },\n }),\n });\n\n if (this.config.prettyPrint && args.length > 0) {\n logMessage += ` ${args\n .map((arg) =>\n typeof arg === \"object\" ? JSON.stringify(arg) : String(arg)\n )\n .join(\" \")}`;\n }\n\n if (this.config.prettyPrint && error) {\n logMessage += ` Error: ${error.message}`;\n if (error.stack) {\n logMessage += `\\n${error.stack}`;\n }\n }\n\n return this.redactSensitiveData(logMessage);\n }\n\n private redactSensitiveData(message: string): string {\n let redactedMessage = message;\n for (const field of this.config.redact) {\n const regex = new RegExp(`\"${field}\"\\\\s*:\\\\s*\"[^\"]*\"`, \"gi\");\n redactedMessage = redactedMessage.replace(\n regex,\n `\"${field}\":\"[REDACTED]\"`\n );\n }\n return redactedMessage;\n }\n\n private log(\n level: keyof typeof LOG_LEVELS,\n message: string,\n args: any[],\n error?: Error\n ): void {\n if (!this.shouldLog(level)) return;\n\n const formattedMessage = this.formatMessage(level, message, args, error);\n\n // Use appropriate console method based on level\n switch (level) {\n case \"trace\":\n console.trace(formattedMessage);\n break;\n case \"debug\":\n console.debug(formattedMessage);\n break;\n case \"info\":\n console.info(formattedMessage);\n break;\n case \"warn\":\n console.warn(formattedMessage);\n break;\n case \"error\":\n case \"fatal\":\n console.error(formattedMessage);\n break;\n }\n }\n\n trace(message: string, ...args: any[]): void {\n this.log(\"trace\", message, args);\n }\n\n debug(message: string, ...args: any[]): void {\n this.log(\"debug\", message, args);\n }\n\n info(message: string, ...args: any[]): void {\n this.log(\"info\", message, args);\n }\n\n warn(message: string, ...args: any[]): void {\n this.log(\"warn\", message, args);\n }\n\n error(message: string, error?: Error, ...args: any[]): void {\n this.log(\"error\", message, args, error);\n }\n\n fatal(message: string, error?: Error, ...args: any[]): void {\n this.log(\"fatal\", message, args, error);\n }\n\n child(bindings: Record<string, any>): Logger {\n return new UniversalLogger(this.config, { ...this.bindings, ...bindings });\n }\n}\n\n/**\n * Create a logger instance\n */\nexport function createLogger(config: LoggerConfig = {}): Logger {\n const fullConfig: Required<LoggerConfig> = {\n level: config.level || \"info\",\n enabled: config.enabled !== false,\n name: config.name || \"@shopkit/analytics\",\n prettyPrint: config.prettyPrint || false,\n redact: config.redact || [],\n };\n\n // Return no-op logger if disabled\n if (!fullConfig.enabled) {\n return new NoOpLogger();\n }\n\n return new UniversalLogger(fullConfig);\n}\n\n/**\n * Default logger instance\n */\nexport const logger = createLogger({\n enabled: true,\n level: \"info\",\n prettyPrint: false,\n});\n\n/**\n * Create adapter-specific logger\n */\nexport function createAdapterLogger(\n adapterName: string,\n config: LoggerConfig = {}\n): Logger {\n const baseLogger = createLogger(config);\n return baseLogger.child({ adapter: adapterName });\n}\n","/**\n * Prima PR A/B test experiment cookie names\n */\nexport const PRIMA_EXPERIMENT_COOKIES = {\n HOME: \"_prima_ptr_ab_home\",\n COLLECTION: \"_prima_ptr_ab_collection\",\n PRODUCT: \"_prima_ptr_ab_product\",\n} as const;\n\n/**\n * Array of all Prima experiment cookie names for iteration\n */\nexport const PRIMA_EXPERIMENT_COOKIE_NAMES = Object.values(\n PRIMA_EXPERIMENT_COOKIES\n);\n","import type { ExperimentData } from \"./types\";\nimport {\n PRIMA_EXPERIMENT_COOKIES,\n PRIMA_EXPERIMENT_COOKIE_NAMES,\n} from \"./constants\";\n\n/**\n * Get cookie value by name\n */\nfunction getCookie(name: string): string | null {\n if (typeof document === \"undefined\") return null;\n\n const value = `; ${document.cookie}`;\n const parts = value.split(`; ${name}=`);\n if (parts.length === 2) {\n return parts.pop()?.split(\";\").shift() || null;\n }\n return null;\n}\n\n/**\n * Get experiment parameters from cookies\n */\nexport function getExperimentParams(): ExperimentData {\n const payload = {\n [PRIMA_EXPERIMENT_COOKIES.HOME]: getCookie(PRIMA_EXPERIMENT_COOKIES.HOME),\n [PRIMA_EXPERIMENT_COOKIES.COLLECTION]: getCookie(\n PRIMA_EXPERIMENT_COOKIES.COLLECTION\n ),\n [PRIMA_EXPERIMENT_COOKIES.PRODUCT]: getCookie(\n PRIMA_EXPERIMENT_COOKIES.PRODUCT\n ),\n };\n\n return payload;\n}\n\n/**\n * Check if any experiment data exists\n */\nexport function hasExperimentData(): boolean {\n return PRIMA_EXPERIMENT_COOKIE_NAMES.some(\n (cookieName) => getCookie(cookieName) !== null\n );\n}\n","import type { TEvent } from \"../types\";\nimport type { TrackingAdapter } from \"../events/subscriber\";\nimport type { TAdapterParams, IBaseAdapterParams } from \"../adapter-params\";\nimport { createAdapterLogger, type Logger, type LoggerConfig } from \"../logger\";\nimport { getExperimentParams } from \"../experiment/experiment-tracker\";\nimport type { ExperimentData } from \"../experiment/types\";\nimport { PRIMA_EXPERIMENT_COOKIES } from \"../experiment\";\n\n/**\n * Abstract base class for tracking adapters\n */\nexport abstract class BaseAdapter implements TrackingAdapter {\n /**\n * Name of the tracking adapter\n */\n public abstract readonly name: string;\n\n /**\n * Configuration for the adapter\n */\n protected config: Record<string, any>;\n\n /**\n * Logger instance for this adapter\n */\n protected logger: Logger;\n\n /**\n * Whether the adapter has been initialized\n */\n protected initialized = false;\n\n /**\n * Constructor\n * @param config Configuration for the adapter\n */\n constructor(config: Record<string, any> = {}) {\n this.config = config;\n\n // Create a generic logger - will be properly initialized by subclasses\n const loggerConfig: LoggerConfig = {\n enabled: config.enableDebugLogs ?? true,\n level: config.logLevel ?? \"info\",\n name: \"@shopkit/analytics\",\n };\n\n this.logger = createAdapterLogger(\"BaseAdapter\", loggerConfig);\n }\n\n /**\n * Initialize logger with proper adapter name\n * Should be called by subclasses after name is available\n */\n protected initializeLogger(): void {\n const loggerConfig: LoggerConfig = {\n enabled: this.config.enableDebugLogs ?? true,\n level: this.config.logLevel ?? \"info\",\n name: \"@shopkit/analytics\",\n };\n\n this.logger = createAdapterLogger(this.name, loggerConfig);\n }\n\n /**\n * Check if the adapter is enabled\n */\n public isEnabled(): boolean {\n return this.initialized;\n }\n\n /**\n * Initialize the adapter\n */\n public abstract initialize(): Promise<void>;\n\n /**\n * Track an event\n * @param event The event to track\n * @param params Optional adapter-specific parameters\n */\n public abstract trackEvent(\n event: TEvent,\n params?: TAdapterParams\n ): Promise<void>;\n\n /**\n * Extract adapter-specific parameters for this adapter\n * @param params The adapter parameters object\n * @returns Adapter-specific parameters or empty object\n */\n protected getAdapterParams(params?: TAdapterParams): IBaseAdapterParams {\n return params?.[this.name] || {};\n }\n\n /**\n * Get custom event name or fall back to default\n * @param params The adapter parameters object\n * @param defaultName The default event name to use\n * @returns Custom event name or default\n */\n protected getEventName(\n params: TAdapterParams | undefined,\n defaultName: string\n ): string {\n const adapterParams = this.getAdapterParams(params);\n return adapterParams.event_name || defaultName;\n }\n\n /**\n * Merge standard event data with adapter-specific parameters\n * @param params The adapter parameters object\n * @param standardData The standard event data\n * @returns Merged data with adapter-specific overrides\n */\n protected mergeEventData(\n standardData: Record<string, any>,\n params: TAdapterParams | undefined\n ): Record<string, any> {\n const adapterParams = this.getAdapterParams(params);\n\n // Remove event_name from adapter params to avoid conflicts\n const { event_name, ...customParams } = adapterParams;\n\n // Merge with priority: adapter-specific > standard\n return {\n ...standardData,\n ...customParams,\n };\n }\n\n /**\n * Get a configuration value\n * @param key The configuration key\n * @param defaultValue The default value if the key is not found\n */\n protected getConfig<T>(key: string, defaultValue?: T): T {\n return key in this.config ? this.config[key] : (defaultValue as T);\n }\n\n /**\n * Get affiliate parameters from session storage or custom implementation\n * Override this method to provide your own affiliate tracking implementation\n * @returns Affiliate parameters or null if not available\n */\n protected getAffiliateParams(): Record<string, string> | null {\n // Default implementation returns null\n // Users can override this method to integrate with their affiliate tracking system\n return null;\n }\n\n /**\n * Get experiment parameters from cookies\n * @returns Experiment parameters\n */\n protected getExperimentParams(): ExperimentData {\n try {\n return getExperimentParams();\n } catch (error) {\n // Silently fail if experiment tracking is not available\n return {\n [PRIMA_EXPERIMENT_COOKIES.HOME]: null,\n [PRIMA_EXPERIMENT_COOKIES.COLLECTION]: null,\n [PRIMA_EXPERIMENT_COOKIES.PRODUCT]: null,\n };\n }\n }\n\n /**\n * Enhance event parameters with affiliate data\n * @param params The original parameters\n * @returns Parameters enhanced with affiliate data\n */\n protected enhanceWithAffiliateParams(\n params: Record<string, any>\n ): Record<string, any> {\n const affiliateParams = this.getAffiliateParams();\n if (!affiliateParams) return params;\n\n return {\n ...params,\n // Add affiliate parameters\n ...affiliateParams,\n };\n }\n\n /**\n * Enhance event parameters with experiment data\n * @param params The original parameters\n * @returns Parameters enhanced with experiment data\n */\n protected enhanceWithExperimentParams(\n params: Record<string, any>\n ): Record<string, any> {\n const experimentParams = this.getExperimentParams();\n\n return {\n ...params,\n experiment: experimentParams,\n };\n }\n}\n","/**\n * Generate a unique event ID for deduplication\n * Format: timestamp_randomString_eventType\n */\nexport function generateEventId(eventType: string): string {\n const timestamp = Date.now();\n const randomString = Math.random().toString(36).substring(2, 8);\n return `${timestamp}_${randomString}_${eventType}`;\n}\n\n/**\n * Extract browser information for CAPI user data\n */\nexport function getBrowserInfo(): {\n clientUserAgent?: string;\n fbc?: string;\n fbp?: string;\n} {\n if (typeof window === \"undefined\") {\n return {};\n }\n\n const userAgent = navigator.userAgent;\n\n // Get Facebook click ID from URL or cookies\n const fbc = getFacebookClickId();\n const fbp = getFacebookBrowserId();\n\n return {\n clientUserAgent: userAgent,\n fbc,\n fbp,\n };\n}\n\n/**\n * Get Facebook click ID (fbc) from URL parameters or cookies\n */\nfunction getFacebookClickId(): string | undefined {\n if (typeof window === \"undefined\") {\n return undefined;\n }\n\n // Check URL parameters first\n const urlParams = new URLSearchParams(window.location.search);\n const fbclid = urlParams.get(\"fbclid\");\n\n if (fbclid) {\n return `fb.1.${Date.now()}.${fbclid}`;\n }\n\n // Check cookies\n const cookies = document.cookie.split(\";\");\n for (const cookie of cookies) {\n const [name, value] = cookie.trim().split(\"=\");\n if (name === \"_fbc\") {\n return value;\n }\n }\n\n return undefined;\n}\n\n/**\n * Get Facebook browser ID (fbp) from cookies\n */\nfunction getFacebookBrowserId(): string | undefined {\n if (typeof window === \"undefined\") {\n return undefined;\n }\n\n const cookies = document.cookie.split(\";\");\n for (const cookie of cookies) {\n const [name, value] = cookie.trim().split(\"=\");\n if (name === \"_fbp\") {\n return value;\n }\n }\n\n return undefined;\n}\n\n/**\n * Get client IP address (this will be handled server-side)\n */\nexport function getClientIpAddress(request?: Request): string | undefined {\n if (!request) {\n return undefined;\n }\n\n // Try different headers for IP address\n const forwarded = request.headers.get(\"x-forwarded-for\");\n const realIp = request.headers.get(\"x-real-ip\");\n const cfConnectingIp = request.headers.get(\"cf-connecting-ip\");\n\n return forwarded?.split(\",\")[0] || realIp || cfConnectingIp || undefined;\n}\n","import { BaseAdapter } from \"./base-adapter\";\nimport {\n type TEvent,\n EventType,\n type IPageViewEvent,\n type IAddToCartEvent,\n type IProductViewEvent,\n type IViewContentEvent,\n type ISearchEvent,\n type IBeginCheckoutEvent,\n type IPurchaseEvent,\n} from \"../types\";\nimport type { TAdapterParams } from \"../adapter-params\";\nimport { generateEventId, getBrowserInfo } from \"../utils/event-id\";\n\ndeclare global {\n interface Window {\n fbq?: (\n type: string,\n eventName: string,\n params?: Record<string, any>,\n options?: { eventID?: string }\n ) => void;\n }\n}\n\n/**\n * Configuration for the Facebook Pixel adapter\n */\nexport interface PixelAdapterConfig {\n /**\n * Facebook Pixel ID\n */\n pixelId: string;\n /**\n * Enable server-side CAPI backup\n * @default true - CAPI is enabled by default for better data reliability\n */\n enableCAPI?: boolean;\n}\n\n/**\n * Facebook Pixel tracking adapter\n * Assumes that the Facebook Pixel script is already loaded in the page\n */\nexport class PixelAdapter extends BaseAdapter {\n public readonly name = \"FacebookPixel\";\n\n constructor(config: PixelAdapterConfig) {\n super(config);\n // Initialize logger with proper adapter name\n this.initializeLogger();\n }\n\n /**\n * Initialize the Facebook Pixel\n */\n public async initialize(): Promise<void> {\n if (this.initialized) {\n this.logger.debug(\"Already initialized\");\n return;\n }\n\n // Skip initialization if running on server\n if (typeof window === \"undefined\") {\n this.logger.debug(\"Skipping initialization on server\");\n return;\n }\n\n // Check if fbq is available\n if (!window.fbq) {\n this.logger.warn(\n \"Facebook Pixel fbq not found. Make sure the script is loaded.\"\n );\n return;\n } else {\n const pixelId = this.getConfig(\"pixelId\") as string;\n if (!pixelId) {\n this.logger.warn(\"Pixel ID not provided. Skipping fbq init.\");\n } else {\n // Check if pixel is already initialized to avoid duplicate init\n const isAlreadyInitialized =\n (window as any)._fbq?.loaded ||\n (window as any).fbq?.loaded ||\n document\n .querySelector(`script[src*=\"fbevents.js\"]`)\n ?.getAttribute(\"data-pixel-id\") === pixelId;\n\n if (!isAlreadyInitialized) {\n window.fbq(\"init\", pixelId);\n this.logger.info(\"Successfully initialized\", { pixelId: \"***\" });\n } else {\n this.logger.info(\"Pixel already initialized, skipping init\", {\n pixelId: \"***\",\n });\n }\n }\n }\n\n this.initialized = true;\n }\n\n /**\n * Track an event with Facebook Pixel\n * @param event The event to track\n * @param adapterParams Optional adapter-specific parameters\n */\n public async trackEvent(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n // Generate unique event ID for deduplication if not already present\n if (!event.eventId) {\n event.eventId = generateEventId(event.type);\n }\n\n // Ensure timestamp is always available for both client and server tracking\n if (!event.timestamp) {\n event.timestamp = Date.now();\n }\n\n this.logger.debug(\"Tracking event\", {\n eventType: event.type,\n eventId: event.eventId,\n timestamp: event.timestamp,\n });\n\n // Track client-side (browser pixel)\n await this.trackClientSide(event, adapterParams);\n\n // Track server-side (CAPI) as backup\n if (this.getConfig(\"enableCAPI\", true)) {\n await this.trackServerSide(event, adapterParams);\n }\n }\n\n /**\n * Track event on client-side (browser pixel)\n */\n private async trackClientSide(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n if (!this.initialized || typeof window === \"undefined\" || !window.fbq) {\n this.logger.warn(\n \"Cannot track client-side event, adapter not initialized or fbq not available\",\n {\n eventType: event.type,\n initialized: this.initialized,\n hasWindow: typeof window !== \"undefined\",\n hasFbq: !!window?.fbq,\n }\n );\n return;\n }\n\n const { eventName, enhancedParams } = this.formatEventPayload(\n event,\n adapterParams\n );\n\n // Skip tracking if eventName is empty (unknown event type)\n if (!eventName) {\n return;\n }\n\n // Add event_id for deduplication\n if (event.eventId) {\n window.fbq?.(\"track\", eventName, enhancedParams, {\n eventID: event.eventId,\n });\n } else {\n window.fbq?.(\"track\", eventName, enhancedParams);\n }\n\n this.logger.debug(`${eventName} event tracked on client-side`, {\n eventName,\n eventId: event.eventId,\n params: enhancedParams,\n });\n }\n\n /**\n * Track event on server-side (CAPI) as backup\n */\n private async trackServerSide(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n try {\n const { eventName, enhancedParams } = this.formatEventPayload(\n event,\n adapterParams\n );\n\n // Skip tracking if eventName is empty (unknown event type)\n if (!eventName) {\n return;\n }\n\n const browserInfo = getBrowserInfo();\n\n const payload = {\n eventName,\n eventId: event.eventId,\n timestamp: event.timestamp,\n enhancedParams: enhancedParams,\n userInfo: browserInfo,\n };\n\n const response = await fetch(\"/api/events\", {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify(payload),\n });\n\n if (!response.ok) {\n throw new Error(`Server-side tracking failed: ${response.status}`);\n }\n\n this.logger.debug(`${eventName} event sent to server-side`, {\n eventId: event.eventId,\n params: enhancedParams,\n });\n } catch (error) {\n this.logger.error(\n `FB Pixel: Server-side tracking failed for event ${event.type}: ${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n }\n\n /**\n * Utility function to format event payload for both client and server-side tracking\n */\n private formatEventPayload(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): { eventName: string; enhancedParams: Record<string, any> } {\n let baseParams: Record<string, any> = {};\n let eventName = \"\";\n\n switch (event.type) {\n case EventType.PAGE_VIEW:\n const pageViewEvent = event as IPageViewEvent;\n eventName = this.getEventName(adapterParams, \"PageView\");\n baseParams = {\n // Facebook-compatible parameters\n content_name: pageViewEvent.page_title || pageViewEvent.title,\n content_category: pageViewEvent.event_category,\n // Custom parameters that Facebook will accept in custom_data\n page_location: pageViewEvent.page_location || pageViewEvent.path,\n page_title: pageViewEvent.page_title || pageViewEvent.title,\n page_referrer: pageViewEvent.page_referrer || pageViewEvent.referrer,\n page_path: pageViewEvent.page_path || pageViewEvent.path,\n };\n break;\n\n case EventType.VIEW_CONTENT:\n const viewContentEvent = event as IViewContentEvent;\n eventName = this.getEventName(adapterParams, \"ViewContent\");\n baseParams = {\n content_type: viewContentEvent.content_type || \"product\",\n content_ids: viewContentEvent.content_ids || [],\n content_name: viewContentEvent.content_name || \"\",\n content_category: viewContentEvent.content_category || \"\",\n value: viewContentEvent.value || 0,\n currency: viewContentEvent.currency || \"INR\",\n };\n break;\n\n case EventType.PRODUCT_VIEW:\n const productEvent = event as IProductViewEvent;\n eventName = this.getEventName(adapterParams, \"ViewContent\");\n baseParams = {\n content_type: \"product_group\",\n content_ids: [productEvent.productId],\n content_name: productEvent.productName,\n content_category: productEvent.category,\n value: productEvent.price,\n currency: productEvent.currency || \"INR\",\n };\n break;\n\n case EventType.ADD_TO_CART:\n const cartEvent = event as IAddToCartEvent;\n eventName = this.getEventName(adapterParams, \"AddToCart\");\n baseParams = {\n content_type: \"product_group\",\n content_ids: [cartEvent.productId],\n content_name: cartEvent.productName,\n value: cartEvent.price,\n currency: cartEvent.currency || \"INR\",\n };\n break;\n\n case EventType.SEARCH:\n const searchEvent = event as ISearchEvent;\n eventName = this.getEventName(adapterParams, \"Search\");\n baseParams = {\n search_string: searchEvent.searchTerm,\n content_category: \"product\",\n content_ids: searchEvent.content_ids || [],\n };\n break;\n\n case EventType.INITIATE_CHECKOUT:\n const checkoutEvent = event as IBeginCheckoutEvent;\n eventName = this.getEventName(adapterParams, \"InitiateCheckout\");\n baseParams = {\n content_type: \"product\",\n currency: checkoutEvent.currency || \"INR\",\n value: checkoutEvent.cartValue,\n num_items:\n checkoutEvent.itemCount || checkoutEvent.items?.length || 0,\n contents:\n checkoutEvent.items?.map((item) => ({\n id: item.productId,\n quantity: item.quantity || 1,\n item_price: item.price,\n })) || [],\n };\n break;\n\n case EventType.PURCHASE:\n const purchaseEvent = event as IPurchaseEvent;\n eventName = this.getEventName(adapterParams, \"Purchase\");\n baseParams = {\n content_type: purchaseEvent.content_type || \"product\",\n content_ids: purchaseEvent.content_ids || [],\n currency: purchaseEvent.currency || \"INR\",\n value: purchaseEvent.value,\n num_items:\n purchaseEvent.num_items || purchaseEvent.contents?.length || 0,\n contents: purchaseEvent.contents || [],\n };\n break;\n\n default:\n // Skip tracking for unknown event types\n this.logger.debug(`Skipping unknown event type: ${event.type}`);\n return { eventName: \"\", enhancedParams: {} };\n }\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n return { eventName, enhancedParams };\n }\n}\n","import { BaseAdapter } from \"./base-adapter\";\nimport {\n type TEvent,\n EventType,\n type IPageViewEvent,\n type IFormSubmissionEvent,\n type IViewItemEvent,\n type IViewedProductEvent,\n type IAddToCartEvent,\n type ICheckoutStartedEvent,\n type ICheckoutCompletedEvent,\n type ISearchEvent,\n type ICustomEvent,\n type IAddPaymentInfoEvent,\n type ICheckoutPaymentEvent,\n type IViewSearchResultsEvent,\n type IOrderPlacedEvent,\n type IBeginCheckoutEvent,\n type IViewContentEvent,\n type IPurchaseEvent,\n type ISpecificEvent,\n} from \"../types\";\nimport type { TAdapterParams } from \"../adapter-params\";\n\ndeclare global {\n interface Window {\n gtag?: (...args: any[]) => void;\n }\n}\n\n/**\n * Configuration for the Google Analytics adapter\n */\nexport interface GoogleAdapterConfig {\n /**\n * Google Analytics measurement ID (e.g., G-XXXXXXXXXX)\n */\n measurementId: string;\n}\n\n/**\n * Google Analytics tracking adapter\n * Assumes that the Google Analytics script is already loaded in the page\n */\nexport class GoogleAdapter extends BaseAdapter {\n public readonly name = \"GoogleAnalytics\";\n\n constructor(config: GoogleAdapterConfig) {\n super(config);\n // Initialize logger with proper adapter name\n this.initializeLogger();\n }\n\n /**\n * Initialize Google Analytics adapter\n * This just checks if gtag is available\n */\n public isEnabled(): boolean {\n const enabled =\n this.initialized && typeof window !== \"undefined\" && !!window.gtag;\n\n this.logger.debug(\"Checking if enabled\", {\n initialized: this.initialized,\n hasWindow: typeof window !== \"undefined\",\n hasGtag: !!window?.gtag,\n enabled,\n });\n\n return enabled;\n }\n\n public async initialize(): Promise<void> {\n this.logger.info(\"Starting initialization\");\n\n if (this.initialized) {\n this.logger.debug(\"Already initialized\");\n return;\n }\n\n // Skip initialization if running on server\n if (typeof window === \"undefined\") {\n this.logger.debug(\"Skipping initialization on server\");\n return;\n }\n\n const measurementId = this.getConfig<string>(\"measurementId\");\n this.logger.info(\"Using measurement ID\", { measurementId });\n\n // Check if gtag is available\n if (!window.gtag) {\n this.logger.warn(\n \"Google Analytics gtag not found. Make sure the script is loaded.\"\n );\n return;\n }\n\n this.initialized = true;\n this.logger.info(\"Successfully initialized\", { measurementId });\n }\n\n /**\n * Track an event with Google Analytics\n * @param event The event to track\n * @param adapterParams Optional adapter-specific parameters\n */\n public async trackEvent(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n this.logger.debug(\"Attempting to track event\", {\n initialized: this.initialized,\n hasWindow: typeof window !== \"undefined\",\n hasGtag: !!window?.gtag,\n eventType: event.type,\n });\n\n if (!this.initialized || typeof window === \"undefined\" || !window.gtag) {\n this.logger.warn(\n \"Cannot track event, adapter not initialized or gtag not available\",\n {\n eventType: event.type,\n initialized: this.initialized,\n hasWindow: typeof window !== \"undefined\",\n hasGtag: !!window?.gtag,\n }\n );\n return;\n }\n\n const measurementId = this.getConfig<string>(\"measurementId\");\n this.logger.info(\"Tracking event\", {\n eventType: event.type,\n measurementId,\n });\n\n switch (event.type) {\n case EventType.PAGE_VIEW:\n this.trackPageView(\n event as IPageViewEvent,\n measurementId,\n adapterParams\n );\n break;\n case EventType.USER_ENGAGEMENT:\n this.trackUserEngagement(adapterParams);\n break;\n case EventType.SESSION_START:\n this.trackSessionStart(adapterParams);\n break;\n case EventType.FIRST_VISIT:\n this.trackFirstVisit(adapterParams);\n break;\n case EventType.VIEW_ITEM:\n this.trackViewItem(event as IViewItemEvent, adapterParams);\n break;\n case EventType.VIEWED_PRODUCT:\n this.trackViewedProduct(event as IViewedProductEvent, adapterParams);\n break;\n case EventType.CHECKOUT_PAYMENT:\n this.trackCheckoutPayment(\n event as ICheckoutPaymentEvent,\n adapterParams\n );\n break;\n case EventType.SCROLL:\n this.trackScroll(adapterParams);\n break;\n case EventType.ADD_TO_CART:\n this.trackAddToCart(event as IAddToCartEvent, adapterParams);\n break;\n case EventType.FORM_START:\n this.trackFormStart(adapterParams);\n break;\n case EventType.FORM_SUBMISSION:\n this.trackFormSubmission(event as IFormSubmissionEvent, adapterParams);\n break;\n case EventType.SEARCH:\n this.trackSearch(event as ISearchEvent, adapterParams);\n break;\n case EventType.VIEW_CONTENT:\n this.trackViewContent(event as IViewContentEvent, adapterParams);\n break;\n case EventType.VIEW_SEARCH_RESULTS:\n this.trackViewSearchResults(\n event as IViewSearchResultsEvent,\n adapterParams\n );\n break;\n case EventType.CHECKOUT_STARTED:\n this.trackCheckoutStarted(\n event as ICheckoutStartedEvent,\n adapterParams\n );\n break;\n case EventType.INITIATE_CHECKOUT:\n this.trackBeginCheckout(event as IBeginCheckoutEvent, adapterParams);\n break;\n case EventType.CHECKOUT_COMPLETED:\n this.trackCheckoutCompleted(\n event as ICheckoutCompletedEvent,\n adapterParams\n );\n break;\n case EventType.PURCHASE:\n this.trackPurchase(event as IPurchaseEvent, adapterParams);\n break;\n case EventType.ADD_PAYMENT_INFO:\n this.trackAddPaymentInfo(event as IAddPaymentInfoEvent, adapterParams);\n break;\n case EventType.ORDER_PLACED:\n this.trackOrderPlaced(event as IOrderPlacedEvent, adapterParams);\n break;\n case EventType.CUSTOM:\n this.trackCustomEvent(event as ICustomEvent, adapterParams);\n break;\n case EventType.SPECIFIC:\n this.trackSpecificEvent(event as ISpecificEvent, adapterParams);\n break;\n default: {\n this.logger.debug(\"Unhandled event type\", { eventType: event.type });\n break;\n }\n }\n }\n\n /**\n * Track a page view event\n * @param event The page view event\n * @param measurementId The Google Analytics measurement ID\n */\n private trackPageView(\n event: IPageViewEvent,\n measurementId: string,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"page_view\");\n const baseParams = {\n page_path: event.path,\n page_title: event.title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer || undefined,\n ga_session_id: event.ga_session_id || \"\",\n ga_session_number: event.ga_session_number || \"\",\n gclid: event.gclid || \"\",\n engagement_time_msec: event.engagement_time_msec,\n campaign: event.campaign || \"\",\n campaign_id: event.campaign_id || \"\",\n content: event.content || \"\",\n debug_mode: event.debug_mode,\n medium: event.medium || \"\",\n source: event.page_source || \"\",\n page_term: event.page_term || \"\",\n ignore_referrer: event.ignore_referrer,\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n user_id: event.user_id,\n user_properties: event.user_properties,\n session_engaged: event.session_engaged,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackUserEngagement(adapterParams?: TAdapterParams): void {\n const eventName = this.getEventName(adapterParams, \"user_engagement\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n campaign: \"\",\n ga_session_id: \"\",\n ga_session_number: \"\",\n gclid: \"\",\n ignore_referrer: \"\",\n medium: \"\",\n page_location: window.location.href,\n page_path: window.location.pathname,\n page_referrer: document.referrer,\n page_title: document.title,\n source: \"\",\n srsltid: \"\",\n term: \"\",\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackSessionStart(adapterParams?: TAdapterParams): void {\n const eventName = this.getEventName(adapterParams, \"session_start\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n campaign: \"\",\n campaign_id: \"\",\n content: \"\",\n ga_session_id: \"\",\n ga_session_number: \"\",\n gclid: \"\",\n ignore_referrer: \"\",\n medium: \"\",\n page_location: window.location.href,\n page_path: window.location.pathname,\n page_referrer: document.referrer,\n page_title: document.title,\n source: \"\",\n srsltid: \"\",\n term: \"\",\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackFirstVisit(adapterParams?: TAdapterParams): void {\n const eventName = this.getEventName(adapterParams, \"first_visit\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n campaign: \"\",\n campaign_id: \"\",\n content: \"\",\n debug_mode: \"\",\n engagement_time_msec: 0,\n entrances: 1,\n ga_session_id: \"\",\n ga_session_number: \"\",\n gclid: \"\",\n ignore_referrer: \"\",\n medium: \"\",\n page_location: window.location.href,\n page_path: window.location.pathname,\n page_referrer: document.referrer,\n page_title: document.title,\n page_source: \"\",\n page_term: \"\",\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackViewItem(\n event: IViewItemEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"view_item\");\n const baseParams = {\n currency: event.currency,\n value: event.value,\n items: event.items,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackScroll(adapterParams?: TAdapterParams): void {\n const eventName = this.getEventName(adapterParams, \"scroll\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n engagement_time_msec: undefined,\n ga_session_id: \"\",\n ga_session_number: \"\",\n ignore_referrer: undefined,\n page_location: window.location.href,\n page_path: window.location.pathname,\n page_referrer: document.referrer,\n page_title: document.title,\n percent_scrolled: 90, // or dynamically captured\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackCheckoutPayment(\n event: ICheckoutPaymentEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"checkout_payment\");\n const baseParams = {\n currency: event.currency,\n value: event.cartValue,\n items: event.items.map((item) => ({\n item_id: item.productId,\n item_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackViewedProduct(\n event: IViewedProductEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"viewed_product\");\n const baseParams = {\n currency: event.currency,\n value: event.price,\n items: [\n {\n item_id: event.productId,\n item_name: event.productName,\n price: event.price,\n currency: event.currency,\n },\n ],\n view_duration: event.viewDuration,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackAddToCart(\n event: IAddToCartEvent,\n adapterParams?: TAdapterParams\n ): void {\n this.logger.debug(\"Processing add_to_cart event\", { event });\n\n const eventName = this.getEventName(adapterParams, \"add_to_cart\");\n const baseParams = {\n currency: event.currency || \"INR\",\n value: event.price * (event.quantity || 1), // Calculate total value\n items: [\n {\n item_id: event.productId,\n item_name: event.productName,\n price: event.price,\n quantity: event.quantity || 1,\n item_variant: event.variant,\n },\n ],\n // GA4-specific parameters\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n campaign: event.campaign || \"\",\n ga_session_id: event.ga_session_id || \"\",\n engagement_time_msec: event.engagement_time_msec,\n session_engaged: event.session_engaged,\n page_title: event.page_title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer,\n user_id: event.user_id,\n user_properties: event.user_properties,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n this.logger.debug(\"Sending add_to_cart event to gtag\", {\n eventName,\n params: enhancedParams,\n });\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackFormStart(adapterParams?: TAdapterParams): void {\n const eventName = this.getEventName(adapterParams, \"form_start\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n engagement_time_msec: undefined,\n first_field_id: \"\",\n first_field_name: \"\",\n first_field_position: \"\",\n first_field_type: \"\",\n form_destination: \"\",\n form_length: \"\",\n form_id: \"\",\n ga_session_id: \"\",\n ga_session_number: \"\",\n ignore_referrer: undefined,\n page_location: window.location.href,\n page_referrer: document.referrer,\n page_title: document.title,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackCheckoutStarted(\n event: ICheckoutStartedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"begin_checkout\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n currency: event.currency || \"INR\",\n ecomm_pagetype: \"\",\n ecomm_prodid: \"\",\n ecomm_totalvalue: event.cartValue,\n value: event.cartValue,\n engagement_time_msec: undefined,\n ga_session_id: \"\",\n ga_session_number: \"\",\n ignore_referrer: undefined,\n page_location: window.location.href,\n page_referrer: document.referrer,\n page_title: document.title,\n items: event.items.map((item) => ({\n item_id: item.productId,\n item_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n item_variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackFormSubmission(\n event: IFormSubmissionEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"form_submit\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n engagement_time_msec: undefined,\n form_destination: \"\",\n form_length: \"\",\n form_id: event.formId,\n ga_session_id: \"\",\n ga_session_number: \"\",\n ignore_referrer: undefined,\n page_location: window.location.href,\n page_referrer: document.referrer,\n page_title: document.title,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n private trackSearch(\n event: ISearchEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"search\");\n const baseParams = {\n batch_ordering_id: \"\",\n batch_page_id: \"\",\n engagement_time_msec: undefined,\n ga_session_id: \"\",\n ga_session_number: \"\",\n ignore_referrer: undefined,\n page_location: window.location.href,\n page_referrer: document.referrer,\n page_title: document.title,\n search_term: event.searchTerm,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a checkout completed event\n * @param event The checkout completed event\n */\n private trackCheckoutCompleted(\n event: ICheckoutCompletedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"purchase\");\n const baseParams = {\n transaction_id: event.orderId,\n currency: event.currency || \"INR\",\n value: event.cartValue,\n items: event.items.map((item) => ({\n item_id: item.productId,\n item_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n item_variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a custom event\n * @param event The custom event\n */\n private trackCustomEvent(\n event: ICustomEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"custom\"); // this will support all custom event\n const affiliateParams = this.enhanceWithAffiliateParams({});\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a specific event\n * @param event The specific event\n */\n private trackSpecificEvent(\n event: ISpecificEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"\"); // this will support all custom event\n if (!eventName) {\n return;\n }\n const affiliateParams = this.enhanceWithAffiliateParams({});\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track an add payment info event\n * @param event The add payment info event\n */\n private trackAddPaymentInfo(\n event: IAddPaymentInfoEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"add_payment_info\");\n const baseParams = {\n currency: event.currency || \"INR\",\n value: event.cartValue,\n payment_type: event.paymentType || \"\",\n items: event.items.map((item) => ({\n item_id: item.productId,\n item_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n item_variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a view content event\n * @param event The view content event\n */\n private trackViewContent(\n event: IViewContentEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"view_item\");\n const baseParams = {\n currency: event.currency || \"USD\",\n value: event.value || 0,\n items:\n event.items?.map((item: any) => ({\n item_id: item.item_id,\n item_name: item.item_name,\n item_category: item.item_category,\n item_category2: item.item_category2,\n item_category3: item.item_category3,\n item_category4: item.item_category4,\n item_category5: item.item_category5,\n item_brand: item.item_brand,\n item_variant: item.item_variant,\n price: item.price,\n quantity: item.quantity,\n })) || [],\n // GA4-specific parameters\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n campaign: event.campaign || \"\",\n ga_session_id: event.ga_session_id || \"\",\n engagement_time_msec: event.engagement_time_msec,\n session_engaged: event.session_engaged,\n page_title: event.page_title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer,\n user_id: event.user_id,\n user_properties: event.user_properties,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a purchase event\n * @param event The purchase event\n */\n private trackPurchase(\n event: IPurchaseEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"purchase\");\n const baseParams = {\n transaction_id: event.transaction_id,\n value: event.value,\n currency: event.currency || \"USD\",\n coupon: event.coupon,\n shipping: event.shipping,\n tax: event.tax,\n items:\n event.items?.map((item: any) => ({\n item_id: item.item_id,\n item_name: item.item_name,\n item_category: item.item_category,\n item_category2: item.item_category2,\n item_category3: item.item_category3,\n item_category4: item.item_category4,\n item_category5: item.item_category5,\n item_brand: item.item_brand,\n item_variant: item.item_variant,\n price: item.price,\n quantity: item.quantity,\n })) || [],\n // GA4-specific parameters\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n campaign: event.campaign || \"\",\n ga_session_id: event.ga_session_id || \"\",\n engagement_time_msec: event.engagement_time_msec,\n session_engaged: event.session_engaged,\n page_title: event.page_title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer,\n user_id: event.user_id,\n user_properties: event.user_properties,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a view search results event\n * @param event The view search results event\n */\n private trackViewSearchResults(\n event: IViewSearchResultsEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"view_search_results\");\n const baseParams = {\n search_term: event.search_term,\n // GA4-specific parameters\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n campaign: event.campaign || \"\",\n ga_session_id: event.ga_session_id || \"\",\n engagement_time_msec: event.engagement_time_msec,\n session_engaged: event.session_engaged,\n page_title: event.page_title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer,\n user_id: event.user_id,\n user_properties: event.user_properties,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track an order placed event\n * @param event The order placed event\n */\n private trackOrderPlaced(\n event: IOrderPlacedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"purchase\");\n const baseParams = {\n transaction_id: event.order_id,\n value: event.total_amount,\n currency: event.currency || \"USD\",\n items:\n event.items?.map((item: any) => ({\n item_id: item.product_id,\n item_name: item.product_name,\n item_category: item.category,\n item_brand: item.brand,\n item_variant: item.variant,\n price: item.price,\n quantity: item.quantity,\n })) || [],\n // GA4-specific parameters\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n campaign: event.campaign || \"\",\n ga_session_id: event.ga_session_id || \"\",\n engagement_time_msec: event.engagement_time_msec,\n session_engaged: event.session_engaged,\n page_title: event.page_title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer,\n user_id: event.user_id,\n user_properties: event.user_properties,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Track a begin checkout event\n * @param event The begin checkout event\n */\n private trackBeginCheckout(\n event: IBeginCheckoutEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"begin_checkout\");\n const baseParams = {\n currency: event.currency || \"USD\",\n value: event.value || 0,\n coupon: event.coupon,\n items:\n event.items?.map((item: any) => ({\n item_id: item.item_id,\n item_name: item.item_name,\n item_category: item.item_category,\n item_category2: item.item_category2,\n item_category3: item.item_category3,\n item_category4: item.item_category4,\n item_category5: item.item_category5,\n item_brand: item.item_brand,\n item_variant: item.item_variant,\n price: item.price,\n quantity: item.quantity,\n })) || [],\n // GA4-specific parameters\n batch_ordering_id: event.batch_ordering_id || \"\",\n batch_page_id: event.batch_page_id || \"\",\n campaign: event.campaign || \"\",\n ga_session_id: event.ga_session_id || \"\",\n engagement_time_msec: event.engagement_time_msec,\n session_engaged: event.session_engaged,\n page_title: event.page_title || document.title,\n page_location: event.page_location || window.location.href,\n page_referrer: event.page_referrer || document.referrer,\n user_id: event.user_id,\n user_properties: event.user_properties,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.gtag?.(\"event\", eventName, enhancedParams);\n }\n\n /**\n * Convert an event to Google Analytics parameters\n * @param event The event to convert\n */\n private eventToParams(event: TEvent): Record<string, any> {\n // Remove type and timestamp from the event\n const { type, timestamp, ...params } = event;\n const affiliateParams = this.enhanceWithAffiliateParams(params);\n return this.enhanceWithExperimentParams(affiliateParams);\n }\n}\n","/**\n * Event name constants for tracking platforms\n */\nexport const EventNames = {\n // Page events\n PAGE_VIEWED: \"Page Viewed\",\n\n // Interaction events\n BUTTON_CLICKED: \"Button Clicked\",\n FORM_SUBMITTED: \"Form Submitted\",\n\n // Product events\n PRODUCT_VIEWED: \"Product Viewed\",\n ADDED_TO_CART: \"Added To Cart\",\n REMOVED_FROM_CART: \"Removed From Cart\",\n\n // Checkout events\n CHECKOUT_STARTED: \"Checkout Started\",\n PURCHASE_COMPLETED: \"Purchase Completed\",\n\n // Search events\n SEARCH: \"Search\",\n\n // User events\n USER_SIGNED_UP: \"User Signed Up\",\n USER_LOGGED_IN: \"User Logged In\",\n};\n\n/**\n * Default currency code\n */\nexport const DEFAULT_CURRENCY = \"INR\";\n","import { BaseAdapter } from \"./base-adapter\";\nimport {\n type TEvent,\n EventType,\n type IPageViewEvent,\n type IButtonClickEvent,\n type IFormSubmissionEvent,\n type IProductViewEvent,\n type IAddToCartEvent,\n type IRemoveFromCartEvent,\n type ICheckoutStartedEvent,\n type ICheckoutCompletedEvent,\n type ISearchEvent,\n type IUserSignupEvent,\n type IUserLoginEvent,\n type ICustomEvent,\n ISpecificEvent,\n} from \"../types\";\nimport { EventNames, DEFAULT_CURRENCY } from \"../constants\";\nimport type { TAdapterParams } from \"../adapter-params\";\n\nimport moengage from \"@moengage/web-sdk\";\n\n/**\n * Configuration for the MoenGage adapter\n */\nexport interface MoengageAdapterConfig {\n /**\n * MoenGage App ID (required for SDK initialization)\n */\n appId: string;\n /**\n * Whether to enable user identification\n */\n enableUserIdentification?: boolean;\n /**\n * Debug mode for MoenGage SDK\n */\n debug?: boolean;\n /**\n * Data center region (default: 'dc_01')\n */\n region?: string;\n}\n\n/**\n * MoenGage tracking adapter using the official Web SDK\n */\nexport class MoengageAdapter extends BaseAdapter {\n public readonly name = \"MoEngage\";\n\n constructor(config: MoengageAdapterConfig) {\n super(config);\n // Initialize logger with proper adapter name\n this.initializeLogger();\n }\n\n /**\n * Get MoEngage SDK instance from window\n */\n private getMoengageSDK(): any {\n return typeof window !== \"undefined\" ? (window as any).Moengage : null;\n }\n\n /**\n * Initialize the MoenGage adapter using the Web SDK\n */\n public async initialize(): Promise<void> {\n if (this.initialized) {\n this.logger.debug(\"Already initialized\");\n return;\n }\n\n // Skip initialization if running on server\n if (typeof window === \"undefined\") {\n this.logger.debug(\"Skipping initialization on server\");\n return;\n }\n\n try {\n const appId = this.getConfig<string>(\"appId\");\n if (!appId) {\n this.logger.warn(\"App ID is required for initialization\");\n return;\n }\n\n // Initialize MoenGage SDK\n const region = this.getConfig<string>(\"region\", \"dc_01\");\n const config = {\n app_id: appId,\n debug_logs: this.getConfig<boolean>(\"debug\", false) ? 1 : 0,\n cluster: region,\n disableCookies: true,\n };\n\n this.logger.info(\"Initializing with config\", {\n region,\n debug: this.getConfig<boolean>(\"debug\", false),\n appId: \"***\", // Hide sensitive data\n });\n\n // Initialize MoEngage SDK\n moengage.initialize(config);\n\n this.logger.info(\"Successfully initialized with SDK\");\n this.initialized = true;\n } catch (error) {\n this.logger.error(\"Failed to initialize\", error as Error);\n }\n }\n\n /**\n * Track an event with MoenGage SDK\n * @param event The event to track\n * @param adapterParams Optional adapter-specific parameters\n */\n public async trackEvent(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n const sdk = this.getMoengageSDK();\n if (!this.initialized || typeof window === \"undefined\" || !sdk) {\n this.logger.warn(\n \"Cannot track event, adapter not initialized or SDK not available\",\n {\n eventType: event.type,\n initialized: this.initialized,\n hasWindow: typeof window !== \"undefined\",\n hasSdk: !!sdk,\n }\n );\n return;\n }\n\n this.logger.debug(\"Tracking event\", { eventType: event.type });\n\n switch (event.type) {\n case EventType.PAGE_VIEW:\n this.trackPageView(event as IPageViewEvent, adapterParams);\n break;\n case EventType.BUTTON_CLICK:\n this.trackButtonClick(event as IButtonClickEvent, adapterParams);\n break;\n case EventType.FORM_SUBMISSION:\n this.trackFormSubmission(event as IFormSubmissionEvent, adapterParams);\n break;\n case EventType.PRODUCT_VIEW:\n this.trackProductView(event as IProductViewEvent, adapterParams);\n break;\n case EventType.ADD_TO_CART:\n this.trackAddToCart(event as IAddToCartEvent, adapterParams);\n break;\n case EventType.REMOVE_FROM_CART:\n this.trackRemoveFromCart(event as IRemoveFromCartEvent, adapterParams);\n break;\n case EventType.CHECKOUT_STARTED:\n case EventType.INITIATE_CHECKOUT:\n this.trackCheckoutStarted(\n event as ICheckoutStartedEvent,\n adapterParams\n );\n break;\n case EventType.CHECKOUT_COMPLETED:\n case EventType.PURCHASE:\n this.trackCheckoutCompleted(\n event as ICheckoutCompletedEvent,\n adapterParams\n );\n break;\n case EventType.SEARCH:\n this.trackSearch(event as ISearchEvent, adapterParams);\n break;\n case EventType.USER_SIGNUP:\n this.trackUserSignup(event as IUserSignupEvent, adapterParams);\n break;\n case EventType.USER_LOGIN:\n this.trackUserLogin(event as IUserLoginEvent, adapterParams);\n break;\n case EventType.CUSTOM:\n this.trackCustomEvent(event as ICustomEvent, adapterParams);\n break;\n case EventType.SPECIFIC:\n this.trackSpecificEvent(event as ISpecificEvent, adapterParams);\n break;\n default:\n this.logger.debug(\"Unhandled event type\", { eventType: event.type });\n break;\n }\n }\n\n /**\n * Track a page view event\n * @param event The page view event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackPageView(\n event: IPageViewEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n if (!sdk?.track_event) {\n this.logger.warn(\"track_event not available for PageView event\");\n return;\n }\n\n const eventName = this.getEventName(adapterParams, EventNames.PAGE_VIEWED);\n const baseEventData = {\n page_path: event.path,\n page_title: event.title,\n page_url: window.location.href,\n referrer: event.referrer || document.referrer,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n\n sdk.track_event(eventName, enhancedEventData);\n this.logger.debug(\"PageView event tracked\", {\n eventName,\n eventData: enhancedEventData,\n });\n }\n\n /**\n * Track a button click event\n * @param event The button click event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackButtonClick(\n event: IButtonClickEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.BUTTON_CLICKED\n );\n const baseEventData = {\n button_id: event.buttonId,\n button_text: event.buttonText,\n location: event.location,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a form submission event\n * @param event The form submission event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackFormSubmission(\n event: IFormSubmissionEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.FORM_SUBMITTED\n );\n const baseEventData = {\n form_id: event.formId,\n form_name: event.formName,\n success: event.success,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a product view event\n * @param event The product view event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackProductView(\n event: IProductViewEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.PRODUCT_VIEWED\n );\n const baseEventData = {\n product_id: event.productId,\n product_name: event.productName,\n price: event.price,\n currency: event.currency || DEFAULT_CURRENCY,\n category: event.category,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track an add to cart event\n * @param event The add to cart event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackAddToCart(\n event: IAddToCartEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const quantity = event.quantity ?? 1;\n const eventName = this.getEventName(\n adapterParams,\n EventNames.ADDED_TO_CART\n );\n const baseEventData = {\n product_id: event.productId,\n product_name: event.productName,\n price: event.price,\n currency: event.currency || DEFAULT_CURRENCY,\n quantity: quantity,\n variant: event.variant,\n total_value: event.price * quantity,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a remove from cart event\n * @param event The remove from cart event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackRemoveFromCart(\n event: IRemoveFromCartEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.REMOVED_FROM_CART\n );\n const baseEventData = {\n product_id: event.productId,\n product_name: event.productName,\n price: event.price,\n currency: event.currency || DEFAULT_CURRENCY,\n quantity: event.quantity,\n variant: event.variant,\n total_value: event.price * event.quantity,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a checkout started event\n * @param event The checkout started event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackCheckoutStarted(\n event: ICheckoutStartedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.CHECKOUT_STARTED\n );\n const baseEventData = {\n cart_value: event.cartValue,\n currency: event.currency || DEFAULT_CURRENCY,\n item_count: event.itemCount,\n items: event.items.map((item) => ({\n product_id: item.productId,\n product_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a checkout completed event\n * @param event The checkout completed event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackCheckoutCompleted(\n event: ICheckoutCompletedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.PURCHASE_COMPLETED\n );\n const baseEventData = {\n order_id: event.orderId,\n cart_value: event.cartValue,\n currency: event.currency || DEFAULT_CURRENCY,\n item_count: event.itemCount,\n items: event.items.map((item) => ({\n product_id: item.productId,\n product_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a search event\n * @param event The search event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackSearch(\n event: ISearchEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(adapterParams, EventNames.SEARCH);\n const baseEventData = {\n search_term: event.searchTerm,\n results_count: event.resultsCount,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a user signup event\n * @param event The user signup event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackUserSignup(\n event: IUserSignupEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.USER_SIGNED_UP\n );\n const baseEventData = {\n user_id: event.userId,\n method: event.method,\n success: event.success,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n\n // Add user attributes if user ID is available and user identification is enabled\n if (\n event.userId &&\n this.getConfig<boolean>(\"enableUserIdentification\", true)\n ) {\n sdk?.add_unique_user_id(event.userId);\n }\n }\n\n /**\n * Track a user login event\n * @param event The user login event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackUserLogin(\n event: IUserLoginEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(\n adapterParams,\n EventNames.USER_LOGGED_IN\n );\n const baseEventData = {\n user_id: event.userId,\n method: event.method,\n success: event.success,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseEventData);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n sdk?.track_event(eventName, enhancedEventData);\n\n // Add user attributes if user ID is available and user identification is enabled\n if (\n event.userId &&\n this.getConfig<boolean>(\"enableUserIdentification\", true)\n ) {\n sdk?.add_unique_user_id(event.userId);\n }\n }\n\n /**\n * Track a custom event\n * @param event The custom event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackCustomEvent(\n event: ICustomEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(adapterParams, \"custom\");\n const affiliateParams = this.enhanceWithAffiliateParams({});\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedEventData = this.mergeEventData(\n experimentParams,\n adapterParams\n );\n\n sdk?.track_event(eventName, enhancedEventData);\n }\n\n /**\n * Track a specific event\n * @param event The specific event\n */\n private trackSpecificEvent(\n event: ISpecificEvent,\n adapterParams?: TAdapterParams\n ): void {\n const sdk = this.getMoengageSDK();\n const eventName = this.getEventName(adapterParams, \"\"); // this will support all custom event\n if (!eventName) {\n return;\n }\n const affiliateParams = this.enhanceWithAffiliateParams({});\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n sdk?.track_event(eventName, enhancedParams);\n }\n\n /**\n * Set user attributes in MoenGage\n * @param attributes User attributes to set\n */\n public setUserAttributes(attributes: Record<string, any>): void {\n const sdk = this.getMoengageSDK();\n if (!this.initialized || typeof window === \"undefined\" || !sdk) {\n this.logger.warn(\"Cannot set user attributes, adapter not initialized\");\n return;\n }\n\n this.logger.debug(\"Setting user attributes\", {\n attributeKeys: Object.keys(attributes),\n });\n\n // Set user attributes\n for (const [key, value] of Object.entries(attributes)) {\n switch (key) {\n case \"firstName\":\n sdk?.add_first_name(value as string);\n break;\n case \"lastName\":\n sdk?.add_last_name(value as string);\n break;\n case \"email\":\n sdk?.add_email(value as string);\n break;\n case \"mobile\":\n sdk?.add_mobile(value as string);\n break;\n case \"username\":\n sdk?.add_user_name(value as string);\n break;\n case \"gender\":\n sdk?.add_gender(value as string);\n break;\n case \"birthday\":\n sdk?.add_birthday(value as Date);\n break;\n case \"userId\":\n sdk?.add_unique_user_id(value as string);\n break;\n default:\n sdk?.add_user_attribute(key, value);\n break;\n }\n }\n }\n}\n","import { BaseAdapter } from \"./base-adapter\";\nimport {\n type TEvent,\n EventType,\n type IPageViewEvent,\n type IAddToCartEvent,\n type ICheckoutStartedEvent,\n type ICheckoutCompletedEvent,\n type IProductViewEvent,\n type ISearchEvent,\n type IAddPaymentInfoEvent,\n ICustomEvent,\n ISpecificEvent,\n} from \"../types\";\nimport type { TAdapterParams } from \"../adapter-params\";\n\ndeclare global {\n interface Window {\n posthog?: {\n capture: (eventName: string, properties?: Record<string, any>) => void;\n identify: (distinctId: string, properties?: Record<string, any>) => void;\n };\n }\n}\n\n/**\n * Configuration for the PostHog adapter\n */\nexport interface PostHogAdapterConfig {\n /**\n * Enable debug logs\n */\n enableDebugLogs?: boolean;\n}\n\n/**\n * PostHog tracking adapter\n * Assumes that the PostHog script is already loaded in the page\n */\nexport class PostHogAdapter extends BaseAdapter {\n public readonly name = \"PostHog\";\n\n constructor(config: PostHogAdapterConfig = {}) {\n super(config);\n // Initialize logger with proper adapter name\n this.initializeLogger();\n }\n\n /**\n * Initialize PostHog\n */\n public async initialize(): Promise<void> {\n if (this.initialized) {\n this.logger.debug(\"Already initialized\");\n return;\n }\n\n // Skip initialization if running on server\n if (typeof window === \"undefined\") {\n this.logger.debug(\"Skipping initialization on server\");\n return;\n }\n\n // Check if posthog is available\n if (!window.posthog) {\n this.logger.warn(\n \"PostHog object not found. Make sure the script is loaded.\"\n );\n return;\n }\n\n this.logger.info(\"Successfully initialized\");\n this.initialized = true;\n }\n\n /**\n * Track an event with PostHog\n * @param event The event to track\n * @param adapterParams Optional adapter-specific parameters\n */\n public async trackEvent(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n if (!this.initialized || typeof window === \"undefined\" || !window.posthog) {\n this.logger.warn(\n \"Cannot track event, adapter not initialized or posthog not available\",\n {\n eventType: event.type,\n initialized: this.initialized,\n hasWindow: typeof window !== \"undefined\",\n hasPosthog: !!window?.posthog,\n }\n );\n return;\n }\n\n this.logger.debug(\"Tracking event\", { eventType: event.type });\n\n switch (event.type) {\n case EventType.PAGE_VIEW:\n this.trackPageView(event, adapterParams);\n break;\n case EventType.PRODUCT_VIEW:\n this.trackProductView(event, adapterParams);\n break;\n case EventType.ADD_TO_CART:\n this.trackAddToCart(event, adapterParams);\n break;\n case EventType.CHECKOUT_STARTED:\n case EventType.INITIATE_CHECKOUT:\n this.trackCheckoutStarted(event as any, adapterParams);\n break;\n case EventType.CHECKOUT_COMPLETED:\n case EventType.PURCHASE:\n this.trackCheckoutCompleted(event as any, adapterParams);\n break;\n case EventType.ADD_PAYMENT_INFO:\n this.trackAddPaymentInfo(event, adapterParams);\n break;\n case EventType.SEARCH:\n this.trackSearch(event, adapterParams);\n break;\n case EventType.CUSTOM:\n this.trackCustomEvent(event, adapterParams);\n break;\n case EventType.SPECIFIC:\n this.trackSpecificEvent(event, adapterParams);\n break;\n default:\n this.logger.debug(\"Unhandled event type\", { eventType: event.type });\n break;\n }\n }\n\n /**\n * Track a custom event\n * @param event The custom event\n */\n private trackCustomEvent(\n event: ICustomEvent,\n adapterParams?: TAdapterParams\n ): void {\n // For other events, track as custom events\n const eventName = this.getEventName(adapterParams, \"custom\");\n const affiliateParams = this.enhanceWithAffiliateParams({});\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n private trackSpecificEvent(\n event: ISpecificEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"\"); // this will support all custom event\n if (!eventName) {\n return;\n }\n const affiliateParams = this.enhanceWithAffiliateParams({});\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n /**\n * Track a page view event\n * @param event The page view event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackPageView(\n event: IPageViewEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"$pageview\");\n const baseParams = {\n path: event.path,\n title: event.title,\n referrer: event.referrer,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n /**\n * Track a product view event\n * @param event The product view event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackProductView(\n event: IProductViewEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"product_viewed\");\n const baseParams = {\n product_id: event.productId,\n product_name: event.productName,\n category: event.category,\n price: event.price,\n currency: event.currency || \"INR\",\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n /**\n * Track an add to cart event\n * @param event The add to cart event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackAddToCart(\n event: IAddToCartEvent,\n adapterParams?: TAdapterParams\n ): void {\n this.logger.debug(\"Handling ADD_TO_CART event\", { event });\n\n const eventName = this.getEventName(adapterParams, \"add_to_cart\");\n const baseParams = {\n product_id: event.productId,\n product_name: event.productName,\n price: event.price,\n currency: event.currency || \"INR\",\n quantity: event.quantity,\n variant: event.variant,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n this.logger.debug(\"Sending to PostHog with params\", {\n eventName,\n params: enhancedParams,\n });\n\n if (!window.posthog) {\n this.logger.error(\"window.posthog is not available\");\n return;\n }\n\n window.posthog.capture(eventName, enhancedParams);\n this.logger.debug(\"Event sent to PostHog successfully\");\n }\n\n /**\n * Track a checkout started event\n * @param event The checkout started event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackCheckoutStarted(\n event: ICheckoutStartedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"checkout_started\");\n const baseParams = {\n cart_value: event.cartValue,\n currency: event.currency || \"INR\",\n item_count: event.itemCount,\n items: event.items.map((item) => ({\n product_id: item.productId,\n product_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n /**\n * Track a checkout completed event\n * @param event The checkout completed event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackCheckoutCompleted(\n event: ICheckoutCompletedEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"purchase\");\n const baseParams = {\n order_id: event.orderId,\n cart_value: event.cartValue,\n currency: event.currency || \"INR\",\n item_count: event.itemCount,\n items: event.items.map((item) => ({\n product_id: item.productId,\n product_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n /**\n * Track a search event\n * @param event The search event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackSearch(\n event: ISearchEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"search\");\n const baseParams = {\n search_term: event.searchTerm,\n results_count: event.resultsCount,\n content_ids: event.content_ids,\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n\n /**\n * Track an add payment info event\n * @param event The add payment info event\n * @param adapterParams Optional adapter-specific parameters\n */\n private trackAddPaymentInfo(\n event: IAddPaymentInfoEvent,\n adapterParams?: TAdapterParams\n ): void {\n const eventName = this.getEventName(adapterParams, \"add_payment_info\");\n const baseParams = {\n cart_value: event.cartValue,\n currency: event.currency || \"INR\",\n payment_type: event.paymentType,\n item_count: event.itemCount,\n items: event.items.map((item) => ({\n product_id: item.productId,\n product_name: item.productName,\n price: item.price,\n quantity: item.quantity,\n variant: item.variant,\n })),\n };\n\n const affiliateParams = this.enhanceWithAffiliateParams(baseParams);\n const experimentParams = this.enhanceWithExperimentParams(affiliateParams);\n const enhancedParams = this.mergeEventData(experimentParams, adapterParams);\n\n window.posthog?.capture(eventName, enhancedParams);\n }\n}\n","import {\n sendShopifyAnalytics,\n getClientBrowserParameters,\n AnalyticsEventName,\n type ShopifyAnalyticsProduct,\n type ShopifyPageViewPayload,\n ShopifySalesChannel,\n} from \"@shopify/hydrogen-react\";\nimport { BaseAdapter } from \"./base-adapter\";\nimport type { TEvent } from \"../types\";\nimport { EventType } from \"../types\";\nimport type { TAdapterParams } from \"../adapter-params\";\n\n// Constants based on environment variables\nconst DEFAULT_CURRENCY = \"INR\";\nconst DEFAULT_LANGUAGE = \"en\";\n\n/**\n * Type definitions for analytics payloads\n */\ntype SendPageViewPayload = {\n pageType?: string;\n products?: ShopifyAnalyticsProduct[];\n collectionHandle?: string;\n searchString?: string;\n totalValue?: number;\n cartId?: string;\n};\n\ntype SendAddToCartPayload = {\n cartId: string;\n products?: ShopifyAnalyticsProduct[];\n totalValue?: ShopifyPageViewPayload[\"totalValue\"];\n};\n\n/**\n * Configuration interface for Shopify Analytics Adapter\n */\nexport interface ShopifyAdapterConfig {\n shopId: string;\n domain: string;\n currency?: string;\n}\n\n/**\n * Shopify Analytics Adapter for tracking events to Shopify Admin Analytics\n */\nexport class ShopifyAdapter extends BaseAdapter {\n public readonly name = \"ShopifyAnalytics\";\n private shopId: string;\n private domain: string;\n private currency: string;\n\n constructor(config: ShopifyAdapterConfig) {\n super(config);\n this.shopId = config.shopId;\n this.domain = config.domain;\n this.currency = config.currency || DEFAULT_CURRENCY;\n }\n\n /**\n * Initialize the Shopify Analytics adapter\n */\n public async initialize(): Promise<void> {\n this.initializeLogger();\n\n try {\n // Enable Shopify cookies for attribution tracking\n if (typeof window !== \"undefined\") {\n this.logger.info(\"Shopify Analytics adapter initialized successfully\", {\n shopId: this.shopId,\n domain: this.domain,\n currency: this.currency,\n });\n\n // Note: useShopifyCookies should be called at the React component level\n // This will be handled in the ShopkitAnalytics component\n }\n this.initialized = true;\n } catch (error) {\n this.logger.error(\n \"Failed to initialize Shopify Analytics\",\n error instanceof Error ? error : new Error(String(error))\n );\n throw error;\n }\n }\n\n /**\n * Convert product ID to Shopify GID format if it's not already\n */\n private formatProductGid(productId: string): string {\n return productId.startsWith(\"gid://shopify/Product/\")\n ? productId\n : `gid://shopify/Product/${productId}`;\n }\n\n /**\n * Get user consent status from your consent management system\n */\n private getUserConsent() {\n // Check if user has given consent (implement based on your consent system)\n const hasConsent = this.checkUserConsent();\n\n return {\n hasUserConsent: hasConsent,\n analyticsAllowed: hasConsent && this.checkAnalyticsConsent(),\n marketingAllowed: hasConsent && this.checkMarketingConsent(),\n saleOfDataAllowed: hasConsent && this.checkSaleOfDataConsent(),\n };\n }\n\n /**\n * Check user consent (implement based on your consent management)\n */\n private checkUserConsent(): boolean {\n // Example implementations:\n // return localStorage.getItem('user_consent') === 'true';\n // return document.cookie.includes('consent=granted');\n // return window.gtag?.('consent', 'query') === 'granted';\n\n // For now, return true but implement your actual consent logic\n return true;\n }\n\n private checkAnalyticsConsent(): boolean {\n // Implement your analytics consent check\n return true;\n }\n\n private checkMarketingConsent(): boolean {\n // Implement your marketing consent check\n return true;\n }\n\n private checkSaleOfDataConsent(): boolean {\n // Implement your sale of data consent check\n return true;\n }\n\n /**\n * Create base payload for all events\n */\n private createBasePayload(): any {\n const consent = this.getUserConsent();\n\n return {\n ...getClientBrowserParameters(),\n ...consent,\n storefrontId: this.shopId,\n shopifySalesChannel: ShopifySalesChannel.headless,\n shopId: `gid://shopify/Shop/${this.shopId}`,\n currency: this.currency,\n acceptedLanguage: DEFAULT_LANGUAGE,\n ...this.enhanceWithExperimentParams(this.enhanceWithAffiliateParams({})),\n };\n }\n\n /**\n * Send page view event using the enhanced functionality\n */\n public sendPageView = (\n eventName: keyof typeof AnalyticsEventName,\n payload?: SendPageViewPayload\n ) => {\n const enhancedPayload = {\n ...this.createBasePayload(),\n ...payload,\n };\n\n return sendShopifyAnalytics({\n eventName,\n payload: enhancedPayload,\n });\n };\n\n /**\n * Send add to cart event using the enhanced functionality\n */\n public sendAddToCart = ({\n cartId,\n totalValue,\n products,\n }: SendAddToCartPayload) => {\n return this.sendPageView(AnalyticsEventName.ADD_TO_CART, {\n cartId,\n totalValue,\n products,\n });\n };\n\n /**\n * Track an event using Shopify Analytics\n */\n public async trackEvent(\n event: TEvent,\n adapterParams?: TAdapterParams\n ): Promise<void> {\n if (!this.initialized || typeof window === \"undefined\") {\n this.logger.warn(\n \"Shopify Analytics not initialized or not in browser environment\"\n );\n return;\n }\n\n // Get adapter-specific parameters\n const params = this.getAdapterParams(adapterParams);\n const customEventName = this.getEventName(params, event.type);\n\n this.logger.debug(\"Processing Shopify Analytics event\", {\n eventType: event.type,\n customEventName,\n eventData: event,\n adapterParams: params,\n });\n\n // Enhance event with affiliate and experiment data\n const affiliateParams = this.enhanceWithAffiliateParams(event);\n const enhancedEvent = this.enhanceWithExperimentParams(affiliateParams);\n\n try {\n switch (event.type) {\n case EventType.SHOPIFY_PAGE_VIEW:\n case EventType.PAGE_VIEW:\n const pageViewPayload = this.mergeEventData(enhancedEvent, params);\n this.sendPageView(AnalyticsEventName.PAGE_VIEW, pageViewPayload);\n this.logger.debug(\"Sent PAGE_VIEW event to Shopify Analytics\", {\n payload: pageViewPayload,\n });\n break;\n\n case EventType.VIEW_CONTENT:\n const viewContentPayload = this.mergeEventData(enhancedEvent, params);\n this.sendPageView(\n AnalyticsEventName.PRODUCT_VIEW,\n viewContentPayload\n );\n this.logger.debug(\"Sent PRODUCT_VIEW event to Shopify Analytics\", {\n payload: viewContentPayload,\n });\n break;\n\n case EventType.ADD_TO_CART:\n const addToCartPayload = this.mergeEventData(enhancedEvent, params);\n this.trackAddToCart(addToCartPayload);\n break;\n\n case EventType.CHECKOUT_STARTED:\n case EventType.INITIATE_CHECKOUT:\n // For checkout events, send as PAGE_VIEW with checkout page context\n const checkoutPayload = this.mergeEventData(\n { ...enhancedEvent, pageType: \"checkout\" },\n params\n );\n this.sendPageView(AnalyticsEventName.PAGE_VIEW, checkoutPayload);\n this.logger.debug(\n \"Sent checkout PAGE_VIEW event to Shopify Analytics\",\n { payload: checkoutPayload }\n );\n break;\n\n default:\n this.logger.debug(\"Event not supported by Shopify Analytics\", {\n eventType: event.type,\n });\n break;\n }\n } catch (error) {\n this.logger.error(\n \"Error tracking Shopify Analytics event\",\n error instanceof Error ? error : new Error(String(error)),\n { eventType: event.type }\n );\n }\n }\n\n /**\n * Check if Shopify session is properly established\n */\n public isSessionValid(): boolean {\n if (typeof window === \"undefined\") return false;\n\n const shopifyY = document.cookie.includes(\"_shopify_y=\");\n const shopifyS = document.cookie.includes(\"_shopify_s=\");\n\n return shopifyY && shopifyS;\n }\n\n /**\n * Get session debug information\n */\n public getSessionDebugInfo(): object {\n if (typeof window === \"undefined\")\n return { error: \"Not in browser environment\" };\n\n return {\n sessionValid: this.isSessionValid(),\n cookies: {\n shopify_y: document.cookie.includes(\"_shopify_y=\"),\n shopify_s: document.cookie.includes(\"_shopify_s=\"),\n all: document.cookie,\n },\n shopId: this.shopId,\n initialized: this.initialized,\n };\n }\n\n /**\n * Track add to cart event with enhanced error handling\n */\n private trackAddToCart(event: any): void {\n if (!event.productId) {\n this.logger.warn(\"ADD_TO_CART event missing productId\");\n return;\n }\n\n try {\n const productGid = this.formatProductGid(event.productId);\n const cartKey = `merchant_${process.env.NEXT_PUBLIC_MERCHANT_NAME}_cartId`;\n const cartId = localStorage.getItem(cartKey) || \"\";\n\n if (!cartId) {\n this.logger.warn(\"No cart ID found, creating anonymous session\");\n // You might want to generate a temporary cart ID or handle this case\n }\n\n // Validate product data\n if (!event.price || event.price <= 0) {\n this.logger.warn(\"Invalid or missing price for ADD_TO_CART\", {\n price: event.price,\n });\n }\n\n // Create products array for analytics\n const productPrice = Math.max(0, event.price || 0);\n const productQuantity = Math.max(1, event.quantity || 1);\n\n const products: ShopifyAnalyticsProduct[] = [\n {\n productGid: productGid,\n variantGid:\n event.variantId || productGid.replace(\"Product\", \"ProductVariant\"),\n quantity: productQuantity,\n price: productPrice.toString(), // Convert to string as required by ShopifyAnalyticsProduct\n name: event.productName || \"Unknown Product\",\n brand: event.brand || \"Unknown Brand\",\n },\n ];\n\n const totalValue = productPrice * productQuantity;\n\n this.logger.debug(\"ADD_TO_CART event tracked successfully\", {\n productGid,\n quantity: productQuantity,\n price: productPrice,\n totalValue,\n });\n\n // Use the enhanced sendAddToCart method\n this.sendAddToCart({\n cartId,\n products,\n totalValue,\n });\n } catch (error) {\n this.logger.error(\n \"Error in trackAddToCart\",\n error instanceof Error ? error : new Error(String(error))\n );\n // Don't throw - just log the error to prevent breaking the user experience\n }\n }\n}\n","import {\n PixelAdapter,\n GoogleAdapter,\n MoengageAdapter,\n PostHogAdapter,\n} from \"../adapters\";\n\n/**\n * Initialize the event tracking system with the provided adapters\n * @param adapters Array of tracking adapters to register\n */\nexport async function initializeEventTracking(\n adapters: import(\"./subscriber\").TrackingAdapter[]\n): Promise<void> {\n // Import the subscriber\n const { eventSubscriber } = await import(\"./subscriber\");\n\n // Register each adapter using for...of instead of forEach\n for (const adapter of adapters) {\n eventSubscriber.registerAdapter(adapter);\n }\n\n // Initialize the subscriber\n await eventSubscriber.initialize();\n}\n\n/**\n * Example initialization function with common adapters\n * This is a convenience function that shows how to set up common tracking adapters\n * Users can customize this based on their needs\n */\nexport async function initTracking(config: {\n pixelId?: string;\n gaId?: string;\n moengageAppId?: string;\n moengageRegion?: string;\n enableDebugLogs?: boolean;\n}) {\n console.log(\"Initializing event tracking system...\");\n\n const adapters = [];\n\n // Facebook Pixel\n if (config.pixelId) {\n const pixelAdapter = new PixelAdapter({\n pixelId: config.pixelId,\n });\n adapters.push(pixelAdapter);\n }\n\n // Google Analytics\n if (config.gaId) {\n const gaAdapter = new GoogleAdapter({\n measurementId: config.gaId,\n });\n adapters.push(gaAdapter);\n }\n\n // MoEngage\n if (config.moengageAppId) {\n const moengageAdapter = new MoengageAdapter({\n appId: config.moengageAppId,\n enableUserIdentification: true,\n region: config.moengageRegion || \"dc_01\",\n debug: config.enableDebugLogs || false,\n });\n adapters.push(moengageAdapter);\n }\n\n // PostHog\n const posthogAdapter = new PostHogAdapter({\n enableDebugLogs: config.enableDebugLogs || false,\n });\n adapters.push(posthogAdapter);\n\n console.log(\n \"Configured adapters:\",\n adapters.map((a) => a.name)\n );\n\n await initializeEventTracking(adapters);\n console.log(\"Event tracking system initialized successfully\");\n}\n"],"mappings":"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","names":["EventType","moengage","eventSubscriber"]}
|