@envive-ai/react-hooks 0.3.9 → 0.3.11

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (210) hide show
  1. package/dist/application/commerce-api.cjs +14 -14
  2. package/dist/application/commerce-api.d.cts +58 -0
  3. package/dist/application/commerce-api.d.ts +59 -0
  4. package/dist/application/commerce-api.js +3 -3
  5. package/dist/application/models/api/widgetText.d.cts +14 -0
  6. package/dist/application/models/api/widgetText.d.ts +14 -0
  7. package/dist/application/models/api/widgetTextRequest.d.cts +15 -0
  8. package/dist/application/models/api/widgetTextRequest.d.ts +15 -0
  9. package/dist/application/models/chatElementDisplayLocationV3.cjs +2 -1
  10. package/dist/application/models/chatElementDisplayLocationV3.d.cts +2 -1
  11. package/dist/application/models/chatElementDisplayLocationV3.d.ts +2 -1
  12. package/dist/application/models/chatElementDisplayLocationV3.js +2 -1
  13. package/dist/application/models/guards/api/index.cjs +4 -4
  14. package/dist/application/models/guards/api/index.js +4 -4
  15. package/dist/application/models/guards/api/isApiFormResponse.cjs +1 -1
  16. package/dist/application/models/guards/api/isApiFormResponse.js +1 -1
  17. package/dist/application/models/guards/api/isApiFormSubmittedResponseAttributes.cjs +1 -1
  18. package/dist/application/models/guards/api/isApiFormSubmittedResponseAttributes.js +1 -1
  19. package/dist/application/models/guards/api/isApiOrderResponseAttributes.cjs +1 -1
  20. package/dist/application/models/guards/api/isApiOrderResponseAttributes.js +1 -1
  21. package/dist/application/models/guards/api/isApiProductResponseAttributes.cjs +1 -1
  22. package/dist/application/models/guards/api/isApiProductResponseAttributes.js +1 -1
  23. package/dist/application/models/guards/api/isApiResponse.cjs +1 -1
  24. package/dist/application/models/guards/api/isApiResponse.js +1 -1
  25. package/dist/application/models/index.cjs +12 -12
  26. package/dist/application/models/index.d.cts +25 -25
  27. package/dist/application/models/index.d.ts +25 -25
  28. package/dist/application/models/index.js +12 -12
  29. package/dist/application/models/validators/validateGraphQLFrontendConfig.cjs +1 -1
  30. package/dist/application/models/validators/validateGraphQLFrontendConfig.js +1 -1
  31. package/dist/application/models/validators/validateUserEvent.cjs +2 -2
  32. package/dist/application/models/validators/validateUserEvent.js +2 -2
  33. package/dist/application/models/variantInfo/variantInfo.cjs +2 -1
  34. package/dist/application/models/variantInfo/variantInfo.d.cts +6 -1
  35. package/dist/application/models/variantInfo/variantInfo.d.ts +6 -1
  36. package/dist/application/models/variantInfo/variantInfo.js +2 -1
  37. package/dist/application/utils/analyticsUtils.cjs +1 -1
  38. package/dist/application/utils/analyticsUtils.js +1 -1
  39. package/dist/application/utils/widgetTextFromApiWidgetTextResponse.cjs +3 -1
  40. package/dist/application/utils/widgetTextFromApiWidgetTextResponse.js +3 -1
  41. package/dist/application/utils/widgetTextRequestToApiRequest.cjs +3 -1
  42. package/dist/application/utils/widgetTextRequestToApiRequest.js +3 -1
  43. package/dist/atoms/app/index.cjs +7 -7
  44. package/dist/atoms/app/index.d.cts +8 -8
  45. package/dist/atoms/app/index.d.ts +8 -8
  46. package/dist/atoms/app/index.js +1 -1
  47. package/dist/atoms/app/variant.cjs +44 -4
  48. package/dist/atoms/app/variant.d.cts +8 -8
  49. package/dist/atoms/app/variant.d.ts +8 -8
  50. package/dist/atoms/app/variant.js +44 -5
  51. package/dist/atoms/chat/chatState.d.cts +17 -17
  52. package/dist/atoms/chat/chatState.d.ts +18 -18
  53. package/dist/atoms/chat/form.cjs +1 -1
  54. package/dist/atoms/chat/form.d.cts +2 -2
  55. package/dist/atoms/chat/form.d.ts +2 -2
  56. package/dist/atoms/chat/form.js +1 -1
  57. package/dist/atoms/chat/index.cjs +1 -1
  58. package/dist/atoms/chat/index.d.cts +2 -2
  59. package/dist/atoms/chat/index.d.ts +2 -2
  60. package/dist/atoms/chat/index.js +1 -1
  61. package/dist/atoms/chat/lastMessage.d.cts +2 -2
  62. package/dist/atoms/chat/lastMessage.d.ts +2 -2
  63. package/dist/atoms/chat/messageQueue.d.cts +6 -6
  64. package/dist/atoms/chat/messageQueue.d.ts +6 -6
  65. package/dist/atoms/chat/performanceMetrics.d.cts +6 -6
  66. package/dist/atoms/chat/performanceMetrics.d.ts +6 -6
  67. package/dist/atoms/chat/renderedWidgetRefs.d.cts +2 -2
  68. package/dist/atoms/chat/renderedWidgetRefs.d.ts +2 -2
  69. package/dist/atoms/chat/replies.d.cts +3 -3
  70. package/dist/atoms/chat/replies.d.ts +3 -3
  71. package/dist/atoms/chat/suggestions.d.cts +2 -2
  72. package/dist/atoms/chat/suggestions.d.ts +2 -2
  73. package/dist/atoms/envive/enviveConfig.d.cts +19 -0
  74. package/dist/atoms/envive/enviveConfig.d.ts +19 -0
  75. package/dist/atoms/globalSearch/globalSearch.d.cts +5 -5
  76. package/dist/atoms/globalSearch/globalSearch.d.ts +5 -5
  77. package/dist/atoms/org/customerService.d.cts +6 -6
  78. package/dist/atoms/org/customerService.d.ts +6 -6
  79. package/dist/atoms/org/graphqlConfig.d.cts +4 -4
  80. package/dist/atoms/org/graphqlConfig.d.ts +4 -4
  81. package/dist/atoms/org/index.cjs +1 -1
  82. package/dist/atoms/org/index.js +1 -1
  83. package/dist/atoms/org/newOrgConfigAtom.d.cts +2 -2
  84. package/dist/atoms/org/newOrgConfigAtom.d.ts +2 -2
  85. package/dist/atoms/org/orgAnalyticsConfig.d.cts +5 -5
  86. package/dist/atoms/org/orgAnalyticsConfig.d.ts +5 -5
  87. package/dist/atoms/search/chatSearch.cjs +2 -2
  88. package/dist/atoms/search/chatSearch.d.cts +17 -17
  89. package/dist/atoms/search/chatSearch.d.ts +17 -17
  90. package/dist/atoms/search/chatSearch.js +2 -2
  91. package/dist/atoms/search/searchAPI.cjs +1 -1
  92. package/dist/atoms/search/searchAPI.d.cts +13 -13
  93. package/dist/atoms/search/searchAPI.d.ts +13 -13
  94. package/dist/atoms/search/searchAPI.js +1 -1
  95. package/dist/atoms/widget/chatPreviewLoading.d.cts +2 -2
  96. package/dist/atoms/widget/chatPreviewLoading.d.ts +2 -2
  97. package/dist/contexts/amplitudeContext/amplitudeContext.cjs +10 -7
  98. package/dist/contexts/amplitudeContext/amplitudeContext.d.cts +2 -2
  99. package/dist/contexts/amplitudeContext/amplitudeContext.d.ts +2 -2
  100. package/dist/contexts/amplitudeContext/amplitudeContext.js +6 -3
  101. package/dist/contexts/amplitudeContext/index.cjs +1 -0
  102. package/dist/contexts/amplitudeContext/index.d.cts +2 -2
  103. package/dist/contexts/amplitudeContext/index.d.ts +2 -2
  104. package/dist/contexts/amplitudeContext/index.js +2 -2
  105. package/dist/contexts/cdnContext/cdnContext.cjs +3 -3
  106. package/dist/contexts/enviveConfigContext/enviveConfigContext.cjs +6 -5
  107. package/dist/contexts/enviveConfigContext/enviveConfigContext.d.cts +2 -1
  108. package/dist/contexts/enviveConfigContext/enviveConfigContext.d.ts +2 -1
  109. package/dist/contexts/enviveConfigContext/enviveConfigContext.js +3 -2
  110. package/dist/contexts/enviveContext/WindowChatToggleBinder.cjs +4 -3
  111. package/dist/contexts/enviveContext/WindowChatToggleBinder.js +4 -3
  112. package/dist/contexts/enviveContext/enviveContext.cjs +15 -11
  113. package/dist/contexts/enviveContext/enviveContext.d.cts +2 -1
  114. package/dist/contexts/enviveContext/enviveContext.d.ts +2 -1
  115. package/dist/contexts/enviveContext/enviveContext.js +15 -11
  116. package/dist/contexts/enviveContext/types.d.cts +1 -1
  117. package/dist/contexts/enviveContext/types.d.ts +1 -1
  118. package/dist/contexts/featureFlagServiceContext/featureFlagServiceContext.cjs +23 -2
  119. package/dist/contexts/featureFlagServiceContext/featureFlagServiceContext.d.cts +11 -2
  120. package/dist/contexts/featureFlagServiceContext/featureFlagServiceContext.d.ts +11 -2
  121. package/dist/contexts/featureFlagServiceContext/featureFlagServiceContext.js +23 -2
  122. package/dist/contexts/featureFlagServiceContext/index.d.cts +2 -2
  123. package/dist/contexts/featureFlagServiceContext/index.d.ts +2 -2
  124. package/dist/contexts/graphqlContext/graphqlContext.cjs +7 -7
  125. package/dist/contexts/graphqlContext/graphqlContext.d.cts +1 -1
  126. package/dist/contexts/graphqlContext/graphqlContext.d.ts +1 -1
  127. package/dist/contexts/graphqlContext/graphqlContext.js +3 -3
  128. package/dist/contexts/graphqlContext/mockV3Config.cjs +10 -4
  129. package/dist/contexts/graphqlContext/mockV3Config.js +10 -4
  130. package/dist/contexts/hardcopyContext/hardcopyContext.cjs +50 -5
  131. package/dist/contexts/hardcopyContext/hardcopyContext.d.cts +4 -1
  132. package/dist/contexts/hardcopyContext/hardcopyContext.d.ts +4 -1
  133. package/dist/contexts/hardcopyContext/hardcopyContext.js +48 -3
  134. package/dist/contexts/newOrgConfigContext/newOrgConfigContext.cjs +4 -4
  135. package/dist/contexts/newOrgConfigContext/newOrgConfigContext.js +1 -1
  136. package/dist/contexts/pageContext/mapping.cjs +6 -2
  137. package/dist/contexts/pageContext/mapping.js +6 -2
  138. package/dist/contexts/pageContext/pageContext.cjs +4 -3
  139. package/dist/contexts/pageContext/pageContext.js +2 -1
  140. package/dist/contexts/pageContext/types.d.cts +7 -3
  141. package/dist/contexts/pageContext/types.d.ts +7 -3
  142. package/dist/contexts/salesAgentContext/chatAPI.cjs +103 -13
  143. package/dist/contexts/salesAgentContext/chatAPI.d.cts +8 -4
  144. package/dist/contexts/salesAgentContext/chatAPI.d.ts +8 -4
  145. package/dist/contexts/salesAgentContext/chatAPI.js +105 -15
  146. package/dist/contexts/salesAgentContext/salesAgentContext.cjs +1 -1
  147. package/dist/contexts/salesAgentContext/salesAgentContext.js +1 -1
  148. package/dist/contexts/salesAgentContext/salesAgentService.cjs +53 -11
  149. package/dist/contexts/salesAgentContext/salesAgentService.js +51 -9
  150. package/dist/contexts/searchContext/searchContext.cjs +4 -4
  151. package/dist/contexts/searchContext/searchContext.js +1 -1
  152. package/dist/contexts/systemSettingsContext/systemSettingsContext.cjs +3 -3
  153. package/dist/contexts/systemSettingsContext/systemSettingsContext.d.cts +2 -2
  154. package/dist/contexts/systemSettingsContext/systemSettingsContext.d.ts +2 -2
  155. package/dist/contexts/types.cjs +1 -1
  156. package/dist/contexts/types.d.cts +7 -2
  157. package/dist/contexts/types.d.ts +7 -2
  158. package/dist/contexts/types.js +1 -1
  159. package/dist/contexts/typesV3.cjs +2 -1
  160. package/dist/contexts/typesV3.d.cts +12 -7
  161. package/dist/contexts/typesV3.d.ts +10 -5
  162. package/dist/contexts/typesV3.js +2 -1
  163. package/dist/contexts/userIdentityContext/userIdentityContext.cjs +3 -3
  164. package/dist/contexts/widgetConfigContext/widgetConfigContext.cjs +1 -1
  165. package/dist/contexts/widgetConfigContext/widgetConfigContext.js +1 -1
  166. package/dist/hooks/AppDetails/useAppDetails.cjs +6 -6
  167. package/dist/hooks/GrabAndScroll/useGrabAndScroll.d.cts +2 -2
  168. package/dist/hooks/GrabAndScroll/useGrabAndScroll.d.ts +2 -2
  169. package/dist/hooks/ImageResolver/useImageResolver.cjs +3 -3
  170. package/dist/hooks/Search/useSearch.cjs +6 -6
  171. package/dist/hooks/Search/useSearch.d.cts +1 -1
  172. package/dist/hooks/Search/useSearch.d.ts +1 -1
  173. package/dist/hooks/Search/useSearch.js +3 -3
  174. package/dist/hooks/Search/useSearchInput.cjs +2 -2
  175. package/dist/hooks/Search/useSearchInput.js +2 -2
  176. package/dist/services/amplitudeService/amplitudeService.cjs +49 -11
  177. package/dist/services/amplitudeService/amplitudeService.d.cts +18 -3
  178. package/dist/services/amplitudeService/amplitudeService.d.ts +18 -3
  179. package/dist/services/amplitudeService/amplitudeService.js +49 -12
  180. package/dist/services/amplitudeService/index.cjs +1 -0
  181. package/dist/services/amplitudeService/index.d.cts +2 -2
  182. package/dist/services/amplitudeService/index.d.ts +2 -2
  183. package/dist/services/amplitudeService/index.js +2 -2
  184. package/dist/types/customerService.d.cts +3 -1
  185. package/dist/types/customerService.d.ts +1 -1
  186. package/package.json +10 -2
  187. package/src/application/models/api/widgetText.ts +2 -0
  188. package/src/application/models/api/widgetTextRequest.ts +1 -0
  189. package/src/application/models/chatElementDisplayLocationV3.ts +1 -0
  190. package/src/application/models/variantInfo/variantInfo.ts +5 -0
  191. package/src/application/utils/widgetTextFromApiWidgetTextResponse.ts +2 -0
  192. package/src/application/utils/widgetTextRequestToApiRequest.ts +4 -0
  193. package/src/atoms/app/variant.ts +56 -2
  194. package/src/contexts/amplitudeContext/__tests__/amplitudeContext.test.tsx +2 -0
  195. package/src/contexts/amplitudeContext/amplitudeContext.tsx +6 -1
  196. package/src/contexts/enviveConfigContext/enviveConfigContext.tsx +3 -0
  197. package/src/contexts/enviveContext/WindowChatToggleBinder.tsx +5 -1
  198. package/src/contexts/enviveContext/enviveContext.tsx +5 -1
  199. package/src/contexts/featureFlagServiceContext/featureFlagServiceContext.tsx +36 -0
  200. package/src/contexts/graphqlContext/mockV3Config.ts +13 -7
  201. package/src/contexts/hardcopyContext/hardcopyContext.tsx +48 -3
  202. package/src/contexts/pageContext/mapping.ts +13 -1
  203. package/src/contexts/pageContext/pageContext.tsx +1 -0
  204. package/src/contexts/pageContext/types.ts +7 -1
  205. package/src/contexts/salesAgentContext/chatAPI.ts +106 -13
  206. package/src/contexts/salesAgentContext/salesAgentService.ts +66 -4
  207. package/src/contexts/types.ts +8 -1
  208. package/src/contexts/typesV3.ts +24 -16
  209. package/src/services/amplitudeService/__tests__/amplitudeService.test.ts +80 -0
  210. package/src/services/amplitudeService/amplitudeService.ts +55 -11
@@ -14,12 +14,14 @@ interface SalesAgentChatAPI {
14
14
  onSuggestionClicked: (suggestion: Suggestion$1, displayLocation: ChatElementDisplayLocationV3) => void;
15
15
  onTypedMessageSubmitted: ({
16
16
  query,
17
- userTyped
17
+ userTyped,
18
+ displayLocation
18
19
  }: {
19
20
  query: string;
20
21
  userTyped: boolean;
22
+ displayLocation: ChatElementDisplayLocationV3;
21
23
  }) => void;
22
- onFormResponseSubmitted: (formResponse: FormSubmittedAttributes$1) => void;
24
+ onFormResponseSubmitted: (form: FormSubmittedAttributes$1) => void;
23
25
  }
24
26
  declare const useSalesAgentChatAPI: () => {
25
27
  logPageVisit: ({
@@ -31,13 +33,15 @@ declare const useSalesAgentChatAPI: () => {
31
33
  onSuggestionClicked: (suggestion: Suggestion$1, displayLocation: ChatElementDisplayLocationV3) => void;
32
34
  onTypedMessageSubmitted: ({
33
35
  query,
34
- userTyped
36
+ userTyped,
37
+ displayLocation
35
38
  }: {
36
39
  query: string;
37
40
  userTyped: boolean;
41
+ displayLocation: ChatElementDisplayLocationV3;
38
42
  }) => void;
39
43
  onFormResponseSubmitted: (form: FormSubmittedAttributes$1) => void;
40
44
  };
41
45
  //#endregion
42
46
  export { SalesAgentChatAPI, useSalesAgentChatAPI };
43
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdEFQSS5kLmN0cyIsIm5hbWVzIjpbXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvY29udGV4dHMvc2FsZXNBZ2VudENvbnRleHQvY2hhdEFQSS5kLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFBhZ2VWaXNpdENhdGVnb3J5IH0gZnJvbSAnQHNwaWZmeS1haS9jb21tZXJjZS1hcGktY2xpZW50JztcbmltcG9ydCB7IENoYXRFbGVtZW50RGlzcGxheUxvY2F0aW9uVjMsIEZvcm1TdWJtaXR0ZWRBdHRyaWJ1dGVzLCBTdWdnZXN0aW9uLCBVc2VyRXZlbnQgfSBmcm9tICdzcmMvYXBwbGljYXRpb24vbW9kZWxzJztcbmV4cG9ydCBpbnRlcmZhY2UgU2FsZXNBZ2VudENoYXRBUEkge1xuICAgIGxvZ1BhZ2VWaXNpdDogKHsgcGFnZVZpc2l0Q2F0ZWdvcnkgfToge1xuICAgICAgICBwYWdlVmlzaXRDYXRlZ29yeTogUGFnZVZpc2l0Q2F0ZWdvcnk7XG4gICAgfSkgPT4gdm9pZDtcbiAgICBsb2dVc2VyRXZlbnQ6IChldmVudDogVXNlckV2ZW50KSA9PiB2b2lkO1xuICAgIG9uU3VnZ2VzdGlvbkNsaWNrZWQ6IChzdWdnZXN0aW9uOiBTdWdnZXN0aW9uLCBkaXNwbGF5TG9jYXRpb246IENoYXRFbGVtZW50RGlzcGxheUxvY2F0aW9uVjMpID0+IHZvaWQ7XG4gICAgb25UeXBlZE1lc3NhZ2VTdWJtaXR0ZWQ6ICh7IHF1ZXJ5LCB1c2VyVHlwZWQgfToge1xuICAgICAgICBxdWVyeTogc3RyaW5nO1xuICAgICAgICB1c2VyVHlwZWQ6IGJvb2xlYW47XG4gICAgfSkgPT4gdm9pZDtcbiAgICBvbkZvcm1SZXNwb25zZVN1Ym1pdHRlZDogKGZvcm1SZXNwb25zZTogRm9ybVN1Ym1pdHRlZEF0dHJpYnV0ZXMpID0+IHZvaWQ7XG59XG5leHBvcnQgZGVjbGFyZSBjb25zdCB1c2VTYWxlc0FnZW50Q2hhdEFQSTogKCkgPT4ge1xuICAgIGxvZ1BhZ2VWaXNpdDogKHsgcGFnZVZpc2l0Q2F0ZWdvcnkgfToge1xuICAgICAgICBwYWdlVmlzaXRDYXRlZ29yeTogUGFnZVZpc2l0Q2F0ZWdvcnk7XG4gICAgfSkgPT4gdm9pZDtcbiAgICBsb2dVc2VyRXZlbnQ6IChldmVudDogVXNlckV2ZW50KSA9PiB2b2lkO1xuICAgIG9uU3VnZ2VzdGlvbkNsaWNrZWQ6IChzdWdnZXN0aW9uOiBTdWdnZXN0aW9uLCBkaXNwbGF5TG9jYXRpb246IENoYXRFbGVtZW50RGlzcGxheUxvY2F0aW9uVjMpID0+IHZvaWQ7XG4gICAgb25UeXBlZE1lc3NhZ2VTdWJtaXR0ZWQ6ICh7IHF1ZXJ5LCB1c2VyVHlwZWQgfToge1xuICAgICAgICBxdWVyeTogc3RyaW5nO1xuICAgICAgICB1c2VyVHlwZWQ6IGJvb2xlYW47XG4gICAgfSkgPT4gdm9pZDtcbiAgICBvbkZvcm1SZXNwb25zZVN1Ym1pdHRlZDogKGZvcm06IEZvcm1TdWJtaXR0ZWRBdHRyaWJ1dGVzKSA9PiB2b2lkO1xufTtcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7O0FBRUEsSUFBVyxvQkFBb0I7Q0FBQztPQUFHO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7Q0FBQTtBQUNuQyxJQUFXLHVCQUF1QjtDQUFDO09BQU87T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtDQUFBIn0=
47
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdEFQSS5kLmN0cyIsIm5hbWVzIjpbXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvY29udGV4dHMvc2FsZXNBZ2VudENvbnRleHQvY2hhdEFQSS5kLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFBhZ2VWaXNpdENhdGVnb3J5IH0gZnJvbSAnQHNwaWZmeS1haS9jb21tZXJjZS1hcGktY2xpZW50JztcbmltcG9ydCB7IENoYXRFbGVtZW50RGlzcGxheUxvY2F0aW9uVjMsIEZvcm1TdWJtaXR0ZWRBdHRyaWJ1dGVzLCBTdWdnZXN0aW9uLCBVc2VyRXZlbnQgfSBmcm9tICdzcmMvYXBwbGljYXRpb24vbW9kZWxzJztcbmV4cG9ydCBpbnRlcmZhY2UgU2FsZXNBZ2VudENoYXRBUEkge1xuICAgIGxvZ1BhZ2VWaXNpdDogKHsgcGFnZVZpc2l0Q2F0ZWdvcnkgfToge1xuICAgICAgICBwYWdlVmlzaXRDYXRlZ29yeTogUGFnZVZpc2l0Q2F0ZWdvcnk7XG4gICAgfSkgPT4gdm9pZDtcbiAgICBsb2dVc2VyRXZlbnQ6IChldmVudDogVXNlckV2ZW50KSA9PiB2b2lkO1xuICAgIG9uU3VnZ2VzdGlvbkNsaWNrZWQ6IChzdWdnZXN0aW9uOiBTdWdnZXN0aW9uLCBkaXNwbGF5TG9jYXRpb246IENoYXRFbGVtZW50RGlzcGxheUxvY2F0aW9uVjMpID0+IHZvaWQ7XG4gICAgb25UeXBlZE1lc3NhZ2VTdWJtaXR0ZWQ6ICh7IHF1ZXJ5LCB1c2VyVHlwZWQsIGRpc3BsYXlMb2NhdGlvbiwgfToge1xuICAgICAgICBxdWVyeTogc3RyaW5nO1xuICAgICAgICB1c2VyVHlwZWQ6IGJvb2xlYW47XG4gICAgICAgIGRpc3BsYXlMb2NhdGlvbjogQ2hhdEVsZW1lbnREaXNwbGF5TG9jYXRpb25WMztcbiAgICB9KSA9PiB2b2lkO1xuICAgIG9uRm9ybVJlc3BvbnNlU3VibWl0dGVkOiAoZm9ybTogRm9ybVN1Ym1pdHRlZEF0dHJpYnV0ZXMpID0+IHZvaWQ7XG59XG5leHBvcnQgZGVjbGFyZSBjb25zdCB1c2VTYWxlc0FnZW50Q2hhdEFQSTogKCkgPT4ge1xuICAgIGxvZ1BhZ2VWaXNpdDogKHsgcGFnZVZpc2l0Q2F0ZWdvcnkgfToge1xuICAgICAgICBwYWdlVmlzaXRDYXRlZ29yeTogUGFnZVZpc2l0Q2F0ZWdvcnk7XG4gICAgfSkgPT4gdm9pZDtcbiAgICBsb2dVc2VyRXZlbnQ6IChldmVudDogVXNlckV2ZW50KSA9PiB2b2lkO1xuICAgIG9uU3VnZ2VzdGlvbkNsaWNrZWQ6IChzdWdnZXN0aW9uOiBTdWdnZXN0aW9uLCBkaXNwbGF5TG9jYXRpb246IENoYXRFbGVtZW50RGlzcGxheUxvY2F0aW9uVjMpID0+IHZvaWQ7XG4gICAgb25UeXBlZE1lc3NhZ2VTdWJtaXR0ZWQ6ICh7IHF1ZXJ5LCB1c2VyVHlwZWQsIGRpc3BsYXlMb2NhdGlvbiwgfToge1xuICAgICAgICBxdWVyeTogc3RyaW5nO1xuICAgICAgICB1c2VyVHlwZWQ6IGJvb2xlYW47XG4gICAgICAgIGRpc3BsYXlMb2NhdGlvbjogQ2hhdEVsZW1lbnREaXNwbGF5TG9jYXRpb25WMztcbiAgICB9KSA9PiB2b2lkO1xuICAgIG9uRm9ybVJlc3BvbnNlU3VibWl0dGVkOiAoZm9ybTogRm9ybVN1Ym1pdHRlZEF0dHJpYnV0ZXMpID0+IHZvaWQ7XG59O1xuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7QUFFQSxJQUFXLG9CQUFvQjtDQUFDO09BQUc7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7Q0FBQTtBQUNuQyxJQUFXLHVCQUF1QjtDQUFDO09BQU87T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7Q0FBQSJ9
@@ -14,12 +14,14 @@ interface SalesAgentChatAPI {
14
14
  onSuggestionClicked: (suggestion: Suggestion$1, displayLocation: ChatElementDisplayLocationV3) => void;
15
15
  onTypedMessageSubmitted: ({
16
16
  query,
17
- userTyped
17
+ userTyped,
18
+ displayLocation
18
19
  }: {
19
20
  query: string;
20
21
  userTyped: boolean;
22
+ displayLocation: ChatElementDisplayLocationV3;
21
23
  }) => void;
22
- onFormResponseSubmitted: (formResponse: FormSubmittedAttributes$1) => void;
24
+ onFormResponseSubmitted: (form: FormSubmittedAttributes$1) => void;
23
25
  }
24
26
  declare const useSalesAgentChatAPI: () => {
25
27
  logPageVisit: ({
@@ -31,13 +33,15 @@ declare const useSalesAgentChatAPI: () => {
31
33
  onSuggestionClicked: (suggestion: Suggestion$1, displayLocation: ChatElementDisplayLocationV3) => void;
32
34
  onTypedMessageSubmitted: ({
33
35
  query,
34
- userTyped
36
+ userTyped,
37
+ displayLocation
35
38
  }: {
36
39
  query: string;
37
40
  userTyped: boolean;
41
+ displayLocation: ChatElementDisplayLocationV3;
38
42
  }) => void;
39
43
  onFormResponseSubmitted: (form: FormSubmittedAttributes$1) => void;
40
44
  };
41
45
  //#endregion
42
46
  export { SalesAgentChatAPI, useSalesAgentChatAPI };
43
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdEFQSS5kLnRzIiwibmFtZXMiOltdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb250ZXh0cy9zYWxlc0FnZW50Q29udGV4dC9jaGF0QVBJLmQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUGFnZVZpc2l0Q2F0ZWdvcnkgfSBmcm9tICdAc3BpZmZ5LWFpL2NvbW1lcmNlLWFwaS1jbGllbnQnO1xuaW1wb3J0IHsgQ2hhdEVsZW1lbnREaXNwbGF5TG9jYXRpb25WMywgRm9ybVN1Ym1pdHRlZEF0dHJpYnV0ZXMsIFN1Z2dlc3Rpb24sIFVzZXJFdmVudCB9IGZyb20gJ3NyYy9hcHBsaWNhdGlvbi9tb2RlbHMnO1xuZXhwb3J0IGludGVyZmFjZSBTYWxlc0FnZW50Q2hhdEFQSSB7XG4gICAgbG9nUGFnZVZpc2l0OiAoeyBwYWdlVmlzaXRDYXRlZ29yeSB9OiB7XG4gICAgICAgIHBhZ2VWaXNpdENhdGVnb3J5OiBQYWdlVmlzaXRDYXRlZ29yeTtcbiAgICB9KSA9PiB2b2lkO1xuICAgIGxvZ1VzZXJFdmVudDogKGV2ZW50OiBVc2VyRXZlbnQpID0+IHZvaWQ7XG4gICAgb25TdWdnZXN0aW9uQ2xpY2tlZDogKHN1Z2dlc3Rpb246IFN1Z2dlc3Rpb24sIGRpc3BsYXlMb2NhdGlvbjogQ2hhdEVsZW1lbnREaXNwbGF5TG9jYXRpb25WMykgPT4gdm9pZDtcbiAgICBvblR5cGVkTWVzc2FnZVN1Ym1pdHRlZDogKHsgcXVlcnksIHVzZXJUeXBlZCB9OiB7XG4gICAgICAgIHF1ZXJ5OiBzdHJpbmc7XG4gICAgICAgIHVzZXJUeXBlZDogYm9vbGVhbjtcbiAgICB9KSA9PiB2b2lkO1xuICAgIG9uRm9ybVJlc3BvbnNlU3VibWl0dGVkOiAoZm9ybVJlc3BvbnNlOiBGb3JtU3VibWl0dGVkQXR0cmlidXRlcykgPT4gdm9pZDtcbn1cbmV4cG9ydCBkZWNsYXJlIGNvbnN0IHVzZVNhbGVzQWdlbnRDaGF0QVBJOiAoKSA9PiB7XG4gICAgbG9nUGFnZVZpc2l0OiAoeyBwYWdlVmlzaXRDYXRlZ29yeSB9OiB7XG4gICAgICAgIHBhZ2VWaXNpdENhdGVnb3J5OiBQYWdlVmlzaXRDYXRlZ29yeTtcbiAgICB9KSA9PiB2b2lkO1xuICAgIGxvZ1VzZXJFdmVudDogKGV2ZW50OiBVc2VyRXZlbnQpID0+IHZvaWQ7XG4gICAgb25TdWdnZXN0aW9uQ2xpY2tlZDogKHN1Z2dlc3Rpb246IFN1Z2dlc3Rpb24sIGRpc3BsYXlMb2NhdGlvbjogQ2hhdEVsZW1lbnREaXNwbGF5TG9jYXRpb25WMykgPT4gdm9pZDtcbiAgICBvblR5cGVkTWVzc2FnZVN1Ym1pdHRlZDogKHsgcXVlcnksIHVzZXJUeXBlZCB9OiB7XG4gICAgICAgIHF1ZXJ5OiBzdHJpbmc7XG4gICAgICAgIHVzZXJUeXBlZDogYm9vbGVhbjtcbiAgICB9KSA9PiB2b2lkO1xuICAgIG9uRm9ybVJlc3BvbnNlU3VibWl0dGVkOiAoZm9ybTogRm9ybVN1Ym1pdHRlZEF0dHJpYnV0ZXMpID0+IHZvaWQ7XG59O1xuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7QUFFQSxJQUFXLG9CQUFvQjtDQUFDO09BQUc7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtDQUFBO0FBQ25DLElBQVcsdUJBQXVCO0NBQUM7T0FBTztPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO0NBQUEifQ==
47
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdEFQSS5kLnRzIiwibmFtZXMiOltdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb250ZXh0cy9zYWxlc0FnZW50Q29udGV4dC9jaGF0QVBJLmQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUGFnZVZpc2l0Q2F0ZWdvcnkgfSBmcm9tICdAc3BpZmZ5LWFpL2NvbW1lcmNlLWFwaS1jbGllbnQnO1xuaW1wb3J0IHsgQ2hhdEVsZW1lbnREaXNwbGF5TG9jYXRpb25WMywgRm9ybVN1Ym1pdHRlZEF0dHJpYnV0ZXMsIFN1Z2dlc3Rpb24sIFVzZXJFdmVudCB9IGZyb20gJ3NyYy9hcHBsaWNhdGlvbi9tb2RlbHMnO1xuZXhwb3J0IGludGVyZmFjZSBTYWxlc0FnZW50Q2hhdEFQSSB7XG4gICAgbG9nUGFnZVZpc2l0OiAoeyBwYWdlVmlzaXRDYXRlZ29yeSB9OiB7XG4gICAgICAgIHBhZ2VWaXNpdENhdGVnb3J5OiBQYWdlVmlzaXRDYXRlZ29yeTtcbiAgICB9KSA9PiB2b2lkO1xuICAgIGxvZ1VzZXJFdmVudDogKGV2ZW50OiBVc2VyRXZlbnQpID0+IHZvaWQ7XG4gICAgb25TdWdnZXN0aW9uQ2xpY2tlZDogKHN1Z2dlc3Rpb246IFN1Z2dlc3Rpb24sIGRpc3BsYXlMb2NhdGlvbjogQ2hhdEVsZW1lbnREaXNwbGF5TG9jYXRpb25WMykgPT4gdm9pZDtcbiAgICBvblR5cGVkTWVzc2FnZVN1Ym1pdHRlZDogKHsgcXVlcnksIHVzZXJUeXBlZCwgZGlzcGxheUxvY2F0aW9uLCB9OiB7XG4gICAgICAgIHF1ZXJ5OiBzdHJpbmc7XG4gICAgICAgIHVzZXJUeXBlZDogYm9vbGVhbjtcbiAgICAgICAgZGlzcGxheUxvY2F0aW9uOiBDaGF0RWxlbWVudERpc3BsYXlMb2NhdGlvblYzO1xuICAgIH0pID0+IHZvaWQ7XG4gICAgb25Gb3JtUmVzcG9uc2VTdWJtaXR0ZWQ6IChmb3JtOiBGb3JtU3VibWl0dGVkQXR0cmlidXRlcykgPT4gdm9pZDtcbn1cbmV4cG9ydCBkZWNsYXJlIGNvbnN0IHVzZVNhbGVzQWdlbnRDaGF0QVBJOiAoKSA9PiB7XG4gICAgbG9nUGFnZVZpc2l0OiAoeyBwYWdlVmlzaXRDYXRlZ29yeSB9OiB7XG4gICAgICAgIHBhZ2VWaXNpdENhdGVnb3J5OiBQYWdlVmlzaXRDYXRlZ29yeTtcbiAgICB9KSA9PiB2b2lkO1xuICAgIGxvZ1VzZXJFdmVudDogKGV2ZW50OiBVc2VyRXZlbnQpID0+IHZvaWQ7XG4gICAgb25TdWdnZXN0aW9uQ2xpY2tlZDogKHN1Z2dlc3Rpb246IFN1Z2dlc3Rpb24sIGRpc3BsYXlMb2NhdGlvbjogQ2hhdEVsZW1lbnREaXNwbGF5TG9jYXRpb25WMykgPT4gdm9pZDtcbiAgICBvblR5cGVkTWVzc2FnZVN1Ym1pdHRlZDogKHsgcXVlcnksIHVzZXJUeXBlZCwgZGlzcGxheUxvY2F0aW9uLCB9OiB7XG4gICAgICAgIHF1ZXJ5OiBzdHJpbmc7XG4gICAgICAgIHVzZXJUeXBlZDogYm9vbGVhbjtcbiAgICAgICAgZGlzcGxheUxvY2F0aW9uOiBDaGF0RWxlbWVudERpc3BsYXlMb2NhdGlvblYzO1xuICAgIH0pID0+IHZvaWQ7XG4gICAgb25Gb3JtUmVzcG9uc2VTdWJtaXR0ZWQ6IChmb3JtOiBGb3JtU3VibWl0dGVkQXR0cmlidXRlcykgPT4gdm9pZDtcbn07XG4iXSwibWFwcGluZ3MiOiI7Ozs7OztBQUVBLElBQVcsb0JBQW9CO0NBQUM7T0FBRztPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtDQUFBO0FBQ25DLElBQVcsdUJBQXVCO0NBQUM7T0FBTztPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtDQUFBIn0=
@@ -1,26 +1,46 @@
1
+ import logger_default from "../../application/logging/logger.js";
1
2
  import { MessageRole, MessageType } from "../../application/models/message.js";
2
3
  import "../../application/models/index.js";
3
- import { SpiffyMetricsEventName } from "../../services/amplitudeService/amplitudeService.js";
4
+ import { analyticsContextAtom, hasParsedVariantInfoAtom } from "../../atoms/app/variant.js";
5
+ import "../../atoms/app/index.js";
6
+ import { EnviveMetricsEventName, SpiffyMetricsEventName } from "../../services/amplitudeService/amplitudeService.js";
4
7
  import { useAmplitude } from "../amplitudeContext/amplitudeContext.js";
5
- import "../amplitudeContext/index.js";
6
8
  import { formSubmitAtom, replyEventCategoryAtom } from "../../atoms/chat/chatState.js";
7
9
  import { queueUserEventAtom } from "../../atoms/chat/messageQueue.js";
8
10
  import "../../atoms/chat/index.js";
11
+ import "../amplitudeContext/index.js";
9
12
  import { UserEventCategory } from "@spiffy-ai/commerce-api-client";
10
- import { v4 } from "uuid";
11
13
  import { useCallback } from "react";
12
- import { useSetAtom } from "jotai";
14
+ import { useAtomValue, useSetAtom } from "jotai";
15
+ import { v4 } from "uuid";
13
16
 
14
17
  //#region src/contexts/salesAgentContext/chatAPI.ts
18
+ const TRACKED_USER_EVENTS = [
19
+ UserEventCategory.PageVisit,
20
+ UserEventCategory.PdpVisit,
21
+ UserEventCategory.PlpVisit
22
+ ];
15
23
  const useSalesAgentChatAPI = () => {
24
+ const context = useAtomValue(analyticsContextAtom);
25
+ const hasParsedVariantInfo = useAtomValue(hasParsedVariantInfoAtom);
16
26
  const queueUserEvent = useSetAtom(queueUserEventAtom);
17
27
  const setReplyEventCategory = useSetAtom(replyEventCategoryAtom);
18
28
  const setFormSubmit = useSetAtom(formSubmitAtom);
19
29
  const { trackEvent } = useAmplitude();
20
30
  return {
21
31
  logPageVisit: useCallback(({ pageVisitCategory }) => {
32
+ const eventId = v4();
33
+ console.log("=== logPageVisit ===", eventId);
34
+ trackEvent({
35
+ eventName: EnviveMetricsEventName.ChatRequest,
36
+ eventProps: {
37
+ ...context,
38
+ "chat.request_id": eventId,
39
+ "chat.request_type": "page_visit"
40
+ }
41
+ });
22
42
  queueUserEvent({
23
- eventId: v4(),
43
+ eventId,
24
44
  category: UserEventCategory.PageVisit,
25
45
  createdAt: (/* @__PURE__ */ new Date()).toISOString(),
26
46
  attributes: {
@@ -28,10 +48,31 @@ const useSalesAgentChatAPI = () => {
28
48
  pageVisitCategory
29
49
  }
30
50
  });
31
- }, [queueUserEvent]),
51
+ }, [
52
+ queueUserEvent,
53
+ context,
54
+ trackEvent
55
+ ]),
32
56
  logUserEvent: useCallback((event) => {
57
+ if (!hasParsedVariantInfo) {
58
+ logger_default.logWarn("[envive-ai] hasParsedVariantInfo is false, not logging user event", void 0, { event });
59
+ return;
60
+ }
61
+ if (TRACKED_USER_EVENTS.includes(event.category)) trackEvent({
62
+ eventName: EnviveMetricsEventName.ChatRequest,
63
+ eventProps: {
64
+ ...context,
65
+ "chat.request_id": event.eventId,
66
+ "chat.request_type": "page_visit"
67
+ }
68
+ });
33
69
  queueUserEvent(event);
34
- }, [queueUserEvent]),
70
+ }, [
71
+ context,
72
+ hasParsedVariantInfo,
73
+ queueUserEvent,
74
+ trackEvent
75
+ ]),
35
76
  onSuggestionClicked: useCallback((suggestion, displayLocation) => {
36
77
  trackEvent({
37
78
  eventName: SpiffyMetricsEventName.ChatSuggestionClicked,
@@ -48,8 +89,24 @@ const useSalesAgentChatAPI = () => {
48
89
  }
49
90
  }
50
91
  });
92
+ const eventId = v4();
93
+ trackEvent({
94
+ eventName: EnviveMetricsEventName.ChatRequest,
95
+ eventProps: {
96
+ ...context,
97
+ "trigger.widget": displayLocation,
98
+ "chat.request_id": eventId,
99
+ "chat.request_type": "suggestion",
100
+ "chat.request_text": suggestion.content,
101
+ "chat.suggestion_id": suggestion.id,
102
+ "chat.suggestion_category": suggestion.category,
103
+ "chat.suggestion_created_at": suggestion.createdAt,
104
+ "chat.suggestion_is_answer": suggestion.isAnswer,
105
+ "chat.user_typed": false
106
+ }
107
+ });
51
108
  queueUserEvent({
52
- eventId: v4(),
109
+ eventId,
53
110
  category: UserEventCategory.SuggestionClicked,
54
111
  createdAt: (/* @__PURE__ */ new Date()).toISOString(),
55
112
  attributes: {
@@ -57,8 +114,12 @@ const useSalesAgentChatAPI = () => {
57
114
  content: suggestion.content
58
115
  }
59
116
  });
60
- }, [queueUserEvent, trackEvent]),
61
- onTypedMessageSubmitted: useCallback(({ query, userTyped }) => {
117
+ }, [
118
+ queueUserEvent,
119
+ trackEvent,
120
+ context
121
+ ]),
122
+ onTypedMessageSubmitted: useCallback(({ query, userTyped, displayLocation }) => {
62
123
  const eventId = v4();
63
124
  trackEvent({
64
125
  eventName: SpiffyMetricsEventName.ChatUserMessageInput,
@@ -75,6 +136,18 @@ const useSalesAgentChatAPI = () => {
75
136
  }
76
137
  }
77
138
  });
139
+ trackEvent({
140
+ eventName: EnviveMetricsEventName.ChatRequest,
141
+ eventProps: {
142
+ ...context,
143
+ "trigger.widget": displayLocation,
144
+ "chat.request_type": "text",
145
+ "chat.request_text": query,
146
+ "chat.request_id": eventId,
147
+ "chat.request_created_at": (/* @__PURE__ */ new Date()).toISOString(),
148
+ "chat.user_typed": userTyped
149
+ }
150
+ });
78
151
  queueUserEvent({
79
152
  eventId,
80
153
  category: UserEventCategory.QueryTyped,
@@ -84,24 +157,41 @@ const useSalesAgentChatAPI = () => {
84
157
  userTyped
85
158
  }
86
159
  });
87
- }, [queueUserEvent, trackEvent]),
160
+ }, [
161
+ queueUserEvent,
162
+ trackEvent,
163
+ context
164
+ ]),
88
165
  onFormResponseSubmitted: useCallback((form) => {
89
166
  setReplyEventCategory(UserEventCategory.FormSubmitted);
90
167
  setFormSubmit(form);
168
+ const eventId = v4();
169
+ trackEvent({
170
+ eventName: EnviveMetricsEventName.ChatRequest,
171
+ eventProps: {
172
+ ...context,
173
+ "chat.request_id": eventId,
174
+ "chat.request_type": "form",
175
+ "chat.form_response_id": form.formResponseId,
176
+ "chat.form_type": form.formType
177
+ }
178
+ });
91
179
  queueUserEvent({
92
- eventId: v4(),
180
+ eventId,
93
181
  category: UserEventCategory.FormSubmitted,
94
182
  createdAt: (/* @__PURE__ */ new Date()).toISOString(),
95
183
  attributes: form
96
184
  });
97
185
  }, [
98
- queueUserEvent,
99
186
  setReplyEventCategory,
100
- setFormSubmit
187
+ setFormSubmit,
188
+ trackEvent,
189
+ context,
190
+ queueUserEvent
101
191
  ])
102
192
  };
103
193
  };
104
194
 
105
195
  //#endregion
106
196
  export { useSalesAgentChatAPI };
107
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"chatAPI.js","names":["uuid"],"sources":["../../../src/contexts/salesAgentContext/chatAPI.ts"],"sourcesContent":["// This component will interact with the backend API to get the responses from the sales agent.\n\nimport { PageVisitCategory, UserEventCategory } from '@spiffy-ai/commerce-api-client';\nimport {\n  ChatElementDisplayLocationV3,\n  FormSubmittedAttributes,\n  MessageRole,\n  MessageType,\n  Suggestion,\n  UserEvent,\n} from 'src/application/models';\nimport { useSetAtom } from 'jotai';\nimport { useCallback } from 'react';\nimport { v4 as uuid } from 'uuid';\nimport { formSubmitAtom, replyEventCategoryAtom } from 'src/atoms/chat';\nimport { queueUserEventAtom } from 'src/atoms/chat/messageQueue';\nimport { SpiffyMetricsEventName, useAmplitude } from '../amplitudeContext';\n\nexport interface SalesAgentChatAPI {\n  logPageVisit: ({ pageVisitCategory }: { pageVisitCategory: PageVisitCategory }) => void;\n  logUserEvent: (event: UserEvent) => void;\n  onSuggestionClicked: (\n    suggestion: Suggestion,\n    displayLocation: ChatElementDisplayLocationV3,\n  ) => void;\n  onTypedMessageSubmitted: ({ query, userTyped }: { query: string; userTyped: boolean }) => void;\n  onFormResponseSubmitted: (formResponse: FormSubmittedAttributes) => void;\n}\n\nexport const useSalesAgentChatAPI = () => {\n  // TODO: Each of these functions will trigger both the necessary amplitude events and initiate the\n  // necessary actions to trigger the backend API\n  const queueUserEvent = useSetAtom(queueUserEventAtom);\n  const setReplyEventCategory = useSetAtom(replyEventCategoryAtom);\n  const setFormSubmit = useSetAtom(formSubmitAtom);\n  const { trackEvent } = useAmplitude();\n\n  const logPageVisit = useCallback(\n    ({ pageVisitCategory }: { pageVisitCategory: PageVisitCategory }) => {\n      const event: UserEvent = {\n        eventId: uuid(),\n        category: UserEventCategory.PageVisit,\n        createdAt: new Date().toISOString(),\n        attributes: {\n          url: window.location.href,\n          pageVisitCategory,\n        },\n      };\n      queueUserEvent(event);\n    },\n    [queueUserEvent],\n  );\n\n  const logUserEvent = useCallback(\n    (event: UserEvent) => {\n      queueUserEvent(event);\n    },\n    [queueUserEvent],\n  );\n  const onSuggestionClicked = useCallback(\n    (suggestion: Suggestion, displayLocation: ChatElementDisplayLocationV3) => {\n      trackEvent({\n        eventName: SpiffyMetricsEventName.ChatSuggestionClicked,\n        eventProps: {\n          message_id: suggestion.id,\n          message_role: MessageRole.User,\n          message_type: MessageType.SuggestionClicked,\n          message_metadata: {\n            content: suggestion.content,\n            created_at: suggestion.createdAt,\n            category: suggestion.category,\n            is_answer: suggestion.isAnswer,\n            display_location: displayLocation,\n          },\n        },\n      });\n      const event: UserEvent = {\n        eventId: uuid(),\n        category: UserEventCategory.SuggestionClicked,\n        createdAt: new Date().toISOString(),\n        attributes: {\n          suggestionId: suggestion.id,\n          content: suggestion.content,\n        },\n      };\n      queueUserEvent(event);\n    },\n    [queueUserEvent, trackEvent],\n  );\n  const onTypedMessageSubmitted = useCallback(\n    ({ query, userTyped }: { query: string; userTyped: boolean }) => {\n      const eventId = uuid();\n      trackEvent({\n        eventName: SpiffyMetricsEventName.ChatUserMessageInput,\n        eventProps: {\n          query,\n          user_typed: userTyped,\n          message_id: eventId,\n          message_role: MessageRole.User,\n          message_type: MessageType.QueryTyped,\n          message_metadata: {\n            content: query,\n            created_at: new Date().toISOString(),\n            user_typed: userTyped,\n          },\n        },\n      });\n      const event: UserEvent = {\n        eventId,\n        category: UserEventCategory.QueryTyped,\n        createdAt: new Date().toISOString(),\n        attributes: {\n          query,\n          userTyped,\n        },\n      };\n      queueUserEvent(event);\n    },\n    [queueUserEvent, trackEvent],\n  );\n  const onFormResponseSubmitted = useCallback(\n    (form: FormSubmittedAttributes) => {\n      setReplyEventCategory(UserEventCategory.FormSubmitted);\n      setFormSubmit(form);\n      const event: UserEvent = {\n        eventId: uuid(),\n        category: UserEventCategory.FormSubmitted,\n        createdAt: new Date().toISOString(),\n        attributes: form,\n      };\n      queueUserEvent(event);\n    },\n    [queueUserEvent, setReplyEventCategory, setFormSubmit],\n  );\n\n  return {\n    logPageVisit,\n    logUserEvent,\n    onSuggestionClicked,\n    onTypedMessageSubmitted,\n    onFormResponseSubmitted,\n  };\n};\n"],"mappings":";;;;;;;;;;;;;;AA6BA,MAAa,6BAA6B;CAGxC,MAAM,iBAAiB,WAAW,mBAAmB;CACrD,MAAM,wBAAwB,WAAW,uBAAuB;CAChE,MAAM,gBAAgB,WAAW,eAAe;CAChD,MAAM,EAAE,eAAe,cAAc;AAoGrC,QAAO;EACL,cAnGmB,aAClB,EAAE,wBAAkE;AAUnE,kBATyB;IACvB,SAASA,IAAM;IACf,UAAU,kBAAkB;IAC5B,4BAAW,IAAI,MAAM,EAAC,aAAa;IACnC,YAAY;KACV,KAAK,OAAO,SAAS;KACrB;KACD;IACF,CACoB;KAEvB,CAAC,eAAe,CACjB;EAsFC,cApFmB,aAClB,UAAqB;AACpB,kBAAe,MAAM;KAEvB,CAAC,eAAe,CACjB;EAgFC,qBA/E0B,aACzB,YAAwB,oBAAkD;AACzE,cAAW;IACT,WAAW,uBAAuB;IAClC,YAAY;KACV,YAAY,WAAW;KACvB,cAAc,YAAY;KAC1B,cAAc,YAAY;KAC1B,kBAAkB;MAChB,SAAS,WAAW;MACpB,YAAY,WAAW;MACvB,UAAU,WAAW;MACrB,WAAW,WAAW;MACtB,kBAAkB;MACnB;KACF;IACF,CAAC;AAUF,kBATyB;IACvB,SAASA,IAAM;IACf,UAAU,kBAAkB;IAC5B,4BAAW,IAAI,MAAM,EAAC,aAAa;IACnC,YAAY;KACV,cAAc,WAAW;KACzB,SAAS,WAAW;KACrB;IACF,CACoB;KAEvB,CAAC,gBAAgB,WAAW,CAC7B;EAmDC,yBAlD8B,aAC7B,EAAE,OAAO,gBAAuD;GAC/D,MAAM,UAAUA,IAAM;AACtB,cAAW;IACT,WAAW,uBAAuB;IAClC,YAAY;KACV;KACA,YAAY;KACZ,YAAY;KACZ,cAAc,YAAY;KAC1B,cAAc,YAAY;KAC1B,kBAAkB;MAChB,SAAS;MACT,6BAAY,IAAI,MAAM,EAAC,aAAa;MACpC,YAAY;MACb;KACF;IACF,CAAC;AAUF,kBATyB;IACvB;IACA,UAAU,kBAAkB;IAC5B,4BAAW,IAAI,MAAM,EAAC,aAAa;IACnC,YAAY;KACV;KACA;KACD;IACF,CACoB;KAEvB,CAAC,gBAAgB,WAAW,CAC7B;EAqBC,yBApB8B,aAC7B,SAAkC;AACjC,yBAAsB,kBAAkB,cAAc;AACtD,iBAAc,KAAK;AAOnB,kBANyB;IACvB,SAASA,IAAM;IACf,UAAU,kBAAkB;IAC5B,4BAAW,IAAI,MAAM,EAAC,aAAa;IACnC,YAAY;IACb,CACoB;KAEvB;GAAC;GAAgB;GAAuB;GAAc,CACvD;EAQA"}
197
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"chatAPI.js","names":["uuid"],"sources":["../../../src/contexts/salesAgentContext/chatAPI.ts"],"sourcesContent":["// This component will interact with the backend API to get the responses from the sales agent.\nimport Logger from 'src/application/logging/logger';\nimport { PageVisitCategory, UserEventCategory } from '@spiffy-ai/commerce-api-client';\nimport {\n  ChatElementDisplayLocationV3,\n  FormSubmittedAttributes,\n  MessageRole,\n  MessageType,\n  Suggestion,\n  UserEvent,\n} from 'src/application/models';\nimport { useAtomValue, useSetAtom } from 'jotai';\nimport { useCallback } from 'react';\nimport { v4 as uuid } from 'uuid';\nimport { formSubmitAtom, replyEventCategoryAtom } from 'src/atoms/chat';\nimport { queueUserEventAtom } from 'src/atoms/chat/messageQueue';\nimport { EnviveMetricsEventName } from 'src/services/amplitudeService/amplitudeService';\nimport { analyticsContextAtom } from 'src/atoms/app/variant';\nimport { hasParsedVariantInfoAtom } from 'src/atoms/app';\nimport { SpiffyMetricsEventName, useAmplitude } from '../amplitudeContext';\n\nconst TRACKED_USER_EVENTS = [\n  UserEventCategory.PageVisit,\n  UserEventCategory.PdpVisit,\n  UserEventCategory.PlpVisit,\n];\nexport interface SalesAgentChatAPI {\n  logPageVisit: ({ pageVisitCategory }: { pageVisitCategory: PageVisitCategory }) => void;\n  logUserEvent: (event: UserEvent) => void;\n  onSuggestionClicked: (\n    suggestion: Suggestion,\n    displayLocation: ChatElementDisplayLocationV3,\n  ) => void;\n  onTypedMessageSubmitted: ({\n    query,\n    userTyped,\n    displayLocation,\n  }: {\n    query: string;\n    userTyped: boolean;\n    displayLocation: ChatElementDisplayLocationV3;\n  }) => void;\n  onFormResponseSubmitted: (form: FormSubmittedAttributes) => void;\n}\n\nexport const useSalesAgentChatAPI = () => {\n  // TODO: Each of these functions will trigger both the necessary amplitude events and initiate the\n  // necessary actions to trigger the backend API\n  const context = useAtomValue(analyticsContextAtom);\n  const hasParsedVariantInfo = useAtomValue(hasParsedVariantInfoAtom);\n  const queueUserEvent = useSetAtom(queueUserEventAtom);\n  const setReplyEventCategory = useSetAtom(replyEventCategoryAtom);\n  const setFormSubmit = useSetAtom(formSubmitAtom);\n  const { trackEvent } = useAmplitude();\n\n  const logPageVisit = useCallback(\n    ({ pageVisitCategory }: { pageVisitCategory: PageVisitCategory }) => {\n      const eventId = uuid();\n      console.log('=== logPageVisit ===', eventId);\n      trackEvent({\n        eventName: EnviveMetricsEventName.ChatRequest,\n        eventProps: {\n          ...context,\n          'chat.request_id': eventId,\n          'chat.request_type': 'page_visit',\n        },\n      });\n      const event: UserEvent = {\n        eventId,\n        category: UserEventCategory.PageVisit,\n        createdAt: new Date().toISOString(),\n        attributes: {\n          url: window.location.href,\n          pageVisitCategory,\n        },\n      };\n      queueUserEvent(event);\n    },\n    [queueUserEvent, context, trackEvent],\n  );\n\n  const logUserEvent = useCallback(\n    (event: UserEvent) => {\n      if (!hasParsedVariantInfo) {\n        Logger.logWarn(\n          '[envive-ai] hasParsedVariantInfo is false, not logging user event',\n          undefined,\n          { event },\n        );\n        return;\n      }\n      if (TRACKED_USER_EVENTS.includes(event.category)) {\n        trackEvent({\n          eventName: EnviveMetricsEventName.ChatRequest,\n          eventProps: {\n            ...context,\n            'chat.request_id': event.eventId,\n            'chat.request_type': 'page_visit',\n          },\n        });\n      }\n      queueUserEvent(event);\n    },\n    [context, hasParsedVariantInfo, queueUserEvent, trackEvent],\n  );\n  const onSuggestionClicked = useCallback(\n    (suggestion: Suggestion, displayLocation: ChatElementDisplayLocationV3) => {\n      trackEvent({\n        eventName: SpiffyMetricsEventName.ChatSuggestionClicked,\n        eventProps: {\n          message_id: suggestion.id,\n          message_role: MessageRole.User,\n          message_type: MessageType.SuggestionClicked,\n          message_metadata: {\n            content: suggestion.content,\n            created_at: suggestion.createdAt,\n            category: suggestion.category,\n            is_answer: suggestion.isAnswer,\n            display_location: displayLocation,\n          },\n        },\n      });\n      const eventId = uuid();\n      trackEvent({\n        eventName: EnviveMetricsEventName.ChatRequest,\n        eventProps: {\n          ...context,\n          'trigger.widget': displayLocation, // This is fairly finegrained, but that may be ok\n          'chat.request_id': eventId,\n          'chat.request_type': 'suggestion',\n          'chat.request_text': suggestion.content,\n          'chat.suggestion_id': suggestion.id,\n          'chat.suggestion_category': suggestion.category,\n          'chat.suggestion_created_at': suggestion.createdAt,\n          'chat.suggestion_is_answer': suggestion.isAnswer,\n          'chat.user_typed': false,\n        },\n      });\n      const event: UserEvent = {\n        eventId,\n        category: UserEventCategory.SuggestionClicked,\n        createdAt: new Date().toISOString(),\n        attributes: {\n          suggestionId: suggestion.id,\n          content: suggestion.content,\n        },\n      };\n      queueUserEvent(event);\n    },\n    [queueUserEvent, trackEvent, context],\n  );\n  const onTypedMessageSubmitted = useCallback(\n    ({\n      query,\n      userTyped,\n      displayLocation,\n    }: {\n      query: string;\n      userTyped: boolean;\n      displayLocation: ChatElementDisplayLocationV3;\n    }) => {\n      const eventId = uuid();\n      trackEvent({\n        eventName: SpiffyMetricsEventName.ChatUserMessageInput,\n        eventProps: {\n          query,\n          user_typed: userTyped,\n          message_id: eventId,\n          message_role: MessageRole.User,\n          message_type: MessageType.QueryTyped,\n          message_metadata: {\n            content: query,\n            created_at: new Date().toISOString(),\n            user_typed: userTyped,\n          },\n        },\n      });\n      trackEvent({\n        eventName: EnviveMetricsEventName.ChatRequest,\n        eventProps: {\n          ...context,\n          'trigger.widget': displayLocation,\n          'chat.request_type': 'text',\n          'chat.request_text': query,\n          'chat.request_id': eventId,\n          'chat.request_created_at': new Date().toISOString(),\n          'chat.user_typed': userTyped,\n        },\n      });\n      const event: UserEvent = {\n        eventId,\n        category: UserEventCategory.QueryTyped,\n        createdAt: new Date().toISOString(),\n        attributes: {\n          query,\n          userTyped,\n        },\n      };\n      queueUserEvent(event);\n    },\n    [queueUserEvent, trackEvent, context],\n  );\n  const onFormResponseSubmitted = useCallback(\n    (form: FormSubmittedAttributes) => {\n      setReplyEventCategory(UserEventCategory.FormSubmitted);\n      setFormSubmit(form);\n      const eventId = uuid();\n      trackEvent({\n        eventName: EnviveMetricsEventName.ChatRequest,\n        eventProps: {\n          ...context,\n          'chat.request_id': eventId,\n          'chat.request_type': 'form',\n          'chat.form_response_id': form.formResponseId,\n          'chat.form_type': form.formType,\n        },\n      });\n      const event: UserEvent = {\n        eventId,\n        category: UserEventCategory.FormSubmitted,\n        createdAt: new Date().toISOString(),\n        attributes: form,\n      };\n      queueUserEvent(event);\n    },\n    [setReplyEventCategory, setFormSubmit, trackEvent, context, queueUserEvent],\n  );\n\n  return {\n    logPageVisit,\n    logUserEvent,\n    onSuggestionClicked,\n    onTypedMessageSubmitted,\n    onFormResponseSubmitted,\n  };\n};\n"],"mappings":"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"}
@@ -1,6 +1,6 @@
1
1
  const require_rolldown_runtime = require('../../_virtual/rolldown_runtime.cjs');
2
- const require_message = require('../../application/models/message.cjs');
3
2
  const require_logger = require('../../application/logging/logger.cjs');
3
+ const require_message = require('../../application/models/message.cjs');
4
4
  const require_atoms_chat_chatState = require('../../atoms/chat/chatState.cjs');
5
5
  const require_messageQueue = require('../../atoms/chat/messageQueue.cjs');
6
6
  const require_useSystemSettingsContext = require('../../hooks/SystemSettingsContext/useSystemSettingsContext.cjs');
@@ -1,5 +1,5 @@
1
- import { MessageRole, MessageType } from "../../application/models/message.js";
2
1
  import logger_default from "../../application/logging/logger.js";
2
+ import { MessageRole, MessageType } from "../../application/models/message.js";
3
3
  import { initializedAtom, messagesAtom, pendingResponseAtom, responseStreamingAtom, suggestionsAtom } from "../../atoms/chat/chatState.js";
4
4
  import { processUserEventAtom, userEventQueueAtom, userQueueEventCountAtom } from "../../atoms/chat/messageQueue.js";
5
5
  import { useSystemSettingsContext } from "../../hooks/SystemSettingsContext/useSystemSettingsContext.js";
@@ -1,23 +1,24 @@
1
1
  const require_rolldown_runtime = require('../../_virtual/rolldown_runtime.cjs');
2
- const require_useMessageInterceptor = require('../../interceptors/useMessageInterceptor.cjs');
3
- const require_message = require('../../application/models/message.cjs');
4
2
  const require_logger = require('../../application/logging/logger.cjs');
3
+ const require_message = require('../../application/models/message.cjs');
5
4
  require('../../application/models/index.cjs');
5
+ const require_variant = require('../../atoms/app/variant.cjs');
6
6
  const require_atoms_app_index = require('../../atoms/app/index.cjs');
7
7
  const require_featureFlagServiceContext = require('../featureFlagServiceContext/featureFlagServiceContext.cjs');
8
8
  const require_amplitudeService = require('../../services/amplitudeService/amplitudeService.cjs');
9
9
  const require_amplitudeContext = require('../amplitudeContext/amplitudeContext.cjs');
10
10
  const require_messageFromResponse = require('../../application/utils/messageFromResponse.cjs');
11
11
  require('../../application/utils/index.cjs');
12
+ const require_application_commerce_api = require('../../application/commerce-api.cjs');
13
+ const require_useMessageInterceptor = require('../../interceptors/useMessageInterceptor.cjs');
14
+ const require_atoms_chat_chatState = require('../../atoms/chat/chatState.cjs');
12
15
  require('../amplitudeContext/index.cjs');
13
16
  require('../featureFlagServiceContext/index.cjs');
14
- const require_commerce_api = require('../../application/commerce-api.cjs');
15
- const require_atoms_chat_chatState = require('../../atoms/chat/chatState.cjs');
16
17
  const require_statusCodeError = require('./statusCodeError.cjs');
17
18
  let __spiffy_ai_commerce_api_client = require("@spiffy-ai/commerce-api-client");
18
- let uuid = require("uuid");
19
19
  let react = require("react");
20
20
  let jotai = require("jotai");
21
+ let uuid = require("uuid");
21
22
 
22
23
  //#region src/contexts/salesAgentContext/salesAgentService.ts
23
24
  const inputPropsToTrackingProps = (payload) => {
@@ -34,6 +35,27 @@ const inputPropsToTrackingProps = (payload) => {
34
35
  };
35
36
  return {};
36
37
  };
38
+ const inputPropsToEnviveTrackingProps = (payload) => {
39
+ const [userEvent] = payload.userEvents || [];
40
+ if (userEvent.category === __spiffy_ai_commerce_api_client.UserEventCategory.SuggestionClicked) return {
41
+ "chat.request_type": "suggestion",
42
+ "chat.request_id": userEvent.eventId,
43
+ "chat.request_text": userEvent.attributes.content,
44
+ "chat.suggestion_id": userEvent.attributes.suggestionId,
45
+ "chat.user_typed": false
46
+ };
47
+ if (userEvent.category === __spiffy_ai_commerce_api_client.UserEventCategory.QueryTyped) return {
48
+ "chat.request_type": "text",
49
+ "chat.request_id": userEvent.eventId,
50
+ "chat.request_text": userEvent.attributes.query,
51
+ "chat.user_typed": userEvent.attributes.userTyped
52
+ };
53
+ if (userEvent.category === __spiffy_ai_commerce_api_client.UserEventCategory.PageVisit || userEvent.category === __spiffy_ai_commerce_api_client.UserEventCategory.PdpVisit || userEvent.category === __spiffy_ai_commerce_api_client.UserEventCategory.PlpVisit) return {
54
+ "chat.request_id": userEvent.eventId,
55
+ "chat.request_type": "page_visit"
56
+ };
57
+ return {};
58
+ };
37
59
  const getQueryParam = (key) => {
38
60
  return new URL(window.location.href).searchParams.get(key);
39
61
  };
@@ -79,12 +101,15 @@ const updateMessageState = (message, lastMessage, setMessages) => {
79
101
  const processStreamingResponse = async (stream, messageInterceptor, setMessages, setResponseStreaming) => {
80
102
  let lastMessage;
81
103
  setResponseStreaming(true);
104
+ const responseAnalytics = {};
82
105
  for await (const response of stream) {
83
106
  try {
84
- if (messageInterceptor.intercept(response)) return;
107
+ if (messageInterceptor.intercept(response)) return {};
85
108
  if (!response) throw new Error("No response from stream");
86
109
  const message = require_messageFromResponse.messageFromResponse(response);
87
110
  if (!message) throw new Error("Failed to transform API response to client message");
111
+ if (message.type === require_message.MessageType.Product) responseAnalytics["chat.product_cards_returned"] = (responseAnalytics["chat.product_cards_returned"] || 0) + 1;
112
+ if (message.type === require_message.MessageType.Review) responseAnalytics["chat.review_cards_returned"] = (responseAnalytics["chat.review_cards_returned"] || 0) + 1;
88
113
  lastMessage = updateMessageState(message, lastMessage, setMessages);
89
114
  } catch (error) {
90
115
  require_logger.default.logWarn(`[spiffy-ai] Failed to generate responses from stream`, error, {
@@ -94,6 +119,7 @@ const processStreamingResponse = async (stream, messageInterceptor, setMessages,
94
119
  }
95
120
  setResponseStreaming(false);
96
121
  }
122
+ return responseAnalytics;
97
123
  };
98
124
  const useSalesAgentService = () => {
99
125
  const setRequestFailure = (0, jotai.useSetAtom)(require_atoms_chat_chatState.requestFailureAtom);
@@ -101,6 +127,7 @@ const useSalesAgentService = () => {
101
127
  const setSuggestions = (0, jotai.useSetAtom)(require_atoms_chat_chatState.suggestionsAtom);
102
128
  const setPendingResponse = (0, jotai.useSetAtom)(require_atoms_chat_chatState.pendingResponseAtom);
103
129
  const setResponseStreaming = (0, jotai.useSetAtom)(require_atoms_chat_chatState.responseStreamingAtom);
130
+ const analyticsContext = (0, jotai.useAtomValue)(require_variant.analyticsContextAtom);
104
131
  const messageInterceptor = require_useMessageInterceptor.useMessageInterceptor();
105
132
  const { trackEvent } = require_amplitudeContext.useAmplitude();
106
133
  const featureFlagService = require_featureFlagServiceContext.useFeatureFlagService();
@@ -124,10 +151,12 @@ const useSalesAgentService = () => {
124
151
  const startTime = Date.now();
125
152
  let successfulResponse;
126
153
  setPendingResponse(true);
127
- const stream = require_commerce_api.default.getNextResponseStreaming(payload);
154
+ setSuggestions([]);
155
+ const stream = require_application_commerce_api.default.getNextResponseStreaming(payload);
156
+ let responseAnalytics;
128
157
  try {
129
158
  setRequestFailure(false);
130
- await processStreamingResponse(stream, messageInterceptor, setMessages, setResponseStreaming);
159
+ responseAnalytics = await processStreamingResponse(stream, messageInterceptor, setMessages, setResponseStreaming);
131
160
  successfulResponse = true;
132
161
  } catch (e) {
133
162
  console.error("error getting streaming responses", e);
@@ -144,21 +173,34 @@ const useSalesAgentService = () => {
144
173
  ...inputPropsToTrackingProps(payload)
145
174
  }
146
175
  });
176
+ trackEvent({
177
+ eventName: require_amplitudeService.EnviveMetricsEventName.ChatResponse,
178
+ eventProps: {
179
+ ...analyticsContext,
180
+ "chat.response_time_ms": responseTime.toString(),
181
+ ...inputPropsToEnviveTrackingProps(payload),
182
+ ...responseAnalytics
183
+ }
184
+ });
147
185
  setPendingResponse(false);
148
186
  setResponseStreaming(false);
149
187
  }
150
188
  }, [
189
+ analyticsContext,
190
+ trackEvent,
191
+ setResponseStreaming,
151
192
  setRequestFailure,
193
+ setSuggestions,
152
194
  messageInterceptor,
153
195
  setMessages,
154
196
  setPendingResponse
155
197
  ]),
156
198
  getSuggestions: (0, react.useCallback)(async (payload) => {
157
- setSuggestions(await require_commerce_api.default.getNextSuggestions(payload));
199
+ setSuggestions(await require_application_commerce_api.default.getNextSuggestions(payload));
158
200
  }, [setSuggestions]),
159
201
  hydrateMessages: (0, react.useCallback)(async () => {
160
202
  const { orgId, chatId, userId } = context;
161
- const { messages: existingMessages, userEvents } = await require_commerce_api.default.getResponses(orgId, chatId, userId);
203
+ const { messages: existingMessages, userEvents } = await require_application_commerce_api.default.getResponses(orgId, chatId, userId);
162
204
  setMessages([...existingMessages]);
163
205
  }, [context, setMessages])
164
206
  };
@@ -166,4 +208,4 @@ const useSalesAgentService = () => {
166
208
 
167
209
  //#endregion
168
210
  exports.useSalesAgentService = useSalesAgentService;
169
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"salesAgentService.cjs","names":["UserEventCategory","StatusCodeError","MessageRole","MessageType","lastMessage: Message | undefined","messageFromResponse","error: unknown","useSalesAgentService: () => SalesAgentService","requestFailureAtom","messagesAtom","suggestionsAtom","pendingResponseAtom","responseStreamingAtom","useMessageInterceptor","useAmplitude","useFeatureFlagService","appDetailsAtom","successfulResponse: boolean | undefined","CommerceApiClient","SpiffyMetricsEventName"],"sources":["../../../src/contexts/salesAgentContext/salesAgentService.ts"],"sourcesContent":["import { useAtomValue, useSetAtom } from 'jotai';\nimport { useCallback } from 'react';\nimport { v4 as uuid } from 'uuid';\nimport CommerceApiClient from 'src/application/commerce-api';\nimport { appDetailsAtom } from 'src/atoms/app';\nimport Logger from 'src/application/logging/logger';\nimport {\n  GenerationParams,\n  Message,\n  MessageRole,\n  MessageType,\n  NextMessageRequest,\n  Response,\n  UserEvent,\n} from 'src/application/models';\nimport { messageFromResponse } from 'src/application/utils';\nimport {\n  messagesAtom,\n  pendingResponseAtom,\n  requestFailureAtom,\n  responseStreamingAtom,\n  suggestionsAtom,\n} from 'src/atoms/chat/chatState';\nimport { useMessageInterceptor } from 'src/interceptors/useMessageInterceptor';\nimport { SpiffyMetricsEventName, useAmplitude } from 'src/contexts/amplitudeContext';\nimport { useFeatureFlagService } from 'src/contexts/featureFlagServiceContext';\nimport { UserEventCategory } from '@spiffy-ai/commerce-api-client';\n\nimport { StatusCodeError } from './statusCodeError';\n\ninterface SalesAgentService {\n  createResponsePayload: (payload: {\n    userEvents: UserEvent[];\n    generationParams?: GenerationParams;\n  }) => NextMessageRequest;\n  getStreamingResponses: (payload: NextMessageRequest) => Promise<void>;\n  getSuggestions: (payload: NextMessageRequest) => Promise<void>;\n  hydrateMessages: () => Promise<void>;\n}\n\nconst inputPropsToTrackingProps = (\n  payload: NextMessageRequest,\n): Record<string, unknown> | undefined => {\n  const [userEvent] = payload.userEvents || [];\n  if (userEvent.category === UserEventCategory.SuggestionClicked) {\n    return {\n      message_id: userEvent.eventId,\n      user_event_type: 'suggestion_clicked',\n      user_query: userEvent.attributes.content,\n    };\n  }\n  if (userEvent.category === UserEventCategory.QueryTyped) {\n    return {\n      message_id: userEvent.eventId,\n      user_event_type: 'query_typed',\n      user_query: userEvent.attributes.query,\n    };\n  }\n  return {};\n};\n\nexport const getQueryParam = (key: string): string | null => {\n  const urlObj = new URL(window.location.href);\n  return urlObj.searchParams.get(key);\n};\n\nconst getErrorMessage = (_error: unknown): string => {\n  if (_error instanceof StatusCodeError) {\n    if (_error.statusCode === 423) {\n      return 'Due to violation of terms and services, this session has been terminated.';\n    }\n    // Keeping the 429 error messages as the generic error message for now.\n    // if (_error.statusCode === 429) {\n    //   return \"I'm sorry! The system is busy right now. Please try again in a few minutes.\";\n    // }\n  }\n\n  return \"I'm sorry! I'm having trouble right now. Please refresh the page or try again in a moment.\";\n};\n\nconst handleStreamingError = (\n  _error: unknown,\n  setRequestFailure: (failed: boolean) => void,\n  setMessages: (updater: (prev: Message[][]) => Message[][]) => void,\n) => {\n  setRequestFailure(true);\n\n  const errorMessage = getErrorMessage(_error);\n  setMessages(prev => [\n    ...prev,\n    [\n      {\n        id: uuid(),\n        role: MessageRole.Assistant,\n        type: MessageType.Text,\n        createdAt: new Date().toISOString(),\n        metadata: {\n          content: errorMessage,\n        },\n      },\n    ],\n  ]);\n};\n\nconst updateMessageState = (\n  message: Message,\n  lastMessage: Message,\n  setMessages: (updater: (prev: Message[][]) => Message[][]) => void,\n): Message => {\n  if (lastMessage == null) {\n    setMessages(prev => [...prev, [message]]);\n    return message;\n  }\n  if (lastMessage.type === MessageType.Text && message.type === MessageType.Text) {\n    const newMessage = {\n      ...lastMessage,\n      metadata: {\n        ...lastMessage.metadata,\n        content: lastMessage.metadata.content + message.metadata.content,\n      },\n    };\n    setMessages(prev => {\n      const lastTurn = prev[prev.length - 1];\n      return [\n        ...prev.slice(0, prev.length - 1),\n        [...lastTurn.slice(0, lastTurn.length - 1), newMessage],\n      ];\n    });\n    return newMessage;\n  }\n  setMessages(prev => [...prev.slice(0, prev.length - 1), [...prev[prev.length - 1], message]]);\n  return message;\n};\n\nconst processStreamingResponse = async (\n  stream: AsyncIterable<Response>,\n  messageInterceptor: { intercept: (response?: Response) => boolean | undefined },\n  setMessages: (updater: (prev: Message[][]) => Message[][]) => void,\n  setResponseStreaming: (responseStreaming: boolean) => void,\n): Promise<void> => {\n  let lastMessage: Message | undefined;\n  setResponseStreaming(true);\n\n  for await (const response of stream) {\n    try {\n      if (messageInterceptor.intercept(response)) {\n        return;\n      }\n\n      if (!response) {\n        throw new Error('No response from stream');\n      }\n\n      const message = messageFromResponse(response);\n      if (!message) {\n        throw new Error('Failed to transform API response to client message');\n      }\n\n      // No support for ChatSearch messages at the current time.\n      // Perhaps we add support back in the future\n      // if (message.type === MessageType.ProductSearch) {\n      //   handleSearchResults(message);\n      //   hasSearchResults = true;\n      //   setSearchIsLoading(false); // Update search loading immediately when results are detected\n      // }\n\n      lastMessage = updateMessageState(message, lastMessage!, setMessages);\n    } catch (error: unknown) {\n      Logger.logWarn(`[spiffy-ai] Failed to generate responses from stream`, error, {\n        lastResponse: lastMessage,\n        response,\n      });\n    }\n    setResponseStreaming(false);\n  }\n};\n\nexport const useSalesAgentService: () => SalesAgentService = () => {\n  const setRequestFailure = useSetAtom(requestFailureAtom);\n  const setMessages = useSetAtom(messagesAtom);\n  const setSuggestions = useSetAtom(suggestionsAtom);\n  const setPendingResponse = useSetAtom(pendingResponseAtom);\n  const setResponseStreaming = useSetAtom(responseStreamingAtom);\n  const messageInterceptor = useMessageInterceptor();\n  const { trackEvent } = useAmplitude();\n\n  const featureFlagService = useFeatureFlagService();\n  const context = useAtomValue(appDetailsAtom);\n\n  const createResponsePayload = useCallback(\n    ({\n      userEvents,\n      generationParams,\n    }: {\n      userEvents: UserEvent[];\n      generationParams?: GenerationParams;\n    }): NextMessageRequest => {\n      const featureFlags = featureFlagService?.featureFlagService?.getFeatureFlags() || {};\n\n      const overrideConfigVersion =\n        getQueryParam('spiffy_config_version') ||\n        getQueryParam('envive_config_version') ||\n        undefined;\n      const overrideModelDatetime = getQueryParam('override_model_datetime') || undefined;\n      return {\n        id: uuid(),\n        context,\n        userEvents,\n        featureFlags,\n        generationParams,\n        overrideConfigVersion,\n        overrideModelDatetime,\n      };\n    },\n    [context, featureFlagService?.featureFlagService],\n  );\n\n  const getStreamingResponses = useCallback(\n    async (payload: NextMessageRequest): Promise<void> => {\n      const startTime = Date.now();\n      let successfulResponse: boolean | undefined;\n      setPendingResponse(true);\n      const stream = CommerceApiClient.getNextResponseStreaming(payload);\n\n      try {\n        setRequestFailure(false);\n\n        await processStreamingResponse(\n          stream,\n          messageInterceptor,\n          setMessages,\n          setResponseStreaming,\n        );\n        successfulResponse = true;\n\n        // TODO: Add support for the Chrome Extension communication\n        // await logBundleEvent({\n        //   level: 'info',\n        //   event: 'NEXT_RESPONSE_SUCCESS',\n        //   context: {\n        //     type: 'next_responses',\n        //     endpoint: '/v1/next_responses',\n        //     statusCode: 200,\n        //     responseTime,\n        //   },\n        // });\n      } catch (e) {\n        console.error('error getting streaming responses', e);\n        successfulResponse = false;\n        // Log failed next_responses call\n        // const responseTime = Date.now() - startTime;\n        // await logBundleEvent({\n        //   level: 'error',\n        //   event: 'NEXT_RESPONSE_FAILED',\n        //   message: `Failed to get streaming response: ${e}`,\n        //   context: {\n        //     type: 'next_responses',\n        //     endpoint: '/v1/next_responses',\n        //     responseTime,\n        //     error: e instanceof Error ? e.message : String(e),\n        //   },\n        // });\n\n        handleStreamingError(e, setRequestFailure, setMessages);\n        throw e;\n      } finally {\n        // Log next_responses call\n        const responseTime = Date.now() - startTime;\n        trackEvent({\n          eventName: SpiffyMetricsEventName.ChatAssistantResponse,\n          eventProps: {\n            responseTimeMs: responseTime.toString(),\n            successful_response: successfulResponse,\n            ...inputPropsToTrackingProps(payload),\n          },\n        });\n        setPendingResponse(false);\n        setResponseStreaming(false);\n        // logPerfMetric(PerfMetricsEvents.FirstResponseCompleted);\n      }\n    },\n    [\n      // logPerfMetric,\n      setRequestFailure,\n      messageInterceptor,\n      setMessages,\n      setPendingResponse,\n    ],\n  );\n\n  const getSuggestions = useCallback(\n    async (payload: NextMessageRequest): Promise<void> => {\n      const newSuggestions = await CommerceApiClient.getNextSuggestions(payload);\n      setSuggestions(newSuggestions);\n    },\n    [setSuggestions],\n  );\n\n  const hydrateMessages = useCallback(async () => {\n    const { orgId, chatId, userId } = context;\n    const { messages: existingMessages, userEvents } = await CommerceApiClient.getResponses(\n      orgId,\n      chatId,\n      userId,\n    );\n    setMessages([...existingMessages]);\n  }, [context, setMessages]);\n\n  return {\n    createResponsePayload,\n    getStreamingResponses,\n    getSuggestions,\n    hydrateMessages,\n  };\n};\n"],"mappings":"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"}
211
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"salesAgentService.cjs","names":["UserEventCategory","StatusCodeError","MessageRole","MessageType","lastMessage: Message | undefined","responseAnalytics: Record<string, number>","messageFromResponse","error: unknown","useSalesAgentService: () => SalesAgentService","requestFailureAtom","messagesAtom","suggestionsAtom","pendingResponseAtom","responseStreamingAtom","analyticsContextAtom","useMessageInterceptor","useAmplitude","useFeatureFlagService","appDetailsAtom","successfulResponse: boolean | undefined","CommerceApiClient","responseAnalytics: Record<string, unknown> | undefined","SpiffyMetricsEventName","EnviveMetricsEventName"],"sources":["../../../src/contexts/salesAgentContext/salesAgentService.ts"],"sourcesContent":["import { useAtomValue, useSetAtom } from 'jotai';\nimport { useCallback } from 'react';\nimport { v4 as uuid } from 'uuid';\nimport CommerceApiClient from 'src/application/commerce-api';\nimport { appDetailsAtom } from 'src/atoms/app';\nimport Logger from 'src/application/logging/logger';\nimport {\n  GenerationParams,\n  Message,\n  MessageRole,\n  MessageType,\n  NextMessageRequest,\n  Response,\n  UserEvent,\n} from 'src/application/models';\nimport { messageFromResponse } from 'src/application/utils';\nimport {\n  messagesAtom,\n  pendingResponseAtom,\n  requestFailureAtom,\n  responseStreamingAtom,\n  suggestionsAtom,\n} from 'src/atoms/chat/chatState';\nimport { useMessageInterceptor } from 'src/interceptors/useMessageInterceptor';\nimport { SpiffyMetricsEventName, useAmplitude } from 'src/contexts/amplitudeContext';\nimport { useFeatureFlagService } from 'src/contexts/featureFlagServiceContext';\nimport { UserEventCategory } from '@spiffy-ai/commerce-api-client';\nimport { EnviveMetricsEventName } from 'src/services/amplitudeService/amplitudeService';\nimport { analyticsContextAtom } from 'src/atoms/app/variant';\n\nimport { StatusCodeError } from './statusCodeError';\n\ninterface SalesAgentService {\n  createResponsePayload: (payload: {\n    userEvents: UserEvent[];\n    generationParams?: GenerationParams;\n  }) => NextMessageRequest;\n  getStreamingResponses: (payload: NextMessageRequest) => Promise<void>;\n  getSuggestions: (payload: NextMessageRequest) => Promise<void>;\n  hydrateMessages: () => Promise<void>;\n}\n\nconst inputPropsToTrackingProps = (\n  payload: NextMessageRequest,\n): Record<string, unknown> | undefined => {\n  const [userEvent] = payload.userEvents || [];\n  if (userEvent.category === UserEventCategory.SuggestionClicked) {\n    return {\n      message_id: userEvent.eventId,\n      user_event_type: 'suggestion_clicked',\n      user_query: userEvent.attributes.content,\n    };\n  }\n  if (userEvent.category === UserEventCategory.QueryTyped) {\n    return {\n      message_id: userEvent.eventId,\n      user_event_type: 'query_typed',\n      user_query: userEvent.attributes.query,\n    };\n  }\n  return {};\n};\n\nconst inputPropsToEnviveTrackingProps = (\n  payload: NextMessageRequest,\n): Record<string, unknown> | undefined => {\n  const [userEvent] = payload.userEvents || [];\n  if (userEvent.category === UserEventCategory.SuggestionClicked) {\n    return {\n      'chat.request_type': 'suggestion',\n      'chat.request_id': userEvent.eventId,\n      'chat.request_text': userEvent.attributes.content,\n      'chat.suggestion_id': userEvent.attributes.suggestionId,\n      'chat.user_typed': false,\n    };\n  }\n  if (userEvent.category === UserEventCategory.QueryTyped) {\n    return {\n      'chat.request_type': 'text',\n      'chat.request_id': userEvent.eventId,\n      'chat.request_text': userEvent.attributes.query,\n      'chat.user_typed': userEvent.attributes.userTyped,\n    };\n  }\n  if (\n    userEvent.category === UserEventCategory.PageVisit ||\n    userEvent.category === UserEventCategory.PdpVisit ||\n    userEvent.category === UserEventCategory.PlpVisit\n  ) {\n    return {\n      'chat.request_id': userEvent.eventId,\n      'chat.request_type': 'page_visit',\n    };\n  }\n\n  return {};\n};\n\nexport const getQueryParam = (key: string): string | null => {\n  const urlObj = new URL(window.location.href);\n  return urlObj.searchParams.get(key);\n};\n\nconst getErrorMessage = (_error: unknown): string => {\n  if (_error instanceof StatusCodeError) {\n    if (_error.statusCode === 423) {\n      return 'Due to violation of terms and services, this session has been terminated.';\n    }\n    // Keeping the 429 error messages as the generic error message for now.\n    // if (_error.statusCode === 429) {\n    //   return \"I'm sorry! The system is busy right now. Please try again in a few minutes.\";\n    // }\n  }\n\n  return \"I'm sorry! I'm having trouble right now. Please refresh the page or try again in a moment.\";\n};\n\nconst handleStreamingError = (\n  _error: unknown,\n  setRequestFailure: (failed: boolean) => void,\n  setMessages: (updater: (prev: Message[][]) => Message[][]) => void,\n) => {\n  setRequestFailure(true);\n\n  const errorMessage = getErrorMessage(_error);\n  setMessages(prev => [\n    ...prev,\n    [\n      {\n        id: uuid(),\n        role: MessageRole.Assistant,\n        type: MessageType.Text,\n        createdAt: new Date().toISOString(),\n        metadata: {\n          content: errorMessage,\n        },\n      },\n    ],\n  ]);\n};\n\nconst updateMessageState = (\n  message: Message,\n  lastMessage: Message,\n  setMessages: (updater: (prev: Message[][]) => Message[][]) => void,\n): Message => {\n  if (lastMessage == null) {\n    setMessages(prev => [...prev, [message]]);\n    return message;\n  }\n  if (lastMessage.type === MessageType.Text && message.type === MessageType.Text) {\n    const newMessage = {\n      ...lastMessage,\n      metadata: {\n        ...lastMessage.metadata,\n        content: lastMessage.metadata.content + message.metadata.content,\n      },\n    };\n    setMessages(prev => {\n      const lastTurn = prev[prev.length - 1];\n      return [\n        ...prev.slice(0, prev.length - 1),\n        [...lastTurn.slice(0, lastTurn.length - 1), newMessage],\n      ];\n    });\n    return newMessage;\n  }\n  setMessages(prev => [...prev.slice(0, prev.length - 1), [...prev[prev.length - 1], message]]);\n  return message;\n};\n\nconst processStreamingResponse = async (\n  stream: AsyncIterable<Response>,\n  messageInterceptor: { intercept: (response?: Response) => boolean | undefined },\n  setMessages: (updater: (prev: Message[][]) => Message[][]) => void,\n  setResponseStreaming: (responseStreaming: boolean) => void,\n): Promise<Record<string, number>> => {\n  let lastMessage: Message | undefined;\n  setResponseStreaming(true);\n  const responseAnalytics: Record<string, number> = {};\n\n  for await (const response of stream) {\n    try {\n      if (messageInterceptor.intercept(response)) {\n        return {};\n      }\n\n      if (!response) {\n        throw new Error('No response from stream');\n      }\n\n      const message = messageFromResponse(response);\n      if (!message) {\n        throw new Error('Failed to transform API response to client message');\n      }\n      if (message.type === MessageType.Product) {\n        responseAnalytics['chat.product_cards_returned'] =\n          (responseAnalytics['chat.product_cards_returned'] || 0) + 1;\n      }\n      if (message.type === MessageType.Review) {\n        responseAnalytics['chat.review_cards_returned'] =\n          (responseAnalytics['chat.review_cards_returned'] || 0) + 1;\n      }\n\n      // No support for ChatSearch messages at the current time.\n      // Perhaps we add support back in the future\n      // if (message.type === MessageType.ProductSearch) {\n      //   handleSearchResults(message);\n      //   hasSearchResults = true;\n      //   setSearchIsLoading(false); // Update search loading immediately when results are detected\n      // }\n\n      lastMessage = updateMessageState(message, lastMessage!, setMessages);\n    } catch (error: unknown) {\n      Logger.logWarn(`[spiffy-ai] Failed to generate responses from stream`, error, {\n        lastResponse: lastMessage,\n        response,\n      });\n    }\n    setResponseStreaming(false);\n  }\n  return responseAnalytics;\n};\n\nexport const useSalesAgentService: () => SalesAgentService = () => {\n  const setRequestFailure = useSetAtom(requestFailureAtom);\n  const setMessages = useSetAtom(messagesAtom);\n  const setSuggestions = useSetAtom(suggestionsAtom);\n  const setPendingResponse = useSetAtom(pendingResponseAtom);\n  const setResponseStreaming = useSetAtom(responseStreamingAtom);\n  const analyticsContext = useAtomValue(analyticsContextAtom);\n  const messageInterceptor = useMessageInterceptor();\n  const { trackEvent } = useAmplitude();\n\n  const featureFlagService = useFeatureFlagService();\n  const context = useAtomValue(appDetailsAtom);\n\n  const createResponsePayload = useCallback(\n    ({\n      userEvents,\n      generationParams,\n    }: {\n      userEvents: UserEvent[];\n      generationParams?: GenerationParams;\n    }): NextMessageRequest => {\n      const featureFlags = featureFlagService?.featureFlagService?.getFeatureFlags() || {};\n\n      const overrideConfigVersion =\n        getQueryParam('spiffy_config_version') ||\n        getQueryParam('envive_config_version') ||\n        undefined;\n      const overrideModelDatetime = getQueryParam('override_model_datetime') || undefined;\n      return {\n        id: uuid(),\n        context,\n        userEvents,\n        featureFlags,\n        generationParams,\n        overrideConfigVersion,\n        overrideModelDatetime,\n      };\n    },\n    [context, featureFlagService?.featureFlagService],\n  );\n\n  const getStreamingResponses = useCallback(\n    async (payload: NextMessageRequest): Promise<void> => {\n      const startTime = Date.now();\n      let successfulResponse: boolean | undefined;\n      setPendingResponse(true);\n      setSuggestions([]);\n      const stream = CommerceApiClient.getNextResponseStreaming(payload);\n      let responseAnalytics: Record<string, unknown> | undefined;\n      try {\n        setRequestFailure(false);\n\n        responseAnalytics = await processStreamingResponse(\n          stream,\n          messageInterceptor,\n          setMessages,\n          setResponseStreaming,\n        );\n        successfulResponse = true;\n\n        // TODO: Add support for the Chrome Extension communication\n        // await logBundleEvent({\n        //   level: 'info',\n        //   event: 'NEXT_RESPONSE_SUCCESS',\n        //   context: {\n        //     type: 'next_responses',\n        //     endpoint: '/v1/next_responses',\n        //     statusCode: 200,\n        //     responseTime,\n        //   },\n        // });\n      } catch (e) {\n        console.error('error getting streaming responses', e);\n        successfulResponse = false;\n        // Log failed next_responses call\n        // const responseTime = Date.now() - startTime;\n        // await logBundleEvent({\n        //   level: 'error',\n        //   event: 'NEXT_RESPONSE_FAILED',\n        //   message: `Failed to get streaming response: ${e}`,\n        //   context: {\n        //     type: 'next_responses',\n        //     endpoint: '/v1/next_responses',\n        //     responseTime,\n        //     error: e instanceof Error ? e.message : String(e),\n        //   },\n        // });\n\n        handleStreamingError(e, setRequestFailure, setMessages);\n        throw e;\n      } finally {\n        // Log next_responses call\n        const responseTime = Date.now() - startTime;\n        trackEvent({\n          eventName: SpiffyMetricsEventName.ChatAssistantResponse,\n          eventProps: {\n            responseTimeMs: responseTime.toString(),\n            successful_response: successfulResponse,\n            ...inputPropsToTrackingProps(payload),\n          },\n        });\n        trackEvent({\n          eventName: EnviveMetricsEventName.ChatResponse,\n          eventProps: {\n            ...analyticsContext,\n            'chat.response_time_ms': responseTime.toString(),\n            ...inputPropsToEnviveTrackingProps(payload),\n            ...responseAnalytics,\n          },\n        });\n        setPendingResponse(false);\n        setResponseStreaming(false);\n        // logPerfMetric(PerfMetricsEvents.FirstResponseCompleted);\n      }\n    },\n    [\n      analyticsContext,\n      trackEvent,\n      setResponseStreaming,\n      // logPerfMetric,\n      setRequestFailure,\n      setSuggestions,\n      messageInterceptor,\n      setMessages,\n      setPendingResponse,\n    ],\n  );\n\n  const getSuggestions = useCallback(\n    async (payload: NextMessageRequest): Promise<void> => {\n      const newSuggestions = await CommerceApiClient.getNextSuggestions(payload);\n      setSuggestions(newSuggestions);\n    },\n    [setSuggestions],\n  );\n\n  const hydrateMessages = useCallback(async () => {\n    const { orgId, chatId, userId } = context;\n    const { messages: existingMessages, userEvents } = await CommerceApiClient.getResponses(\n      orgId,\n      chatId,\n      userId,\n    );\n    setMessages([...existingMessages]);\n  }, [context, setMessages]);\n\n  return {\n    createResponsePayload,\n    getStreamingResponses,\n    getSuggestions,\n    hydrateMessages,\n  };\n};\n"],"mappings":"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"}