@envive-ai/react-hooks 0.3.21 → 0.3.23
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/application/models/featureGates.cjs +2 -1
- package/dist/application/models/featureGates.d.cts +2 -1
- package/dist/application/models/featureGates.d.ts +2 -1
- package/dist/application/models/featureGates.js +2 -1
- package/dist/atoms/app/index.d.cts +7 -7
- package/dist/atoms/app/variant.d.cts +6 -6
- package/dist/atoms/app/variant.d.ts +6 -6
- package/dist/atoms/chat/chatState.cjs +3 -1
- package/dist/atoms/chat/chatState.d.cts +22 -19
- package/dist/atoms/chat/chatState.d.ts +22 -19
- package/dist/atoms/chat/chatState.js +3 -2
- package/dist/atoms/chat/form.d.cts +3 -3
- package/dist/atoms/chat/form.d.ts +2 -2
- package/dist/atoms/chat/index.cjs +1 -0
- package/dist/atoms/chat/index.d.cts +4 -4
- package/dist/atoms/chat/index.d.ts +4 -4
- package/dist/atoms/chat/index.js +2 -2
- package/dist/atoms/chat/lastMessage.d.cts +2 -2
- package/dist/atoms/chat/lastMessage.d.ts +2 -2
- package/dist/atoms/chat/messageQueue.d.cts +7 -7
- package/dist/atoms/chat/messageQueue.d.ts +6 -6
- package/dist/atoms/chat/performanceMetrics.d.cts +6 -6
- package/dist/atoms/chat/performanceMetrics.d.ts +6 -6
- package/dist/atoms/chat/renderedWidgetRefs.d.cts +3 -3
- package/dist/atoms/chat/renderedWidgetRefs.d.ts +2 -2
- package/dist/atoms/chat/replies.d.cts +3 -3
- package/dist/atoms/chat/replies.d.ts +2 -2
- package/dist/atoms/chat/suggestions.d.cts +3 -3
- package/dist/atoms/chat/suggestions.d.ts +2 -2
- package/dist/atoms/envive/enviveConfig.d.cts +13 -13
- package/dist/atoms/envive/enviveConfig.d.ts +1 -1
- package/dist/atoms/globalSearch/globalSearch.d.cts +5 -5
- package/dist/atoms/globalSearch/globalSearch.d.ts +5 -5
- package/dist/atoms/org/customerService.d.cts +6 -6
- package/dist/atoms/org/customerService.d.ts +6 -6
- package/dist/atoms/org/graphqlConfig.d.cts +4 -4
- package/dist/atoms/org/graphqlConfig.d.ts +4 -4
- package/dist/atoms/org/newOrgConfigAtom.d.cts +2 -2
- package/dist/atoms/org/newOrgConfigAtom.d.ts +2 -2
- package/dist/atoms/org/orgAnalyticsConfig.d.cts +5 -5
- package/dist/atoms/org/orgAnalyticsConfig.d.ts +5 -5
- package/dist/atoms/search/chatSearch.d.cts +17 -17
- package/dist/atoms/search/chatSearch.d.ts +17 -17
- package/dist/atoms/search/searchAPI.d.cts +13 -13
- package/dist/atoms/search/types.d.cts +1 -1
- package/dist/atoms/search/types.d.ts +1 -1
- package/dist/atoms/search/utils.d.ts +1 -1
- package/dist/atoms/widget/chatPreviewLoading.d.cts +2 -2
- package/dist/atoms/widget/chatPreviewLoading.d.ts +2 -2
- package/dist/contexts/amplitudeContext/amplitudeContext.cjs +9 -3
- package/dist/contexts/amplitudeContext/amplitudeContext.d.cts +2 -1
- package/dist/contexts/amplitudeContext/amplitudeContext.d.ts +2 -1
- package/dist/contexts/amplitudeContext/amplitudeContext.js +9 -3
- package/dist/contexts/enviveContext/enviveContext.cjs +3 -3
- package/dist/contexts/enviveContext/enviveContext.js +3 -3
- package/dist/contexts/enviveContext/types.d.ts +1 -1
- package/dist/contexts/hardcopyContext/hardcopyContext.cjs +5 -3
- package/dist/contexts/hardcopyContext/hardcopyContext.js +5 -3
- package/dist/contexts/salesAgentContext/chatAPI.cjs +12 -5
- package/dist/contexts/salesAgentContext/chatAPI.js +13 -6
- package/dist/contexts/systemSettingsContext/systemSettingsContext.d.cts +2 -2
- package/dist/contexts/types.d.cts +1 -1
- package/dist/contexts/typesV3.cjs +1 -1
- package/dist/contexts/typesV3.d.cts +2 -1
- package/dist/contexts/typesV3.d.ts +2 -1
- package/dist/contexts/typesV3.js +1 -1
- package/dist/hooks/GrabAndScroll/useGrabAndScroll.d.cts +2 -2
- package/dist/hooks/Search/useSearch.cjs +12 -4
- package/dist/hooks/Search/useSearch.js +12 -4
- package/dist/hooks/Search/useSearchInput.cjs +1 -1
- package/dist/hooks/Search/useSearchInput.js +1 -1
- package/dist/hooks/SystemSettingsContext/useSystemSettingsContext.d.cts +2 -2
- package/dist/hooks/TrackComponentVisibleEvent/useTrackComponentVisibleEvent.cjs +26 -27
- package/dist/hooks/TrackComponentVisibleEvent/useTrackComponentVisibleEvent.d.cts +8 -8
- package/dist/hooks/TrackComponentVisibleEvent/useTrackComponentVisibleEvent.d.ts +8 -8
- package/dist/hooks/TrackComponentVisibleEvent/useTrackComponentVisibleEvent.js +27 -28
- package/dist/hooks/WidgetInteraction/types.cjs +29 -1
- package/dist/hooks/WidgetInteraction/types.d.cts +17 -3
- package/dist/hooks/WidgetInteraction/types.d.ts +17 -3
- package/dist/hooks/WidgetInteraction/types.js +28 -2
- package/dist/hooks/WidgetInteraction/useWidgetInteraction.cjs +6 -2
- package/dist/hooks/WidgetInteraction/useWidgetInteraction.js +6 -2
- package/dist/hooks/utils.d.cts +1 -1
- package/dist/hooks/utils.d.ts +1 -1
- package/dist/services/amplitudeService/amplitudeService.cjs +9 -1
- package/dist/services/amplitudeService/amplitudeService.d.cts +2 -1
- package/dist/services/amplitudeService/amplitudeService.d.ts +2 -1
- package/dist/services/amplitudeService/amplitudeService.js +9 -1
- package/dist/services/ga4ProjectionService/ga4EventSchema.cjs +31 -27
- package/dist/services/ga4ProjectionService/ga4EventSchema.js +31 -27
- package/dist/services/ga4ProjectionService/ga4ProjectionService.cjs +31 -5
- package/dist/services/ga4ProjectionService/ga4ProjectionService.js +31 -5
- package/package.json +1 -1
- package/src/application/models/featureGates.ts +1 -0
- package/src/atoms/chat/chatState.ts +1 -0
- package/src/contexts/amplitudeContext/__tests__/amplitudeContext.test.tsx +31 -27
- package/src/contexts/amplitudeContext/amplitudeContext.tsx +5 -2
- package/src/contexts/hardcopyContext/hardcopyContext.tsx +10 -2
- package/src/contexts/pageContext/__tests__/pageContext.test.tsx +10 -0
- package/src/contexts/salesAgentContext/chatAPI.ts +6 -2
- package/src/contexts/typesV3.ts +1 -0
- package/src/hooks/Search/__tests__/useSearch.test.tsx +0 -4
- package/src/hooks/Search/useSearch.tsx +14 -8
- package/src/hooks/TrackComponentVisibleEvent/useTrackComponentVisibleEvent.ts +36 -35
- package/src/hooks/WidgetInteraction/types.ts +35 -2
- package/src/hooks/WidgetInteraction/useWidgetInteraction.ts +3 -1
- package/src/services/amplitudeService/__tests__/amplitudeService.test.ts +69 -6
- package/src/services/amplitudeService/amplitudeService.ts +13 -0
- package/src/services/ga4ProjectionService/__tests__/ga4ProjectionService.test.ts +110 -49
- package/src/services/ga4ProjectionService/ga4EventSchema.ts +35 -27
- package/src/services/ga4ProjectionService/ga4ProjectionService.ts +60 -6
|
@@ -24,31 +24,34 @@ import { EnviveMetricsEventName } from "../amplitudeService/eventNames.js";
|
|
|
24
24
|
* IF IN DOUBT, DON'T HESITATE TO REACH OUT TO THE ANALYTICS TEAM!!!
|
|
25
25
|
*/
|
|
26
26
|
const WIDGET_INTERACTION_DATA_PROJECTIONS = {
|
|
27
|
-
widget_collapsed: {
|
|
28
|
-
product_card_clicked: {
|
|
29
|
-
suggestion_scrolled: {
|
|
27
|
+
widget_collapsed: { interaction_collapse_source: "widget_collapsed.collapse_source" },
|
|
28
|
+
product_card_clicked: { interaction_product_id: "product_card_clicked.product_id" },
|
|
29
|
+
suggestion_scrolled: { interaction_suggestion_id: "suggestion_scrolled.suggestion_id" }
|
|
30
30
|
};
|
|
31
31
|
const GA4_EVENT_SCHEMA = {
|
|
32
32
|
[EnviveMetricsEventName.WidgetRendered]: {
|
|
33
33
|
gaEventName: "envive_widget_rendered",
|
|
34
|
-
allowedFields: [
|
|
35
|
-
|
|
36
|
-
"context.
|
|
37
|
-
"
|
|
38
|
-
"
|
|
39
|
-
"
|
|
40
|
-
|
|
34
|
+
allowedFields: [],
|
|
35
|
+
fieldProjections: {
|
|
36
|
+
page_type: "context.page_type",
|
|
37
|
+
page_id: "context.page_id",
|
|
38
|
+
surface: "context.surface",
|
|
39
|
+
widget: "trigger.widget",
|
|
40
|
+
widget_role: "trigger.widget_role"
|
|
41
|
+
}
|
|
41
42
|
},
|
|
42
43
|
[EnviveMetricsEventName.WidgetInteraction]: {
|
|
43
44
|
gaEventName: "envive_widget_interaction",
|
|
44
|
-
allowedFields: [
|
|
45
|
-
|
|
46
|
-
"context.
|
|
47
|
-
"
|
|
48
|
-
"
|
|
49
|
-
"trigger.
|
|
50
|
-
"
|
|
51
|
-
|
|
45
|
+
allowedFields: [],
|
|
46
|
+
fieldProjections: {
|
|
47
|
+
page_type: "context.page_type",
|
|
48
|
+
page_id: "context.page_id",
|
|
49
|
+
surface: "context.surface",
|
|
50
|
+
widget: "trigger.widget",
|
|
51
|
+
interaction_type: "trigger.widget_interaction",
|
|
52
|
+
interaction_class: "trigger.interaction_class",
|
|
53
|
+
widget_role: "trigger.widget_role"
|
|
54
|
+
},
|
|
52
55
|
widgetInteractionDataProjections: WIDGET_INTERACTION_DATA_PROJECTIONS
|
|
53
56
|
},
|
|
54
57
|
[EnviveMetricsEventName.ChatRequest]: {
|
|
@@ -90,14 +93,15 @@ const GA4_EVENT_SCHEMA = {
|
|
|
90
93
|
},
|
|
91
94
|
[EnviveMetricsEventName.PageViewed]: {
|
|
92
95
|
gaEventName: "envive_page_context_evaluated",
|
|
93
|
-
allowedFields: [
|
|
94
|
-
|
|
95
|
-
"context.
|
|
96
|
-
"context.
|
|
97
|
-
"context.
|
|
98
|
-
"context.
|
|
99
|
-
"
|
|
100
|
-
|
|
96
|
+
allowedFields: [],
|
|
97
|
+
fieldProjections: {
|
|
98
|
+
page_type: "context.page_type",
|
|
99
|
+
page_id: "context.page_id",
|
|
100
|
+
context_supported: "context.supported",
|
|
101
|
+
context_ready: "context.ready",
|
|
102
|
+
context_page_variant_id: "context.page_variant_id",
|
|
103
|
+
environment_envive_enabled: "environment.envive_enabled"
|
|
104
|
+
}
|
|
101
105
|
},
|
|
102
106
|
[EnviveMetricsEventName.WidgetTextRequest]: { gaEventName: null },
|
|
103
107
|
[EnviveMetricsEventName.WidgetTextResponse]: { gaEventName: null },
|
|
@@ -106,4 +110,4 @@ const GA4_EVENT_SCHEMA = {
|
|
|
106
110
|
|
|
107
111
|
//#endregion
|
|
108
112
|
export { GA4_EVENT_SCHEMA };
|
|
109
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
113
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ga4EventSchema.js","names":["WIDGET_INTERACTION_DATA_PROJECTIONS: WidgetInteractionDataProjection","GA4_EVENT_SCHEMA: Record<EnviveMetricsEventName, GA4EventSchemaEntry>"],"sources":["../../../src/services/ga4ProjectionService/ga4EventSchema.ts"],"sourcesContent":["/**\n * GA4 Event Schema — single source of truth for Envive → GA4 projection.\n *\n * Every EnviveMetricsEventName MUST have an entry here. The Record type enforces\n * this at compile time — adding a new enum member without a schema entry will\n * cause a TypeScript error.\n *\n * ## How to update this file\n *\n * When adding or modifying event properties, decide whether they belong in `allowedFields` using these criteria:\n * - **Low cardinality** — avoid unbounded values (free text, IDs, URLs, timestamps).\n * - **Non-sensitive** — no PII, user input text, or internal implementation details.\n * - **Flat scalar** — GA4 custom dimensions are flat strings/numbers/booleans.\n *   Nested objects should NOT be added; extract specific sub-fields instead.\n * - **Snake_case key** — must already be dot-notation snake_case (e.g. `chat.user_typed`).\n * - **Analytically useful** — the field should enable meaningful filtering/segmentation\n *   for merchants in their GA4 dashboards.\n *\n * If excluded from `allowedFields`, the projection layer will automatically drop it.\n *\n * IF IN DOUBT, DON'T HESITATE TO REACH OUT TO THE ANALYTICS TEAM!!!\n */\nimport { EnviveMetricsEventName } from '../amplitudeService/eventNames';\n\n/**\n * Defines which fields to extract from `trigger.widget_interaction_data`\n * for a given `trigger.widget_interaction` value.\n *\n * Key: the value of `trigger.widget_interaction` (e.g. \"widget_collapsed\")\n * Value: mapping of { ga4FlatKey: sourceFieldInWidgetInteractionData }\n */\nexport type WidgetInteractionDataProjection = Record<string, Record<string, string>>;\n\nexport interface GA4ProjectedEventConfig {\n  gaEventName: string;\n  allowedFields: readonly string[];\n  /**\n   * When present, used instead of `allowedFields` to both filter and rename fields.\n   * Key: the desired GA4 parameter name. Value: the source dot-notation key in eventProps.\n   */\n  fieldProjections?: Record<string, string>;\n  widgetInteractionDataProjections?: WidgetInteractionDataProjection;\n}\n\nexport interface GA4ExcludedEventConfig {\n  gaEventName: null;\n}\n\nexport type GA4EventSchemaEntry = GA4ProjectedEventConfig | GA4ExcludedEventConfig;\n\nconst WIDGET_INTERACTION_DATA_PROJECTIONS: WidgetInteractionDataProjection = {\n  widget_collapsed: { interaction_collapse_source: 'widget_collapsed.collapse_source' },\n  product_card_clicked: { interaction_product_id: 'product_card_clicked.product_id' },\n  suggestion_scrolled: { interaction_suggestion_id: 'suggestion_scrolled.suggestion_id' },\n};\n\nexport const GA4_EVENT_SCHEMA: Record<EnviveMetricsEventName, GA4EventSchemaEntry> = {\n  [EnviveMetricsEventName.WidgetRendered]: {\n    gaEventName: 'envive_widget_rendered',\n    allowedFields: [],\n    fieldProjections: {\n      page_type: 'context.page_type',\n      page_id: 'context.page_id',\n      surface: 'context.surface',\n      widget: 'trigger.widget',\n      widget_role: 'trigger.widget_role',\n    },\n  },\n\n  [EnviveMetricsEventName.WidgetInteraction]: {\n    gaEventName: 'envive_widget_interaction',\n    allowedFields: [],\n    fieldProjections: {\n      page_type: 'context.page_type',\n      page_id: 'context.page_id',\n      surface: 'context.surface',\n      widget: 'trigger.widget',\n      interaction_type: 'trigger.widget_interaction',\n      interaction_class: 'trigger.interaction_class',\n      widget_role: 'trigger.widget_role',\n    },\n    widgetInteractionDataProjections: WIDGET_INTERACTION_DATA_PROJECTIONS,\n  },\n\n  [EnviveMetricsEventName.ChatRequest]: {\n    gaEventName: 'envive_chat_request',\n    allowedFields: [\n      'page_type',\n      'page_id',\n      'trigger.widget',\n      'chat.request_type',\n      // 'chat.request_text', // This is high cardinality and potentially PII but we might want to include it later\n      'chat.user_typed',\n      'chat.suggestion_id', // not currently implemented\n      'chat.suggestion_category', // not currently implemented\n      'chat.suggestion_created_at', // not currently implemented\n      'chat.suggestion_is_answer', // not currently implemented\n      'chat.form_type', // not currently implemented\n    ],\n  },\n\n  [EnviveMetricsEventName.ChatResponse]: {\n    gaEventName: 'envive_chat_response',\n    allowedFields: [\n      'page_type',\n      'page_id',\n      'trigger.widget', // not currently implemented (maybe it should'nt be???)\n      'chat.user_typed',\n      'chat.response_time_ms',\n      'chat.product_cards_returned',\n      'chat.product_ids_returned',\n      'chat.review_cards_returned',\n    ],\n  },\n\n  [EnviveMetricsEventName.EnviveInitialized]: {\n    gaEventName: 'envive_initialized',\n    allowedFields: [\n      'environment.sales_agent_enabled',\n      'environment.envive_enabled',\n      // 'environment.search_enabled', // For now, search is not implemented. We will add it later.\n      'performance.start_time_ms',\n      'performance.initialize_time_ms',\n    ],\n  },\n\n  // This event is not currently implemented\n  [EnviveMetricsEventName.PageViewed]: {\n    gaEventName: 'envive_page_context_evaluated',\n    allowedFields: [],\n    fieldProjections: {\n      page_type: 'context.page_type',\n      page_id: 'context.page_id',\n      context_supported: 'context.supported',\n      context_ready: 'context.ready',\n      context_page_variant_id: 'context.page_variant_id',\n      environment_envive_enabled: 'environment.envive_enabled',\n    },\n  },\n\n  [EnviveMetricsEventName.WidgetTextRequest]: {\n    gaEventName: null,\n  },\n\n  [EnviveMetricsEventName.WidgetTextResponse]: {\n    gaEventName: null,\n  },\n\n  [EnviveMetricsEventName.WidgetTextClicked]: {\n    gaEventName: null,\n  },\n};\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;AAkDA,MAAMA,sCAAuE;CAC3E,kBAAkB,EAAE,6BAA6B,oCAAoC;CACrF,sBAAsB,EAAE,wBAAwB,mCAAmC;CACnF,qBAAqB,EAAE,2BAA2B,qCAAqC;CACxF;AAED,MAAaC,mBAAwE;EAClF,uBAAuB,iBAAiB;EACvC,aAAa;EACb,eAAe,EAAE;EACjB,kBAAkB;GAChB,WAAW;GACX,SAAS;GACT,SAAS;GACT,QAAQ;GACR,aAAa;GACd;EACF;EAEA,uBAAuB,oBAAoB;EAC1C,aAAa;EACb,eAAe,EAAE;EACjB,kBAAkB;GAChB,WAAW;GACX,SAAS;GACT,SAAS;GACT,QAAQ;GACR,kBAAkB;GAClB,mBAAmB;GACnB,aAAa;GACd;EACD,kCAAkC;EACnC;EAEA,uBAAuB,cAAc;EACpC,aAAa;EACb,eAAe;GACb;GACA;GACA;GACA;GAEA;GACA;GACA;GACA;GACA;GACA;GACD;EACF;EAEA,uBAAuB,eAAe;EACrC,aAAa;EACb,eAAe;GACb;GACA;GACA;GACA;GACA;GACA;GACA;GACA;GACD;EACF;EAEA,uBAAuB,oBAAoB;EAC1C,aAAa;EACb,eAAe;GACb;GACA;GAEA;GACA;GACD;EACF;EAGA,uBAAuB,aAAa;EACnC,aAAa;EACb,eAAe,EAAE;EACjB,kBAAkB;GAChB,WAAW;GACX,SAAS;GACT,mBAAmB;GACnB,eAAe;GACf,yBAAyB;GACzB,4BAA4B;GAC7B;EACF;EAEA,uBAAuB,oBAAoB,EAC1C,aAAa,MACd;EAEA,uBAAuB,qBAAqB,EAC3C,aAAa,MACd;EAEA,uBAAuB,oBAAoB,EAC1C,aAAa,MACd;CACF"}
|
|
@@ -8,18 +8,39 @@ const filterToSchema = (eventProps, allowedFields) => {
|
|
|
8
8
|
for (const field of allowedFields) if (field in eventProps) result[field] = eventProps[field];
|
|
9
9
|
return result;
|
|
10
10
|
};
|
|
11
|
+
const flattenOneLevel = (obj) => {
|
|
12
|
+
const result = {};
|
|
13
|
+
for (const [key, value] of Object.entries(obj)) if (!key.includes(".") && value !== null && typeof value === "object" && !Array.isArray(value)) for (const [subKey, subValue] of Object.entries(value)) result[`${key}.${subKey}`] = subValue;
|
|
14
|
+
else result[key] = value;
|
|
15
|
+
return result;
|
|
16
|
+
};
|
|
17
|
+
const filterWithProjections = (eventProps, projections) => {
|
|
18
|
+
const result = {};
|
|
19
|
+
for (const [ga4Key, sourceKey] of Object.entries(projections)) if (sourceKey in eventProps) result[ga4Key] = eventProps[sourceKey];
|
|
20
|
+
return result;
|
|
21
|
+
};
|
|
11
22
|
const flattenDotKeys = (obj) => {
|
|
12
23
|
const result = {};
|
|
13
24
|
for (const [key, value] of Object.entries(obj)) result[key.replace(/\./g, "_")] = value;
|
|
14
25
|
return result;
|
|
15
26
|
};
|
|
16
27
|
const sanitizePageId = (filtered) => {
|
|
17
|
-
const pageType = filtered["context.page_type"];
|
|
28
|
+
const pageType = filtered["page_type"] ?? filtered["context.page_type"];
|
|
18
29
|
if (pageType === "pdp" || pageType === "plp") return filtered;
|
|
19
30
|
const rest = { ...filtered };
|
|
31
|
+
delete rest["page_id"];
|
|
20
32
|
delete rest["context.page_id"];
|
|
21
33
|
return rest;
|
|
22
34
|
};
|
|
35
|
+
const getNestedValue = (obj, path) => {
|
|
36
|
+
const parts = path.split(".");
|
|
37
|
+
let current = obj;
|
|
38
|
+
for (const part of parts) {
|
|
39
|
+
if (current === null || typeof current !== "object") return void 0;
|
|
40
|
+
current = current[part];
|
|
41
|
+
}
|
|
42
|
+
return current;
|
|
43
|
+
};
|
|
23
44
|
const projectWidgetInteractionData = (eventProps, config) => {
|
|
24
45
|
if (!config.widgetInteractionDataProjections) return {};
|
|
25
46
|
const interaction = eventProps["trigger.widget_interaction"];
|
|
@@ -30,7 +51,10 @@ const projectWidgetInteractionData = (eventProps, config) => {
|
|
|
30
51
|
if (interactionData === null || interactionData === void 0 || typeof interactionData !== "object") return {};
|
|
31
52
|
const data = interactionData;
|
|
32
53
|
const result = {};
|
|
33
|
-
for (const [gaKey,
|
|
54
|
+
for (const [gaKey, sourcePath] of Object.entries(projectionMap)) {
|
|
55
|
+
const value = getNestedValue(data, sourcePath);
|
|
56
|
+
if (value !== void 0) result[gaKey] = value;
|
|
57
|
+
}
|
|
34
58
|
return result;
|
|
35
59
|
};
|
|
36
60
|
const truncateString = (value) => {
|
|
@@ -50,8 +74,10 @@ const projectToGA4 = (eventName, eventProps) => {
|
|
|
50
74
|
const schemaEntry = require_ga4EventSchema.GA4_EVENT_SCHEMA[eventName];
|
|
51
75
|
if (schemaEntry.gaEventName === null) return;
|
|
52
76
|
const config = schemaEntry;
|
|
53
|
-
const props = eventProps ?? {};
|
|
54
|
-
let filtered
|
|
77
|
+
const props = flattenOneLevel(eventProps ?? {});
|
|
78
|
+
let filtered;
|
|
79
|
+
if (config.fieldProjections) filtered = filterWithProjections(props, config.fieldProjections);
|
|
80
|
+
else filtered = filterToSchema(props, config.allowedFields);
|
|
55
81
|
filtered = sanitizePageId(filtered);
|
|
56
82
|
const interactionFields = projectWidgetInteractionData(props, config);
|
|
57
83
|
const truncatedParams = truncateValues({
|
|
@@ -69,4 +95,4 @@ const projectToGA4 = (eventName, eventProps) => {
|
|
|
69
95
|
|
|
70
96
|
//#endregion
|
|
71
97
|
exports.projectToGA4 = projectToGA4;
|
|
72
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2E0UHJvamVjdGlvblNlcnZpY2UuY2pzIiwibmFtZXMiOlsiTG9nZ2VyIiwicmVzdWx0OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiIsIkdBNF9FVkVOVF9TQ0hFTUEiXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc2VydmljZXMvZ2E0UHJvamVjdGlvblNlcnZpY2UvZ2E0UHJvamVjdGlvblNlcnZpY2UudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IExvZ2dlciBmcm9tICdzcmMvYXBwbGljYXRpb24vbG9nZ2luZy9sb2dnZXInO1xuaW1wb3J0IHsgRW52aXZlTWV0cmljc0V2ZW50TmFtZSB9IGZyb20gJy4uL2FtcGxpdHVkZVNlcnZpY2UvZXZlbnROYW1lcyc7XG5pbXBvcnQgeyBHQTRQcm9qZWN0ZWRFdmVudENvbmZpZywgR0E0X0VWRU5UX1NDSEVNQSB9IGZyb20gJy4vZ2E0RXZlbnRTY2hlbWEnO1xuXG5jb25zdCBsb2dnZXIgPSBuZXcgTG9nZ2VyKCdnYTRQcm9qZWN0aW9uU2VydmljZScpO1xuXG5jb25zdCBmaWx0ZXJUb1NjaGVtYSA9IChcbiAgZXZlbnRQcm9wczogUmVjb3JkPHN0cmluZywgdW5rbm93bj4sXG4gIGFsbG93ZWRGaWVsZHM6IHJlYWRvbmx5IHN0cmluZ1tdLFxuKTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPT4ge1xuICBjb25zdCByZXN1bHQ6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge307XG4gIGZvciAoY29uc3QgZmllbGQgb2YgYWxsb3dlZEZpZWxkcykge1xuICAgIGlmIChmaWVsZCBpbiBldmVudFByb3BzKSB7XG4gICAgICByZXN1bHRbZmllbGRdID0gZXZlbnRQcm9wc1tmaWVsZF07XG4gICAgfVxuICB9XG4gIHJldHVybiByZXN1bHQ7XG59O1xuXG4vLyBcImNvbnRleHQucGFnZV90eXBlXCIg4oaSIFwiY29udGV4dF9wYWdlX3R5cGVcIlxuY29uc3QgZmxhdHRlbkRvdEtleXMgPSAob2JqOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPik6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgY29uc3QgcmVzdWx0OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IHt9O1xuICBmb3IgKGNvbnN0IFtrZXksIHZhbHVlXSBvZiBPYmplY3QuZW50cmllcyhvYmopKSB7XG4gICAgcmVzdWx0W2tleS5yZXBsYWNlKC9cXC4vZywgJ18nKV0gPSB2YWx1ZTtcbiAgfVxuICByZXR1cm4gcmVzdWx0O1xufTtcblxuLy8gT21pdCBjb250ZXh0LnBhZ2VfaWQgZm9yIG5vbi1wZHAvcGxwIHBhZ2UgdHlwZXMuIFRoZSBjdXJyZW50IGltcGxlbWVudGF0aW9uIGZvciBjb250ZXh0LnBhZ2VfaWQgaXM6XG4vLyBQRFA6IHByb2R1Y3RfaWRcbi8vIFBMUDogcGxwX2lkXG4vLyBTZWFyY2g6IHNlYXJjaCBxdWVyeVxuLy8gT3RoZXI6IHBhZ2UgdXJsXG4vLyBXZSB3YW50IHRvIG9taXQgYWxsIGJ1dCBwZHAgYW5kIHBscCBwYWdlIHR5cGVzIHRvIHByb3ZpZGUgYSBjbGVhciwgY29uc2lzdGVudCBpbnRlcmZhY2UgZm9yIG1lcmNoYW50cy5cbmNvbnN0IHNhbml0aXplUGFnZUlkID0gKGZpbHRlcmVkOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPik6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgY29uc3QgcGFnZVR5cGUgPSBmaWx0ZXJlZFsnY29udGV4dC5wYWdlX3R5cGUnXTtcbiAgaWYgKHBhZ2VUeXBlID09PSAncGRwJyB8fCBwYWdlVHlwZSA9PT0gJ3BscCcpIHtcbiAgICByZXR1cm4gZmlsdGVyZWQ7XG4gIH1cbiAgY29uc3QgcmVzdCA9IHsgLi4uZmlsdGVyZWQgfTtcbiAgZGVsZXRlIHJlc3RbJ2NvbnRleHQucGFnZV9pZCddO1xuICByZXR1cm4gcmVzdDtcbn07XG5cbi8vIEV4dHJhY3Qgd2hpdGVsaXN0ZWQgc3ViLWZpZWxkcyBmcm9tIHRyaWdnZXIud2lkZ2V0X2ludGVyYWN0aW9uX2RhdGFcbmNvbnN0IHByb2plY3RXaWRnZXRJbnRlcmFjdGlvbkRhdGEgPSAoXG4gIGV2ZW50UHJvcHM6IFJlY29yZDxzdHJpbmcsIHVua25vd24+LFxuICBjb25maWc6IEdBNFByb2plY3RlZEV2ZW50Q29uZmlnLFxuKTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPT4ge1xuICBpZiAoIWNvbmZpZy53aWRnZXRJbnRlcmFjdGlvbkRhdGFQcm9qZWN0aW9ucykge1xuICAgIHJldHVybiB7fTtcbiAgfVxuXG4gIGNvbnN0IGludGVyYWN0aW9uID0gZXZlbnRQcm9wc1sndHJpZ2dlci53aWRnZXRfaW50ZXJhY3Rpb24nXTtcbiAgaWYgKHR5cGVvZiBpbnRlcmFjdGlvbiAhPT0gJ3N0cmluZycpIHtcbiAgICByZXR1cm4ge307XG4gIH1cblxuICBjb25zdCBwcm9qZWN0aW9uTWFwID0gY29uZmlnLndpZGdldEludGVyYWN0aW9uRGF0YVByb2plY3Rpb25zW2ludGVyYWN0aW9uXTtcbiAgaWYgKCFwcm9qZWN0aW9uTWFwKSB7XG4gICAgcmV0dXJuIHt9O1xuICB9XG5cbiAgY29uc3QgaW50ZXJhY3Rpb25EYXRhID0gZXZlbnRQcm9wc1sndHJpZ2dlci53aWRnZXRfaW50ZXJhY3Rpb25fZGF0YSddO1xuICBpZiAoXG4gICAgaW50ZXJhY3Rpb25EYXRhID09PSBudWxsIHx8XG4gICAgaW50ZXJhY3Rpb25EYXRhID09PSB1bmRlZmluZWQgfHxcbiAgICB0eXBlb2YgaW50ZXJhY3Rpb25EYXRhICE9PSAnb2JqZWN0J1xuICApIHtcbiAgICByZXR1cm4ge307XG4gIH1cblxuICBjb25zdCBkYXRhID0gaW50ZXJhY3Rpb25EYXRhIGFzIFJlY29yZDxzdHJpbmcsIHVua25vd24+O1xuICBjb25zdCByZXN1bHQ6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge307XG4gIGZvciAoY29uc3QgW2dhS2V5LCBzb3VyY2VGaWVsZF0gb2YgT2JqZWN0LmVudHJpZXMocHJvamVjdGlvbk1hcCkpIHtcbiAgICBpZiAoc291cmNlRmllbGQgaW4gZGF0YSkge1xuICAgICAgcmVzdWx0W2dhS2V5XSA9IGRhdGFbc291cmNlRmllbGRdO1xuICAgIH1cbiAgfVxuICByZXR1cm4gcmVzdWx0O1xufTtcblxuY29uc3QgdHJ1bmNhdGVTdHJpbmcgPSAodmFsdWU6IHVua25vd24pOiB1bmtub3duID0+IHtcbiAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gJ3N0cmluZycgJiYgdmFsdWUubGVuZ3RoID4gMTAwKSB7XG4gICAgcmV0dXJuIGAke3ZhbHVlLnN1YnN0cmluZygwLCA5Nyl9Li4uYDtcbiAgfVxuICByZXR1cm4gdmFsdWU7XG59O1xuXG5jb25zdCB0cnVuY2F0ZVZhbHVlcyA9IChvYmo6IFJlY29yZDxzdHJpbmcsIHVua25vd24+KTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPT4ge1xuICBjb25zdCByZXN1bHQ6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge307XG4gIGZvciAoY29uc3QgW2tleSwgdmFsdWVdIG9mIE9iamVjdC5lbnRyaWVzKG9iaikpIHtcbiAgICByZXN1bHRba2V5XSA9IHRydW5jYXRlU3RyaW5nKHZhbHVlKTtcbiAgfVxuICByZXR1cm4gcmVzdWx0O1xufTtcblxuY29uc3QgcHVzaFRvRGF0YUxheWVyID0gKGdhRXZlbnQ6IFJlY29yZDxzdHJpbmcsIHVua25vd24+KTogdm9pZCA9PiB7XG4gIGlmICh0eXBlb2Ygd2luZG93ICE9PSAndW5kZWZpbmVkJyAmJiB3aW5kb3cuZGF0YUxheWVyKSB7XG4gICAgd2luZG93LmRhdGFMYXllci5wdXNoKGdhRXZlbnQpO1xuICB9XG59O1xuXG5leHBvcnQgY29uc3QgcHJvamVjdFRvR0E0ID0gKFxuICBldmVudE5hbWU6IEVudml2ZU1ldHJpY3NFdmVudE5hbWUsXG4gIGV2ZW50UHJvcHM/OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPixcbik6IHZvaWQgPT4ge1xuICB0cnkge1xuICAgIGNvbnN0IHNjaGVtYUVudHJ5ID0gR0E0X0VWRU5UX1NDSEVNQVtldmVudE5hbWVdO1xuXG4gICAgaWYgKHNjaGVtYUVudHJ5LmdhRXZlbnROYW1lID09PSBudWxsKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgY29uc3QgY29uZmlnID0gc2NoZW1hRW50cnk7XG4gICAgY29uc3QgcHJvcHMgPSBldmVudFByb3BzID8/IHt9O1xuXG4gICAgbGV0IGZpbHRlcmVkID0gZmlsdGVyVG9TY2hlbWEocHJvcHMsIGNvbmZpZy5hbGxvd2VkRmllbGRzKTtcbiAgICBmaWx0ZXJlZCA9IHNhbml0aXplUGFnZUlkKGZpbHRlcmVkKTtcblxuICAgIGNvbnN0IGludGVyYWN0aW9uRmllbGRzID0gcHJvamVjdFdpZGdldEludGVyYWN0aW9uRGF0YShwcm9wcywgY29uZmlnKTtcblxuICAgIGNvbnN0IGZsYXRQYXJhbXMgPSB7XG4gICAgICAuLi5mbGF0dGVuRG90S2V5cyhmaWx0ZXJlZCksXG4gICAgICAuLi5pbnRlcmFjdGlvbkZpZWxkcyxcbiAgICB9O1xuXG4gICAgY29uc3QgdHJ1bmNhdGVkUGFyYW1zID0gdHJ1bmNhdGVWYWx1ZXMoZmxhdFBhcmFtcyk7XG5cbiAgICBwdXNoVG9EYXRhTGF5ZXIoe1xuICAgICAgZXZlbnQ6IGNvbmZpZy5nYUV2ZW50TmFtZSxcbiAgICAgIC4uLnRydW5jYXRlZFBhcmFtcyxcbiAgICB9KTtcbiAgfSBjYXRjaCAoZXJyKSB7XG4gICAgbG9nZ2VyLmxvZ0Vycm9yKCdFcnJvciBwcm9qZWN0aW5nIGV2ZW50IHRvIEdBNCcsIGVyciwge1xuICAgICAgZXZlbnROYW1lLFxuICAgIH0pO1xuICB9XG59O1xuIl0sIm1hcHBpbmdzIjoiOzs7O0FBSUEsTUFBTSxTQUFTLElBQUlBLHVCQUFPLHVCQUF1QjtBQUVqRCxNQUFNLGtCQUNKLFlBQ0Esa0JBQzRCO0NBQzVCLE1BQU1DLFNBQWtDLEVBQUU7QUFDMUMsTUFBSyxNQUFNLFNBQVMsY0FDbEIsS0FBSSxTQUFTLFdBQ1gsUUFBTyxTQUFTLFdBQVc7QUFHL0IsUUFBTzs7QUFJVCxNQUFNLGtCQUFrQixRQUEwRDtDQUNoRixNQUFNQSxTQUFrQyxFQUFFO0FBQzFDLE1BQUssTUFBTSxDQUFDLEtBQUssVUFBVSxPQUFPLFFBQVEsSUFBSSxDQUM1QyxRQUFPLElBQUksUUFBUSxPQUFPLElBQUksSUFBSTtBQUVwQyxRQUFPOztBQVNULE1BQU0sa0JBQWtCLGFBQStEO0NBQ3JGLE1BQU0sV0FBVyxTQUFTO0FBQzFCLEtBQUksYUFBYSxTQUFTLGFBQWEsTUFDckMsUUFBTztDQUVULE1BQU0sT0FBTyxFQUFFLEdBQUcsVUFBVTtBQUM1QixRQUFPLEtBQUs7QUFDWixRQUFPOztBQUlULE1BQU0sZ0NBQ0osWUFDQSxXQUM0QjtBQUM1QixLQUFJLENBQUMsT0FBTyxpQ0FDVixRQUFPLEVBQUU7Q0FHWCxNQUFNLGNBQWMsV0FBVztBQUMvQixLQUFJLE9BQU8sZ0JBQWdCLFNBQ3pCLFFBQU8sRUFBRTtDQUdYLE1BQU0sZ0JBQWdCLE9BQU8saUNBQWlDO0FBQzlELEtBQUksQ0FBQyxjQUNILFFBQU8sRUFBRTtDQUdYLE1BQU0sa0JBQWtCLFdBQVc7QUFDbkMsS0FDRSxvQkFBb0IsUUFDcEIsb0JBQW9CLFVBQ3BCLE9BQU8sb0JBQW9CLFNBRTNCLFFBQU8sRUFBRTtDQUdYLE1BQU0sT0FBTztDQUNiLE1BQU1BLFNBQWtDLEVBQUU7QUFDMUMsTUFBSyxNQUFNLENBQUMsT0FBTyxnQkFBZ0IsT0FBTyxRQUFRLGNBQWMsQ0FDOUQsS0FBSSxlQUFlLEtBQ2pCLFFBQU8sU0FBUyxLQUFLO0FBR3pCLFFBQU87O0FBR1QsTUFBTSxrQkFBa0IsVUFBNEI7QUFDbEQsS0FBSSxPQUFPLFVBQVUsWUFBWSxNQUFNLFNBQVMsSUFDOUMsUUFBTyxHQUFHLE1BQU0sVUFBVSxHQUFHLEdBQUcsQ0FBQztBQUVuQyxRQUFPOztBQUdULE1BQU0sa0JBQWtCLFFBQTBEO0NBQ2hGLE1BQU1BLFNBQWtDLEVBQUU7QUFDMUMsTUFBSyxNQUFNLENBQUMsS0FBSyxVQUFVLE9BQU8sUUFBUSxJQUFJLENBQzVDLFFBQU8sT0FBTyxlQUFlLE1BQU07QUFFckMsUUFBTzs7QUFHVCxNQUFNLG1CQUFtQixZQUEyQztBQUNsRSxLQUFJLE9BQU8sV0FBVyxlQUFlLE9BQU8sVUFDMUMsUUFBTyxVQUFVLEtBQUssUUFBUTs7QUFJbEMsTUFBYSxnQkFDWCxXQUNBLGVBQ1M7QUFDVCxLQUFJO0VBQ0YsTUFBTSxjQUFjQyx3Q0FBaUI7QUFFckMsTUFBSSxZQUFZLGdCQUFnQixLQUM5QjtFQUdGLE1BQU0sU0FBUztFQUNmLE1BQU0sUUFBUSxjQUFjLEVBQUU7RUFFOUIsSUFBSSxXQUFXLGVBQWUsT0FBTyxPQUFPLGNBQWM7QUFDMUQsYUFBVyxlQUFlLFNBQVM7RUFFbkMsTUFBTSxvQkFBb0IsNkJBQTZCLE9BQU8sT0FBTztFQU9yRSxNQUFNLGtCQUFrQixlQUxMO0dBQ2pCLEdBQUcsZUFBZSxTQUFTO0dBQzNCLEdBQUc7R0FDSixDQUVpRDtBQUVsRCxrQkFBZ0I7R0FDZCxPQUFPLE9BQU87R0FDZCxHQUFHO0dBQ0osQ0FBQztVQUNLLEtBQUs7QUFDWixTQUFPLFNBQVMsaUNBQWlDLEtBQUssRUFDcEQsV0FDRCxDQUFDIn0=
|
|
98
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ga4ProjectionService.cjs","names":["Logger","result: Record<string, unknown>","current: unknown","GA4_EVENT_SCHEMA","filtered: Record<string, unknown>"],"sources":["../../../src/services/ga4ProjectionService/ga4ProjectionService.ts"],"sourcesContent":["import Logger from 'src/application/logging/logger';\nimport { EnviveMetricsEventName } from '../amplitudeService/eventNames';\nimport { GA4ProjectedEventConfig, GA4_EVENT_SCHEMA } from './ga4EventSchema';\n\nconst logger = new Logger('ga4ProjectionService');\n\nconst filterToSchema = (\n  eventProps: Record<string, unknown>,\n  allowedFields: readonly string[],\n): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const field of allowedFields) {\n    if (field in eventProps) {\n      result[field] = eventProps[field];\n    }\n  }\n  return result;\n};\n\n// Flatten only true top-level nested objects (keys with no dots) one level deep.\n// Keys that already contain dots are left intact so that e.g.\n// `trigger.widget_interaction_data: { ... }` is preserved for downstream extraction.\nconst flattenOneLevel = (obj: Record<string, unknown>): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(obj)) {\n    if (\n      !key.includes('.') &&\n      value !== null &&\n      typeof value === 'object' &&\n      !Array.isArray(value)\n    ) {\n      for (const [subKey, subValue] of Object.entries(value as Record<string, unknown>)) {\n        result[`${key}.${subKey}`] = subValue;\n      }\n    } else {\n      result[key] = value;\n    }\n  }\n  return result;\n};\n\n// Apply a GA4 projection map { ga4Key: sourceKey } to extract and rename fields.\nconst filterWithProjections = (\n  eventProps: Record<string, unknown>,\n  projections: Record<string, string>,\n): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [ga4Key, sourceKey] of Object.entries(projections)) {\n    if (sourceKey in eventProps) {\n      result[ga4Key] = eventProps[sourceKey];\n    }\n  }\n  return result;\n};\n\n// \"context.page_type\" → \"context_page_type\"\nconst flattenDotKeys = (obj: Record<string, unknown>): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(obj)) {\n    result[key.replace(/\\./g, '_')] = value;\n  }\n  return result;\n};\n\n// Omit context.page_id for non-pdp/plp page types. The current implementation for context.page_id is:\n// PDP: product_id\n// PLP: plp_id\n// Search: search query\n// Other: page url\n// We want to omit all but pdp and plp page types to provide a clear, consistent interface for merchants.\n// Handles both dot-notation keys (allowedFields path) and renamed GA4 keys (fieldProjections path).\nconst sanitizePageId = (filtered: Record<string, unknown>): Record<string, unknown> => {\n  const pageType = filtered['page_type'] ?? filtered['context.page_type'];\n  if (pageType === 'pdp' || pageType === 'plp') {\n    return filtered;\n  }\n  const rest = { ...filtered };\n  delete rest['page_id'];\n  delete rest['context.page_id'];\n  return rest;\n};\n\nconst getNestedValue = (obj: Record<string, unknown>, path: string): unknown => {\n  const parts = path.split('.');\n  let current: unknown = obj;\n  for (const part of parts) {\n    if (current === null || typeof current !== 'object') return undefined;\n    current = (current as Record<string, unknown>)[part];\n  }\n  return current;\n};\n\n// Extract whitelisted sub-fields from trigger.widget_interaction_data\nconst projectWidgetInteractionData = (\n  eventProps: Record<string, unknown>,\n  config: GA4ProjectedEventConfig,\n): Record<string, unknown> => {\n  if (!config.widgetInteractionDataProjections) {\n    return {};\n  }\n\n  const interaction = eventProps['trigger.widget_interaction'];\n  if (typeof interaction !== 'string') {\n    return {};\n  }\n\n  const projectionMap = config.widgetInteractionDataProjections[interaction];\n  if (!projectionMap) {\n    return {};\n  }\n\n  const interactionData = eventProps['trigger.widget_interaction_data'];\n  if (\n    interactionData === null ||\n    interactionData === undefined ||\n    typeof interactionData !== 'object'\n  ) {\n    return {};\n  }\n\n  const data = interactionData as Record<string, unknown>;\n  const result: Record<string, unknown> = {};\n  for (const [gaKey, sourcePath] of Object.entries(projectionMap)) {\n    const value = getNestedValue(data, sourcePath);\n    if (value !== undefined) {\n      result[gaKey] = value;\n    }\n  }\n  return result;\n};\n\nconst truncateString = (value: unknown): unknown => {\n  if (typeof value === 'string' && value.length > 100) {\n    return `${value.substring(0, 97)}...`;\n  }\n  return value;\n};\n\nconst truncateValues = (obj: Record<string, unknown>): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(obj)) {\n    result[key] = truncateString(value);\n  }\n  return result;\n};\n\nconst pushToDataLayer = (gaEvent: Record<string, unknown>): void => {\n  if (typeof window !== 'undefined' && window.dataLayer) {\n    window.dataLayer.push(gaEvent);\n  }\n};\n\nexport const projectToGA4 = (\n  eventName: EnviveMetricsEventName,\n  eventProps?: Record<string, unknown>,\n): void => {\n  try {\n    const schemaEntry = GA4_EVENT_SCHEMA[eventName];\n\n    if (schemaEntry.gaEventName === null) {\n      return;\n    }\n\n    const config = schemaEntry;\n    const props = flattenOneLevel(eventProps ?? {});\n\n    let filtered: Record<string, unknown>;\n    if (config.fieldProjections) {\n      filtered = filterWithProjections(props, config.fieldProjections);\n    } else {\n      filtered = filterToSchema(props, config.allowedFields);\n    }\n    filtered = sanitizePageId(filtered);\n\n    const interactionFields = projectWidgetInteractionData(props, config);\n\n    const flatParams = {\n      ...flattenDotKeys(filtered),\n      ...interactionFields,\n    };\n\n    const truncatedParams = truncateValues(flatParams);\n\n    pushToDataLayer({\n      event: config.gaEventName,\n      ...truncatedParams,\n    });\n  } catch (err) {\n    logger.logError('Error projecting event to GA4', err, {\n      eventName,\n    });\n  }\n};\n"],"mappings":"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"}
|
|
@@ -8,18 +8,39 @@ const filterToSchema = (eventProps, allowedFields) => {
|
|
|
8
8
|
for (const field of allowedFields) if (field in eventProps) result[field] = eventProps[field];
|
|
9
9
|
return result;
|
|
10
10
|
};
|
|
11
|
+
const flattenOneLevel = (obj) => {
|
|
12
|
+
const result = {};
|
|
13
|
+
for (const [key, value] of Object.entries(obj)) if (!key.includes(".") && value !== null && typeof value === "object" && !Array.isArray(value)) for (const [subKey, subValue] of Object.entries(value)) result[`${key}.${subKey}`] = subValue;
|
|
14
|
+
else result[key] = value;
|
|
15
|
+
return result;
|
|
16
|
+
};
|
|
17
|
+
const filterWithProjections = (eventProps, projections) => {
|
|
18
|
+
const result = {};
|
|
19
|
+
for (const [ga4Key, sourceKey] of Object.entries(projections)) if (sourceKey in eventProps) result[ga4Key] = eventProps[sourceKey];
|
|
20
|
+
return result;
|
|
21
|
+
};
|
|
11
22
|
const flattenDotKeys = (obj) => {
|
|
12
23
|
const result = {};
|
|
13
24
|
for (const [key, value] of Object.entries(obj)) result[key.replace(/\./g, "_")] = value;
|
|
14
25
|
return result;
|
|
15
26
|
};
|
|
16
27
|
const sanitizePageId = (filtered) => {
|
|
17
|
-
const pageType = filtered["context.page_type"];
|
|
28
|
+
const pageType = filtered["page_type"] ?? filtered["context.page_type"];
|
|
18
29
|
if (pageType === "pdp" || pageType === "plp") return filtered;
|
|
19
30
|
const rest = { ...filtered };
|
|
31
|
+
delete rest["page_id"];
|
|
20
32
|
delete rest["context.page_id"];
|
|
21
33
|
return rest;
|
|
22
34
|
};
|
|
35
|
+
const getNestedValue = (obj, path) => {
|
|
36
|
+
const parts = path.split(".");
|
|
37
|
+
let current = obj;
|
|
38
|
+
for (const part of parts) {
|
|
39
|
+
if (current === null || typeof current !== "object") return void 0;
|
|
40
|
+
current = current[part];
|
|
41
|
+
}
|
|
42
|
+
return current;
|
|
43
|
+
};
|
|
23
44
|
const projectWidgetInteractionData = (eventProps, config) => {
|
|
24
45
|
if (!config.widgetInteractionDataProjections) return {};
|
|
25
46
|
const interaction = eventProps["trigger.widget_interaction"];
|
|
@@ -30,7 +51,10 @@ const projectWidgetInteractionData = (eventProps, config) => {
|
|
|
30
51
|
if (interactionData === null || interactionData === void 0 || typeof interactionData !== "object") return {};
|
|
31
52
|
const data = interactionData;
|
|
32
53
|
const result = {};
|
|
33
|
-
for (const [gaKey,
|
|
54
|
+
for (const [gaKey, sourcePath] of Object.entries(projectionMap)) {
|
|
55
|
+
const value = getNestedValue(data, sourcePath);
|
|
56
|
+
if (value !== void 0) result[gaKey] = value;
|
|
57
|
+
}
|
|
34
58
|
return result;
|
|
35
59
|
};
|
|
36
60
|
const truncateString = (value) => {
|
|
@@ -50,8 +74,10 @@ const projectToGA4 = (eventName, eventProps) => {
|
|
|
50
74
|
const schemaEntry = GA4_EVENT_SCHEMA[eventName];
|
|
51
75
|
if (schemaEntry.gaEventName === null) return;
|
|
52
76
|
const config = schemaEntry;
|
|
53
|
-
const props = eventProps ?? {};
|
|
54
|
-
let filtered
|
|
77
|
+
const props = flattenOneLevel(eventProps ?? {});
|
|
78
|
+
let filtered;
|
|
79
|
+
if (config.fieldProjections) filtered = filterWithProjections(props, config.fieldProjections);
|
|
80
|
+
else filtered = filterToSchema(props, config.allowedFields);
|
|
55
81
|
filtered = sanitizePageId(filtered);
|
|
56
82
|
const interactionFields = projectWidgetInteractionData(props, config);
|
|
57
83
|
const truncatedParams = truncateValues({
|
|
@@ -69,4 +95,4 @@ const projectToGA4 = (eventName, eventProps) => {
|
|
|
69
95
|
|
|
70
96
|
//#endregion
|
|
71
97
|
export { projectToGA4 };
|
|
72
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2E0UHJvamVjdGlvblNlcnZpY2UuanMiLCJuYW1lcyI6WyJMb2dnZXIiLCJyZXN1bHQ6IFJlY29yZDxzdHJpbmcsIHVua25vd24+Il0sInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3NlcnZpY2VzL2dhNFByb2plY3Rpb25TZXJ2aWNlL2dhNFByb2plY3Rpb25TZXJ2aWNlLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBMb2dnZXIgZnJvbSAnc3JjL2FwcGxpY2F0aW9uL2xvZ2dpbmcvbG9nZ2VyJztcbmltcG9ydCB7IEVudml2ZU1ldHJpY3NFdmVudE5hbWUgfSBmcm9tICcuLi9hbXBsaXR1ZGVTZXJ2aWNlL2V2ZW50TmFtZXMnO1xuaW1wb3J0IHsgR0E0UHJvamVjdGVkRXZlbnRDb25maWcsIEdBNF9FVkVOVF9TQ0hFTUEgfSBmcm9tICcuL2dhNEV2ZW50U2NoZW1hJztcblxuY29uc3QgbG9nZ2VyID0gbmV3IExvZ2dlcignZ2E0UHJvamVjdGlvblNlcnZpY2UnKTtcblxuY29uc3QgZmlsdGVyVG9TY2hlbWEgPSAoXG4gIGV2ZW50UHJvcHM6IFJlY29yZDxzdHJpbmcsIHVua25vd24+LFxuICBhbGxvd2VkRmllbGRzOiByZWFkb25seSBzdHJpbmdbXSxcbik6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgY29uc3QgcmVzdWx0OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IHt9O1xuICBmb3IgKGNvbnN0IGZpZWxkIG9mIGFsbG93ZWRGaWVsZHMpIHtcbiAgICBpZiAoZmllbGQgaW4gZXZlbnRQcm9wcykge1xuICAgICAgcmVzdWx0W2ZpZWxkXSA9IGV2ZW50UHJvcHNbZmllbGRdO1xuICAgIH1cbiAgfVxuICByZXR1cm4gcmVzdWx0O1xufTtcblxuLy8gXCJjb250ZXh0LnBhZ2VfdHlwZVwiIOKGkiBcImNvbnRleHRfcGFnZV90eXBlXCJcbmNvbnN0IGZsYXR0ZW5Eb3RLZXlzID0gKG9iajogUmVjb3JkPHN0cmluZywgdW5rbm93bj4pOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9PiB7XG4gIGNvbnN0IHJlc3VsdDogUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPSB7fTtcbiAgZm9yIChjb25zdCBba2V5LCB2YWx1ZV0gb2YgT2JqZWN0LmVudHJpZXMob2JqKSkge1xuICAgIHJlc3VsdFtrZXkucmVwbGFjZSgvXFwuL2csICdfJyldID0gdmFsdWU7XG4gIH1cbiAgcmV0dXJuIHJlc3VsdDtcbn07XG5cbi8vIE9taXQgY29udGV4dC5wYWdlX2lkIGZvciBub24tcGRwL3BscCBwYWdlIHR5cGVzLiBUaGUgY3VycmVudCBpbXBsZW1lbnRhdGlvbiBmb3IgY29udGV4dC5wYWdlX2lkIGlzOlxuLy8gUERQOiBwcm9kdWN0X2lkXG4vLyBQTFA6IHBscF9pZFxuLy8gU2VhcmNoOiBzZWFyY2ggcXVlcnlcbi8vIE90aGVyOiBwYWdlIHVybFxuLy8gV2Ugd2FudCB0byBvbWl0IGFsbCBidXQgcGRwIGFuZCBwbHAgcGFnZSB0eXBlcyB0byBwcm92aWRlIGEgY2xlYXIsIGNvbnNpc3RlbnQgaW50ZXJmYWNlIGZvciBtZXJjaGFudHMuXG5jb25zdCBzYW5pdGl6ZVBhZ2VJZCA9IChmaWx0ZXJlZDogUmVjb3JkPHN0cmluZywgdW5rbm93bj4pOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9PiB7XG4gIGNvbnN0IHBhZ2VUeXBlID0gZmlsdGVyZWRbJ2NvbnRleHQucGFnZV90eXBlJ107XG4gIGlmIChwYWdlVHlwZSA9PT0gJ3BkcCcgfHwgcGFnZVR5cGUgPT09ICdwbHAnKSB7XG4gICAgcmV0dXJuIGZpbHRlcmVkO1xuICB9XG4gIGNvbnN0IHJlc3QgPSB7IC4uLmZpbHRlcmVkIH07XG4gIGRlbGV0ZSByZXN0Wydjb250ZXh0LnBhZ2VfaWQnXTtcbiAgcmV0dXJuIHJlc3Q7XG59O1xuXG4vLyBFeHRyYWN0IHdoaXRlbGlzdGVkIHN1Yi1maWVsZHMgZnJvbSB0cmlnZ2VyLndpZGdldF9pbnRlcmFjdGlvbl9kYXRhXG5jb25zdCBwcm9qZWN0V2lkZ2V0SW50ZXJhY3Rpb25EYXRhID0gKFxuICBldmVudFByb3BzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPixcbiAgY29uZmlnOiBHQTRQcm9qZWN0ZWRFdmVudENvbmZpZyxcbik6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgaWYgKCFjb25maWcud2lkZ2V0SW50ZXJhY3Rpb25EYXRhUHJvamVjdGlvbnMpIHtcbiAgICByZXR1cm4ge307XG4gIH1cblxuICBjb25zdCBpbnRlcmFjdGlvbiA9IGV2ZW50UHJvcHNbJ3RyaWdnZXIud2lkZ2V0X2ludGVyYWN0aW9uJ107XG4gIGlmICh0eXBlb2YgaW50ZXJhY3Rpb24gIT09ICdzdHJpbmcnKSB7XG4gICAgcmV0dXJuIHt9O1xuICB9XG5cbiAgY29uc3QgcHJvamVjdGlvbk1hcCA9IGNvbmZpZy53aWRnZXRJbnRlcmFjdGlvbkRhdGFQcm9qZWN0aW9uc1tpbnRlcmFjdGlvbl07XG4gIGlmICghcHJvamVjdGlvbk1hcCkge1xuICAgIHJldHVybiB7fTtcbiAgfVxuXG4gIGNvbnN0IGludGVyYWN0aW9uRGF0YSA9IGV2ZW50UHJvcHNbJ3RyaWdnZXIud2lkZ2V0X2ludGVyYWN0aW9uX2RhdGEnXTtcbiAgaWYgKFxuICAgIGludGVyYWN0aW9uRGF0YSA9PT0gbnVsbCB8fFxuICAgIGludGVyYWN0aW9uRGF0YSA9PT0gdW5kZWZpbmVkIHx8XG4gICAgdHlwZW9mIGludGVyYWN0aW9uRGF0YSAhPT0gJ29iamVjdCdcbiAgKSB7XG4gICAgcmV0dXJuIHt9O1xuICB9XG5cbiAgY29uc3QgZGF0YSA9IGludGVyYWN0aW9uRGF0YSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgY29uc3QgcmVzdWx0OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IHt9O1xuICBmb3IgKGNvbnN0IFtnYUtleSwgc291cmNlRmllbGRdIG9mIE9iamVjdC5lbnRyaWVzKHByb2plY3Rpb25NYXApKSB7XG4gICAgaWYgKHNvdXJjZUZpZWxkIGluIGRhdGEpIHtcbiAgICAgIHJlc3VsdFtnYUtleV0gPSBkYXRhW3NvdXJjZUZpZWxkXTtcbiAgICB9XG4gIH1cbiAgcmV0dXJuIHJlc3VsdDtcbn07XG5cbmNvbnN0IHRydW5jYXRlU3RyaW5nID0gKHZhbHVlOiB1bmtub3duKTogdW5rbm93biA9PiB7XG4gIGlmICh0eXBlb2YgdmFsdWUgPT09ICdzdHJpbmcnICYmIHZhbHVlLmxlbmd0aCA+IDEwMCkge1xuICAgIHJldHVybiBgJHt2YWx1ZS5zdWJzdHJpbmcoMCwgOTcpfS4uLmA7XG4gIH1cbiAgcmV0dXJuIHZhbHVlO1xufTtcblxuY29uc3QgdHJ1bmNhdGVWYWx1ZXMgPSAob2JqOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPik6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgY29uc3QgcmVzdWx0OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IHt9O1xuICBmb3IgKGNvbnN0IFtrZXksIHZhbHVlXSBvZiBPYmplY3QuZW50cmllcyhvYmopKSB7XG4gICAgcmVzdWx0W2tleV0gPSB0cnVuY2F0ZVN0cmluZyh2YWx1ZSk7XG4gIH1cbiAgcmV0dXJuIHJlc3VsdDtcbn07XG5cbmNvbnN0IHB1c2hUb0RhdGFMYXllciA9IChnYUV2ZW50OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPik6IHZvaWQgPT4ge1xuICBpZiAodHlwZW9mIHdpbmRvdyAhPT0gJ3VuZGVmaW5lZCcgJiYgd2luZG93LmRhdGFMYXllcikge1xuICAgIHdpbmRvdy5kYXRhTGF5ZXIucHVzaChnYUV2ZW50KTtcbiAgfVxufTtcblxuZXhwb3J0IGNvbnN0IHByb2plY3RUb0dBNCA9IChcbiAgZXZlbnROYW1lOiBFbnZpdmVNZXRyaWNzRXZlbnROYW1lLFxuICBldmVudFByb3BzPzogUmVjb3JkPHN0cmluZywgdW5rbm93bj4sXG4pOiB2b2lkID0+IHtcbiAgdHJ5IHtcbiAgICBjb25zdCBzY2hlbWFFbnRyeSA9IEdBNF9FVkVOVF9TQ0hFTUFbZXZlbnROYW1lXTtcblxuICAgIGlmIChzY2hlbWFFbnRyeS5nYUV2ZW50TmFtZSA9PT0gbnVsbCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGNvbnN0IGNvbmZpZyA9IHNjaGVtYUVudHJ5O1xuICAgIGNvbnN0IHByb3BzID0gZXZlbnRQcm9wcyA/PyB7fTtcblxuICAgIGxldCBmaWx0ZXJlZCA9IGZpbHRlclRvU2NoZW1hKHByb3BzLCBjb25maWcuYWxsb3dlZEZpZWxkcyk7XG4gICAgZmlsdGVyZWQgPSBzYW5pdGl6ZVBhZ2VJZChmaWx0ZXJlZCk7XG5cbiAgICBjb25zdCBpbnRlcmFjdGlvbkZpZWxkcyA9IHByb2plY3RXaWRnZXRJbnRlcmFjdGlvbkRhdGEocHJvcHMsIGNvbmZpZyk7XG5cbiAgICBjb25zdCBmbGF0UGFyYW1zID0ge1xuICAgICAgLi4uZmxhdHRlbkRvdEtleXMoZmlsdGVyZWQpLFxuICAgICAgLi4uaW50ZXJhY3Rpb25GaWVsZHMsXG4gICAgfTtcblxuICAgIGNvbnN0IHRydW5jYXRlZFBhcmFtcyA9IHRydW5jYXRlVmFsdWVzKGZsYXRQYXJhbXMpO1xuXG4gICAgcHVzaFRvRGF0YUxheWVyKHtcbiAgICAgIGV2ZW50OiBjb25maWcuZ2FFdmVudE5hbWUsXG4gICAgICAuLi50cnVuY2F0ZWRQYXJhbXMsXG4gICAgfSk7XG4gIH0gY2F0Y2ggKGVycikge1xuICAgIGxvZ2dlci5sb2dFcnJvcignRXJyb3IgcHJvamVjdGluZyBldmVudCB0byBHQTQnLCBlcnIsIHtcbiAgICAgIGV2ZW50TmFtZSxcbiAgICB9KTtcbiAgfVxufTtcbiJdLCJtYXBwaW5ncyI6Ijs7OztBQUlBLE1BQU0sU0FBUyxJQUFJQSxlQUFPLHVCQUF1QjtBQUVqRCxNQUFNLGtCQUNKLFlBQ0Esa0JBQzRCO0NBQzVCLE1BQU1DLFNBQWtDLEVBQUU7QUFDMUMsTUFBSyxNQUFNLFNBQVMsY0FDbEIsS0FBSSxTQUFTLFdBQ1gsUUFBTyxTQUFTLFdBQVc7QUFHL0IsUUFBTzs7QUFJVCxNQUFNLGtCQUFrQixRQUEwRDtDQUNoRixNQUFNQSxTQUFrQyxFQUFFO0FBQzFDLE1BQUssTUFBTSxDQUFDLEtBQUssVUFBVSxPQUFPLFFBQVEsSUFBSSxDQUM1QyxRQUFPLElBQUksUUFBUSxPQUFPLElBQUksSUFBSTtBQUVwQyxRQUFPOztBQVNULE1BQU0sa0JBQWtCLGFBQStEO0NBQ3JGLE1BQU0sV0FBVyxTQUFTO0FBQzFCLEtBQUksYUFBYSxTQUFTLGFBQWEsTUFDckMsUUFBTztDQUVULE1BQU0sT0FBTyxFQUFFLEdBQUcsVUFBVTtBQUM1QixRQUFPLEtBQUs7QUFDWixRQUFPOztBQUlULE1BQU0sZ0NBQ0osWUFDQSxXQUM0QjtBQUM1QixLQUFJLENBQUMsT0FBTyxpQ0FDVixRQUFPLEVBQUU7Q0FHWCxNQUFNLGNBQWMsV0FBVztBQUMvQixLQUFJLE9BQU8sZ0JBQWdCLFNBQ3pCLFFBQU8sRUFBRTtDQUdYLE1BQU0sZ0JBQWdCLE9BQU8saUNBQWlDO0FBQzlELEtBQUksQ0FBQyxjQUNILFFBQU8sRUFBRTtDQUdYLE1BQU0sa0JBQWtCLFdBQVc7QUFDbkMsS0FDRSxvQkFBb0IsUUFDcEIsb0JBQW9CLFVBQ3BCLE9BQU8sb0JBQW9CLFNBRTNCLFFBQU8sRUFBRTtDQUdYLE1BQU0sT0FBTztDQUNiLE1BQU1BLFNBQWtDLEVBQUU7QUFDMUMsTUFBSyxNQUFNLENBQUMsT0FBTyxnQkFBZ0IsT0FBTyxRQUFRLGNBQWMsQ0FDOUQsS0FBSSxlQUFlLEtBQ2pCLFFBQU8sU0FBUyxLQUFLO0FBR3pCLFFBQU87O0FBR1QsTUFBTSxrQkFBa0IsVUFBNEI7QUFDbEQsS0FBSSxPQUFPLFVBQVUsWUFBWSxNQUFNLFNBQVMsSUFDOUMsUUFBTyxHQUFHLE1BQU0sVUFBVSxHQUFHLEdBQUcsQ0FBQztBQUVuQyxRQUFPOztBQUdULE1BQU0sa0JBQWtCLFFBQTBEO0NBQ2hGLE1BQU1BLFNBQWtDLEVBQUU7QUFDMUMsTUFBSyxNQUFNLENBQUMsS0FBSyxVQUFVLE9BQU8sUUFBUSxJQUFJLENBQzVDLFFBQU8sT0FBTyxlQUFlLE1BQU07QUFFckMsUUFBTzs7QUFHVCxNQUFNLG1CQUFtQixZQUEyQztBQUNsRSxLQUFJLE9BQU8sV0FBVyxlQUFlLE9BQU8sVUFDMUMsUUFBTyxVQUFVLEtBQUssUUFBUTs7QUFJbEMsTUFBYSxnQkFDWCxXQUNBLGVBQ1M7QUFDVCxLQUFJO0VBQ0YsTUFBTSxjQUFjLGlCQUFpQjtBQUVyQyxNQUFJLFlBQVksZ0JBQWdCLEtBQzlCO0VBR0YsTUFBTSxTQUFTO0VBQ2YsTUFBTSxRQUFRLGNBQWMsRUFBRTtFQUU5QixJQUFJLFdBQVcsZUFBZSxPQUFPLE9BQU8sY0FBYztBQUMxRCxhQUFXLGVBQWUsU0FBUztFQUVuQyxNQUFNLG9CQUFvQiw2QkFBNkIsT0FBTyxPQUFPO0VBT3JFLE1BQU0sa0JBQWtCLGVBTEw7R0FDakIsR0FBRyxlQUFlLFNBQVM7R0FDM0IsR0FBRztHQUNKLENBRWlEO0FBRWxELGtCQUFnQjtHQUNkLE9BQU8sT0FBTztHQUNkLEdBQUc7R0FDSixDQUFDO1VBQ0ssS0FBSztBQUNaLFNBQU8sU0FBUyxpQ0FBaUMsS0FBSyxFQUNwRCxXQUNELENBQUMifQ==
|
|
98
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ga4ProjectionService.js","names":["Logger","result: Record<string, unknown>","current: unknown","filtered: Record<string, unknown>"],"sources":["../../../src/services/ga4ProjectionService/ga4ProjectionService.ts"],"sourcesContent":["import Logger from 'src/application/logging/logger';\nimport { EnviveMetricsEventName } from '../amplitudeService/eventNames';\nimport { GA4ProjectedEventConfig, GA4_EVENT_SCHEMA } from './ga4EventSchema';\n\nconst logger = new Logger('ga4ProjectionService');\n\nconst filterToSchema = (\n  eventProps: Record<string, unknown>,\n  allowedFields: readonly string[],\n): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const field of allowedFields) {\n    if (field in eventProps) {\n      result[field] = eventProps[field];\n    }\n  }\n  return result;\n};\n\n// Flatten only true top-level nested objects (keys with no dots) one level deep.\n// Keys that already contain dots are left intact so that e.g.\n// `trigger.widget_interaction_data: { ... }` is preserved for downstream extraction.\nconst flattenOneLevel = (obj: Record<string, unknown>): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(obj)) {\n    if (\n      !key.includes('.') &&\n      value !== null &&\n      typeof value === 'object' &&\n      !Array.isArray(value)\n    ) {\n      for (const [subKey, subValue] of Object.entries(value as Record<string, unknown>)) {\n        result[`${key}.${subKey}`] = subValue;\n      }\n    } else {\n      result[key] = value;\n    }\n  }\n  return result;\n};\n\n// Apply a GA4 projection map { ga4Key: sourceKey } to extract and rename fields.\nconst filterWithProjections = (\n  eventProps: Record<string, unknown>,\n  projections: Record<string, string>,\n): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [ga4Key, sourceKey] of Object.entries(projections)) {\n    if (sourceKey in eventProps) {\n      result[ga4Key] = eventProps[sourceKey];\n    }\n  }\n  return result;\n};\n\n// \"context.page_type\" → \"context_page_type\"\nconst flattenDotKeys = (obj: Record<string, unknown>): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(obj)) {\n    result[key.replace(/\\./g, '_')] = value;\n  }\n  return result;\n};\n\n// Omit context.page_id for non-pdp/plp page types. The current implementation for context.page_id is:\n// PDP: product_id\n// PLP: plp_id\n// Search: search query\n// Other: page url\n// We want to omit all but pdp and plp page types to provide a clear, consistent interface for merchants.\n// Handles both dot-notation keys (allowedFields path) and renamed GA4 keys (fieldProjections path).\nconst sanitizePageId = (filtered: Record<string, unknown>): Record<string, unknown> => {\n  const pageType = filtered['page_type'] ?? filtered['context.page_type'];\n  if (pageType === 'pdp' || pageType === 'plp') {\n    return filtered;\n  }\n  const rest = { ...filtered };\n  delete rest['page_id'];\n  delete rest['context.page_id'];\n  return rest;\n};\n\nconst getNestedValue = (obj: Record<string, unknown>, path: string): unknown => {\n  const parts = path.split('.');\n  let current: unknown = obj;\n  for (const part of parts) {\n    if (current === null || typeof current !== 'object') return undefined;\n    current = (current as Record<string, unknown>)[part];\n  }\n  return current;\n};\n\n// Extract whitelisted sub-fields from trigger.widget_interaction_data\nconst projectWidgetInteractionData = (\n  eventProps: Record<string, unknown>,\n  config: GA4ProjectedEventConfig,\n): Record<string, unknown> => {\n  if (!config.widgetInteractionDataProjections) {\n    return {};\n  }\n\n  const interaction = eventProps['trigger.widget_interaction'];\n  if (typeof interaction !== 'string') {\n    return {};\n  }\n\n  const projectionMap = config.widgetInteractionDataProjections[interaction];\n  if (!projectionMap) {\n    return {};\n  }\n\n  const interactionData = eventProps['trigger.widget_interaction_data'];\n  if (\n    interactionData === null ||\n    interactionData === undefined ||\n    typeof interactionData !== 'object'\n  ) {\n    return {};\n  }\n\n  const data = interactionData as Record<string, unknown>;\n  const result: Record<string, unknown> = {};\n  for (const [gaKey, sourcePath] of Object.entries(projectionMap)) {\n    const value = getNestedValue(data, sourcePath);\n    if (value !== undefined) {\n      result[gaKey] = value;\n    }\n  }\n  return result;\n};\n\nconst truncateString = (value: unknown): unknown => {\n  if (typeof value === 'string' && value.length > 100) {\n    return `${value.substring(0, 97)}...`;\n  }\n  return value;\n};\n\nconst truncateValues = (obj: Record<string, unknown>): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(obj)) {\n    result[key] = truncateString(value);\n  }\n  return result;\n};\n\nconst pushToDataLayer = (gaEvent: Record<string, unknown>): void => {\n  if (typeof window !== 'undefined' && window.dataLayer) {\n    window.dataLayer.push(gaEvent);\n  }\n};\n\nexport const projectToGA4 = (\n  eventName: EnviveMetricsEventName,\n  eventProps?: Record<string, unknown>,\n): void => {\n  try {\n    const schemaEntry = GA4_EVENT_SCHEMA[eventName];\n\n    if (schemaEntry.gaEventName === null) {\n      return;\n    }\n\n    const config = schemaEntry;\n    const props = flattenOneLevel(eventProps ?? {});\n\n    let filtered: Record<string, unknown>;\n    if (config.fieldProjections) {\n      filtered = filterWithProjections(props, config.fieldProjections);\n    } else {\n      filtered = filterToSchema(props, config.allowedFields);\n    }\n    filtered = sanitizePageId(filtered);\n\n    const interactionFields = projectWidgetInteractionData(props, config);\n\n    const flatParams = {\n      ...flattenDotKeys(filtered),\n      ...interactionFields,\n    };\n\n    const truncatedParams = truncateValues(flatParams);\n\n    pushToDataLayer({\n      event: config.gaEventName,\n      ...truncatedParams,\n    });\n  } catch (err) {\n    logger.logError('Error projecting event to GA4', err, {\n      eventName,\n    });\n  }\n};\n"],"mappings":";;;;AAIA,MAAM,SAAS,IAAIA,eAAO,uBAAuB;AAEjD,MAAM,kBACJ,YACA,kBAC4B;CAC5B,MAAMC,SAAkC,EAAE;AAC1C,MAAK,MAAM,SAAS,cAClB,KAAI,SAAS,WACX,QAAO,SAAS,WAAW;AAG/B,QAAO;;AAMT,MAAM,mBAAmB,QAA0D;CACjF,MAAMA,SAAkC,EAAE;AAC1C,MAAK,MAAM,CAAC,KAAK,UAAU,OAAO,QAAQ,IAAI,CAC5C,KACE,CAAC,IAAI,SAAS,IAAI,IAClB,UAAU,QACV,OAAO,UAAU,YACjB,CAAC,MAAM,QAAQ,MAAM,CAErB,MAAK,MAAM,CAAC,QAAQ,aAAa,OAAO,QAAQ,MAAiC,CAC/E,QAAO,GAAG,IAAI,GAAG,YAAY;KAG/B,QAAO,OAAO;AAGlB,QAAO;;AAIT,MAAM,yBACJ,YACA,gBAC4B;CAC5B,MAAMA,SAAkC,EAAE;AAC1C,MAAK,MAAM,CAAC,QAAQ,cAAc,OAAO,QAAQ,YAAY,CAC3D,KAAI,aAAa,WACf,QAAO,UAAU,WAAW;AAGhC,QAAO;;AAIT,MAAM,kBAAkB,QAA0D;CAChF,MAAMA,SAAkC,EAAE;AAC1C,MAAK,MAAM,CAAC,KAAK,UAAU,OAAO,QAAQ,IAAI,CAC5C,QAAO,IAAI,QAAQ,OAAO,IAAI,IAAI;AAEpC,QAAO;;AAUT,MAAM,kBAAkB,aAA+D;CACrF,MAAM,WAAW,SAAS,gBAAgB,SAAS;AACnD,KAAI,aAAa,SAAS,aAAa,MACrC,QAAO;CAET,MAAM,OAAO,EAAE,GAAG,UAAU;AAC5B,QAAO,KAAK;AACZ,QAAO,KAAK;AACZ,QAAO;;AAGT,MAAM,kBAAkB,KAA8B,SAA0B;CAC9E,MAAM,QAAQ,KAAK,MAAM,IAAI;CAC7B,IAAIC,UAAmB;AACvB,MAAK,MAAM,QAAQ,OAAO;AACxB,MAAI,YAAY,QAAQ,OAAO,YAAY,SAAU,QAAO;AAC5D,YAAW,QAAoC;;AAEjD,QAAO;;AAIT,MAAM,gCACJ,YACA,WAC4B;AAC5B,KAAI,CAAC,OAAO,iCACV,QAAO,EAAE;CAGX,MAAM,cAAc,WAAW;AAC/B,KAAI,OAAO,gBAAgB,SACzB,QAAO,EAAE;CAGX,MAAM,gBAAgB,OAAO,iCAAiC;AAC9D,KAAI,CAAC,cACH,QAAO,EAAE;CAGX,MAAM,kBAAkB,WAAW;AACnC,KACE,oBAAoB,QACpB,oBAAoB,UACpB,OAAO,oBAAoB,SAE3B,QAAO,EAAE;CAGX,MAAM,OAAO;CACb,MAAMD,SAAkC,EAAE;AAC1C,MAAK,MAAM,CAAC,OAAO,eAAe,OAAO,QAAQ,cAAc,EAAE;EAC/D,MAAM,QAAQ,eAAe,MAAM,WAAW;AAC9C,MAAI,UAAU,OACZ,QAAO,SAAS;;AAGpB,QAAO;;AAGT,MAAM,kBAAkB,UAA4B;AAClD,KAAI,OAAO,UAAU,YAAY,MAAM,SAAS,IAC9C,QAAO,GAAG,MAAM,UAAU,GAAG,GAAG,CAAC;AAEnC,QAAO;;AAGT,MAAM,kBAAkB,QAA0D;CAChF,MAAMA,SAAkC,EAAE;AAC1C,MAAK,MAAM,CAAC,KAAK,UAAU,OAAO,QAAQ,IAAI,CAC5C,QAAO,OAAO,eAAe,MAAM;AAErC,QAAO;;AAGT,MAAM,mBAAmB,YAA2C;AAClE,KAAI,OAAO,WAAW,eAAe,OAAO,UAC1C,QAAO,UAAU,KAAK,QAAQ;;AAIlC,MAAa,gBACX,WACA,eACS;AACT,KAAI;EACF,MAAM,cAAc,iBAAiB;AAErC,MAAI,YAAY,gBAAgB,KAC9B;EAGF,MAAM,SAAS;EACf,MAAM,QAAQ,gBAAgB,cAAc,EAAE,CAAC;EAE/C,IAAIE;AACJ,MAAI,OAAO,iBACT,YAAW,sBAAsB,OAAO,OAAO,iBAAiB;MAEhE,YAAW,eAAe,OAAO,OAAO,cAAc;AAExD,aAAW,eAAe,SAAS;EAEnC,MAAM,oBAAoB,6BAA6B,OAAO,OAAO;EAOrE,MAAM,kBAAkB,eALL;GACjB,GAAG,eAAe,SAAS;GAC3B,GAAG;GACJ,CAEiD;AAElD,kBAAgB;GACd,OAAO,OAAO;GACd,GAAG;GACJ,CAAC;UACK,KAAK;AACZ,SAAO,SAAS,iCAAiC,KAAK,EACpD,WACD,CAAC"}
|
package/package.json
CHANGED
|
@@ -27,4 +27,5 @@ export enum FeatureGates {
|
|
|
27
27
|
IsAiSuggestionsVariantEnabled = 'is_ai_suggestions_variant_enabled',
|
|
28
28
|
IsAiSuggestionsVariantAEnabled = 'is_ai_suggestions_variant_a_enabled',
|
|
29
29
|
IsAiSuggestionsVariantBEnabled = 'is_ai_suggestions_variant_b_enabled',
|
|
30
|
+
IsVoiceInputEnabled = 'is_voice_input_enabled',
|
|
30
31
|
}
|
|
@@ -26,6 +26,7 @@ export const initializedAtom = atom<boolean>(false);
|
|
|
26
26
|
export const chatIsOpenAtom = atom<boolean>(false);
|
|
27
27
|
export const requestFailureAtom = atom<boolean>(false);
|
|
28
28
|
export const formSubmitAtom = atom<FormSubmittedAttributes>();
|
|
29
|
+
export const listeningToSpeechAtom = atom<boolean>(false);
|
|
29
30
|
export const chatOnToggleAtom = atom(
|
|
30
31
|
null,
|
|
31
32
|
(
|
|
@@ -20,6 +20,7 @@ import { AmplitudeProvider, SpiffyMetricsEventName, useAmplitude } from '../ampl
|
|
|
20
20
|
const mockTrackEvent = vi.fn().mockResolvedValue(undefined);
|
|
21
21
|
const mockSetSupplementalDefaultProps = vi.fn();
|
|
22
22
|
const mockIsReady = vi.fn().mockReturnValue(true);
|
|
23
|
+
const mockIsMockApiKey = vi.fn().mockReturnValue(false);
|
|
23
24
|
|
|
24
25
|
vi.mock('src/services/amplitudeService/amplitudeService', async () => {
|
|
25
26
|
const actual = await vi.importActual<
|
|
@@ -34,6 +35,10 @@ vi.mock('src/services/amplitudeService/amplitudeService', async () => {
|
|
|
34
35
|
get isReady(): boolean {
|
|
35
36
|
return mockIsReady();
|
|
36
37
|
}
|
|
38
|
+
|
|
39
|
+
get isMockApiKey(): boolean {
|
|
40
|
+
return mockIsMockApiKey();
|
|
41
|
+
}
|
|
37
42
|
}
|
|
38
43
|
|
|
39
44
|
function MockAmplitudeServiceConstructor() {
|
|
@@ -55,6 +60,7 @@ const MockAmplitudeComponent: React.FC = () => {
|
|
|
55
60
|
return (
|
|
56
61
|
<div data-testid="amplitude-component">
|
|
57
62
|
<div data-testid="is-ready">{amplitude.isReady.toString()}</div>
|
|
63
|
+
<div data-testid="is-mock-mode">{amplitude.isMockMode.toString()}</div>
|
|
58
64
|
<button
|
|
59
65
|
data-testid="track-event-button"
|
|
60
66
|
type="button"
|
|
@@ -213,6 +219,7 @@ describe('AmplitudeProvider - React Context Integration', () => {
|
|
|
213
219
|
mockTrackEvent.mockClear();
|
|
214
220
|
mockSetSupplementalDefaultProps.mockClear();
|
|
215
221
|
mockIsReady.mockReturnValue(true);
|
|
222
|
+
mockIsMockApiKey.mockReturnValue(false);
|
|
216
223
|
if (AmplitudeService && typeof AmplitudeService === 'function') {
|
|
217
224
|
(AmplitudeService as unknown as ReturnType<typeof vi.fn>).mockClear();
|
|
218
225
|
}
|
|
@@ -332,38 +339,35 @@ describe('AmplitudeProvider - React Context Integration', () => {
|
|
|
332
339
|
});
|
|
333
340
|
});
|
|
334
341
|
|
|
335
|
-
it('should
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
);
|
|
356
|
-
};
|
|
342
|
+
it('should render children with isReady=false when isMockMode=true', async () => {
|
|
343
|
+
mockIsReady.mockReturnValue(false);
|
|
344
|
+
mockIsMockApiKey.mockReturnValue(true);
|
|
345
|
+
|
|
346
|
+
render(
|
|
347
|
+
<TestWrapper amplitudeApiKey="mock-amplitude-key">
|
|
348
|
+
<MockAmplitudeComponent />
|
|
349
|
+
</TestWrapper>,
|
|
350
|
+
);
|
|
351
|
+
|
|
352
|
+
await waitFor(() => {
|
|
353
|
+
expect(screen.getByTestId('amplitude-component')).toBeInTheDocument();
|
|
354
|
+
expect(screen.getByTestId('is-ready').textContent).toBe('false');
|
|
355
|
+
expect(screen.getByTestId('is-mock-mode').textContent).toBe('true');
|
|
356
|
+
});
|
|
357
|
+
});
|
|
358
|
+
|
|
359
|
+
it('should NOT render children when isReady=false and isMockMode=false', async () => {
|
|
360
|
+
mockIsReady.mockReturnValue(false);
|
|
361
|
+
mockIsMockApiKey.mockReturnValue(false);
|
|
357
362
|
|
|
358
363
|
render(
|
|
359
|
-
<
|
|
360
|
-
<
|
|
361
|
-
</
|
|
364
|
+
<TestWrapper>
|
|
365
|
+
<MockAmplitudeComponent />
|
|
366
|
+
</TestWrapper>,
|
|
362
367
|
);
|
|
363
368
|
|
|
364
|
-
// AmplitudeProvider returns null when not ready, so children should not render
|
|
365
369
|
await waitFor(() => {
|
|
366
|
-
expect(screen.queryByTestId('
|
|
370
|
+
expect(screen.queryByTestId('amplitude-component')).not.toBeInTheDocument();
|
|
367
371
|
});
|
|
368
372
|
});
|
|
369
373
|
});
|
|
@@ -23,6 +23,7 @@ export { EnviveMetricsEventName, SpiffyMetricsEventName };
|
|
|
23
23
|
interface AmplitudeContextType {
|
|
24
24
|
trackEvent: (params: TrackEventParams) => Promise<void>;
|
|
25
25
|
isReady: boolean;
|
|
26
|
+
isMockMode: boolean;
|
|
26
27
|
setSupplementalDefaultProps: (props: Record<string, unknown>) => void;
|
|
27
28
|
}
|
|
28
29
|
|
|
@@ -55,6 +56,7 @@ export const AmplitudeProvider: React.FC<{
|
|
|
55
56
|
const [service, setService] = useState<AmplitudeService | null>(null);
|
|
56
57
|
|
|
57
58
|
const isReady = Boolean(userId && service && service.isReady);
|
|
59
|
+
const isMockMode = Boolean(service?.isMockApiKey);
|
|
58
60
|
|
|
59
61
|
// Create service instance when dependencies are ready
|
|
60
62
|
useEffect(() => {
|
|
@@ -109,16 +111,17 @@ export const AmplitudeProvider: React.FC<{
|
|
|
109
111
|
}
|
|
110
112
|
},
|
|
111
113
|
isReady,
|
|
114
|
+
isMockMode,
|
|
112
115
|
setSupplementalDefaultProps: (props: Record<string, unknown>) => {
|
|
113
116
|
if (service) {
|
|
114
117
|
service.setSupplementalDefaultProps(props);
|
|
115
118
|
}
|
|
116
119
|
},
|
|
117
120
|
}),
|
|
118
|
-
[service, isReady],
|
|
121
|
+
[service, isReady, isMockMode],
|
|
119
122
|
);
|
|
120
123
|
|
|
121
|
-
if (!isReady) {
|
|
124
|
+
if (!isReady && !isMockMode) {
|
|
122
125
|
return null;
|
|
123
126
|
}
|
|
124
127
|
|
|
@@ -219,10 +219,18 @@ export const HardcopyProvider: React.FC<HardcopyProviderProps> = ({
|
|
|
219
219
|
const { featureFlagService } = useFeatureFlagService();
|
|
220
220
|
const getHardcopyFromBackend = useCallback(
|
|
221
221
|
async (request: HardcopyRequest): Promise<HardcopyResponse> => {
|
|
222
|
-
|
|
222
|
+
const fallbackWidgetType =
|
|
223
|
+
request.widgetType === WidgetTypeV3.ProductCardV3
|
|
224
|
+
? WidgetTypeV3.ImagePromptCardV3
|
|
225
|
+
: request.widgetType;
|
|
226
|
+
|
|
227
|
+
const overrideEntry =
|
|
228
|
+
hardcopyOverride?.[request.widgetType] ?? hardcopyOverride?.[fallbackWidgetType];
|
|
229
|
+
if (overrideEntry) {
|
|
223
230
|
logger.logDebug('using hardcopy override', request.widgetType);
|
|
224
|
-
return
|
|
231
|
+
return overrideEntry;
|
|
225
232
|
}
|
|
233
|
+
|
|
226
234
|
const overrideConfigVersion =
|
|
227
235
|
getQueryParam('spiffy_config_version') ||
|
|
228
236
|
getQueryParam('envive_config_version') ||
|
|
@@ -19,6 +19,16 @@ vi.spyOn(Logger.prototype, 'logInfo').mockImplementation(() => {});
|
|
|
19
19
|
vi.spyOn(Logger.prototype, 'logWarn').mockImplementation(() => {});
|
|
20
20
|
vi.spyOn(Logger.prototype, 'logError').mockImplementation(() => {});
|
|
21
21
|
|
|
22
|
+
vi.mock('src/contexts/amplitudeContext', () => ({
|
|
23
|
+
useAmplitude: () => ({
|
|
24
|
+
trackEvent: vi.fn(),
|
|
25
|
+
isReady: true,
|
|
26
|
+
}),
|
|
27
|
+
EnviveMetricsEventName: {
|
|
28
|
+
PageViewed: 'Page Viewed',
|
|
29
|
+
},
|
|
30
|
+
}));
|
|
31
|
+
|
|
22
32
|
// Mock CommerceApiClient
|
|
23
33
|
const mockResolveUrl = vi.fn();
|
|
24
34
|
vi.mock('src/application/commerce-api', () => ({
|
|
@@ -14,6 +14,7 @@ import {
|
|
|
14
14
|
import { hasParsedVariantInfoAtom } from 'src/atoms/app';
|
|
15
15
|
import { analyticsContextAtom } from 'src/atoms/app/variant';
|
|
16
16
|
import { formSubmitAtom, replyEventCategoryAtom } from 'src/atoms/chat';
|
|
17
|
+
import { suggestionsAtom } from 'src/atoms/chat/chatState';
|
|
17
18
|
import { queueUserEventAtom } from 'src/atoms/chat/messageQueue';
|
|
18
19
|
import { useWidgetInteraction } from 'src/hooks/WidgetInteraction';
|
|
19
20
|
import {
|
|
@@ -56,6 +57,7 @@ export const useSalesAgentChatAPI = (widget?: WidgetInteractionComponent) => {
|
|
|
56
57
|
const context = useAtomValue(analyticsContextAtom);
|
|
57
58
|
const hasParsedVariantInfo = useAtomValue(hasParsedVariantInfoAtom);
|
|
58
59
|
const queueUserEvent = useSetAtom(queueUserEventAtom);
|
|
60
|
+
const setSuggestions = useSetAtom(suggestionsAtom);
|
|
59
61
|
const setReplyEventCategory = useSetAtom(replyEventCategoryAtom);
|
|
60
62
|
const setFormSubmit = useSetAtom(formSubmitAtom);
|
|
61
63
|
const { trackEvent } = useAmplitude();
|
|
@@ -164,9 +166,10 @@ export const useSalesAgentChatAPI = (widget?: WidgetInteractionComponent) => {
|
|
|
164
166
|
content: suggestion.content,
|
|
165
167
|
},
|
|
166
168
|
};
|
|
169
|
+
setSuggestions([]);
|
|
167
170
|
queueUserEvent(event);
|
|
168
171
|
},
|
|
169
|
-
[queueUserEvent, trackEvent, context],
|
|
172
|
+
[queueUserEvent, setSuggestions, trackEvent, context],
|
|
170
173
|
);
|
|
171
174
|
const onTypedMessageSubmitted = useCallback(
|
|
172
175
|
({
|
|
@@ -229,9 +232,10 @@ export const useSalesAgentChatAPI = (widget?: WidgetInteractionComponent) => {
|
|
|
229
232
|
userTyped,
|
|
230
233
|
},
|
|
231
234
|
};
|
|
235
|
+
setSuggestions([]);
|
|
232
236
|
queueUserEvent(event);
|
|
233
237
|
},
|
|
234
|
-
[queueUserEvent, trackEvent, context],
|
|
238
|
+
[queueUserEvent, setSuggestions, trackEvent, context],
|
|
235
239
|
);
|
|
236
240
|
const onFormResponseSubmitted = useCallback(
|
|
237
241
|
(form: FormSubmittedAttributes) => {
|
package/src/contexts/typesV3.ts
CHANGED
|
@@ -29,10 +29,6 @@ import { UserIdentityService } from 'src/services/userIdentityService';
|
|
|
29
29
|
import { useSearch } from '../useSearch';
|
|
30
30
|
|
|
31
31
|
// Mock dependencies
|
|
32
|
-
vi.mock('src/hooks/TrackComponentVisibleEvent/useTrackComponentVisibleEvent', () => ({
|
|
33
|
-
useTrackComponentVisibleEvent: vi.fn(),
|
|
34
|
-
}));
|
|
35
|
-
|
|
36
32
|
vi.mock('src/hooks/Intersection/useIntersection', () => ({
|
|
37
33
|
useIntersection: vi.fn(() => false),
|
|
38
34
|
}));
|