@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.
- package/dist/application/commerce-api.cjs +14 -14
- package/dist/application/commerce-api.d.cts +58 -0
- package/dist/application/commerce-api.d.ts +59 -0
- package/dist/application/commerce-api.js +3 -3
- package/dist/application/models/api/widgetText.d.cts +14 -0
- package/dist/application/models/api/widgetText.d.ts +14 -0
- package/dist/application/models/api/widgetTextRequest.d.cts +15 -0
- package/dist/application/models/api/widgetTextRequest.d.ts +15 -0
- package/dist/application/models/chatElementDisplayLocationV3.cjs +2 -1
- package/dist/application/models/chatElementDisplayLocationV3.d.cts +2 -1
- package/dist/application/models/chatElementDisplayLocationV3.d.ts +2 -1
- package/dist/application/models/chatElementDisplayLocationV3.js +2 -1
- package/dist/application/models/guards/api/index.cjs +4 -4
- package/dist/application/models/guards/api/index.js +4 -4
- package/dist/application/models/guards/api/isApiFormResponse.cjs +1 -1
- package/dist/application/models/guards/api/isApiFormResponse.js +1 -1
- package/dist/application/models/guards/api/isApiFormSubmittedResponseAttributes.cjs +1 -1
- package/dist/application/models/guards/api/isApiFormSubmittedResponseAttributes.js +1 -1
- package/dist/application/models/guards/api/isApiOrderResponseAttributes.cjs +1 -1
- package/dist/application/models/guards/api/isApiOrderResponseAttributes.js +1 -1
- package/dist/application/models/guards/api/isApiProductResponseAttributes.cjs +1 -1
- package/dist/application/models/guards/api/isApiProductResponseAttributes.js +1 -1
- package/dist/application/models/guards/api/isApiResponse.cjs +1 -1
- package/dist/application/models/guards/api/isApiResponse.js +1 -1
- package/dist/application/models/index.cjs +12 -12
- package/dist/application/models/index.d.cts +25 -25
- package/dist/application/models/index.d.ts +25 -25
- package/dist/application/models/index.js +12 -12
- package/dist/application/models/validators/validateGraphQLFrontendConfig.cjs +1 -1
- package/dist/application/models/validators/validateGraphQLFrontendConfig.js +1 -1
- package/dist/application/models/validators/validateUserEvent.cjs +2 -2
- package/dist/application/models/validators/validateUserEvent.js +2 -2
- package/dist/application/models/variantInfo/variantInfo.cjs +2 -1
- package/dist/application/models/variantInfo/variantInfo.d.cts +6 -1
- package/dist/application/models/variantInfo/variantInfo.d.ts +6 -1
- package/dist/application/models/variantInfo/variantInfo.js +2 -1
- package/dist/application/utils/analyticsUtils.cjs +1 -1
- package/dist/application/utils/analyticsUtils.js +1 -1
- package/dist/application/utils/widgetTextFromApiWidgetTextResponse.cjs +3 -1
- package/dist/application/utils/widgetTextFromApiWidgetTextResponse.js +3 -1
- package/dist/application/utils/widgetTextRequestToApiRequest.cjs +3 -1
- package/dist/application/utils/widgetTextRequestToApiRequest.js +3 -1
- package/dist/atoms/app/index.cjs +7 -7
- package/dist/atoms/app/index.d.cts +8 -8
- package/dist/atoms/app/index.d.ts +8 -8
- package/dist/atoms/app/index.js +1 -1
- package/dist/atoms/app/variant.cjs +44 -4
- package/dist/atoms/app/variant.d.cts +8 -8
- package/dist/atoms/app/variant.d.ts +8 -8
- package/dist/atoms/app/variant.js +44 -5
- package/dist/atoms/chat/chatState.d.cts +17 -17
- package/dist/atoms/chat/chatState.d.ts +18 -18
- package/dist/atoms/chat/form.cjs +1 -1
- package/dist/atoms/chat/form.d.cts +2 -2
- package/dist/atoms/chat/form.d.ts +2 -2
- package/dist/atoms/chat/form.js +1 -1
- package/dist/atoms/chat/index.cjs +1 -1
- package/dist/atoms/chat/index.d.cts +2 -2
- package/dist/atoms/chat/index.d.ts +2 -2
- package/dist/atoms/chat/index.js +1 -1
- 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 +6 -6
- 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 +2 -2
- 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 +3 -3
- package/dist/atoms/chat/suggestions.d.cts +2 -2
- package/dist/atoms/chat/suggestions.d.ts +2 -2
- package/dist/atoms/envive/enviveConfig.d.cts +19 -0
- package/dist/atoms/envive/enviveConfig.d.ts +19 -0
- 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/index.cjs +1 -1
- package/dist/atoms/org/index.js +1 -1
- 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.cjs +2 -2
- package/dist/atoms/search/chatSearch.d.cts +17 -17
- package/dist/atoms/search/chatSearch.d.ts +17 -17
- package/dist/atoms/search/chatSearch.js +2 -2
- package/dist/atoms/search/searchAPI.cjs +1 -1
- package/dist/atoms/search/searchAPI.d.cts +13 -13
- package/dist/atoms/search/searchAPI.d.ts +13 -13
- package/dist/atoms/search/searchAPI.js +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 +10 -7
- package/dist/contexts/amplitudeContext/amplitudeContext.d.cts +2 -2
- package/dist/contexts/amplitudeContext/amplitudeContext.d.ts +2 -2
- package/dist/contexts/amplitudeContext/amplitudeContext.js +6 -3
- package/dist/contexts/amplitudeContext/index.cjs +1 -0
- package/dist/contexts/amplitudeContext/index.d.cts +2 -2
- package/dist/contexts/amplitudeContext/index.d.ts +2 -2
- package/dist/contexts/amplitudeContext/index.js +2 -2
- package/dist/contexts/cdnContext/cdnContext.cjs +3 -3
- package/dist/contexts/enviveConfigContext/enviveConfigContext.cjs +6 -5
- package/dist/contexts/enviveConfigContext/enviveConfigContext.d.cts +2 -1
- package/dist/contexts/enviveConfigContext/enviveConfigContext.d.ts +2 -1
- package/dist/contexts/enviveConfigContext/enviveConfigContext.js +3 -2
- package/dist/contexts/enviveContext/WindowChatToggleBinder.cjs +4 -3
- package/dist/contexts/enviveContext/WindowChatToggleBinder.js +4 -3
- package/dist/contexts/enviveContext/enviveContext.cjs +15 -11
- package/dist/contexts/enviveContext/enviveContext.d.cts +2 -1
- package/dist/contexts/enviveContext/enviveContext.d.ts +2 -1
- package/dist/contexts/enviveContext/enviveContext.js +15 -11
- package/dist/contexts/enviveContext/types.d.cts +1 -1
- package/dist/contexts/enviveContext/types.d.ts +1 -1
- package/dist/contexts/featureFlagServiceContext/featureFlagServiceContext.cjs +23 -2
- package/dist/contexts/featureFlagServiceContext/featureFlagServiceContext.d.cts +11 -2
- package/dist/contexts/featureFlagServiceContext/featureFlagServiceContext.d.ts +11 -2
- package/dist/contexts/featureFlagServiceContext/featureFlagServiceContext.js +23 -2
- package/dist/contexts/featureFlagServiceContext/index.d.cts +2 -2
- package/dist/contexts/featureFlagServiceContext/index.d.ts +2 -2
- package/dist/contexts/graphqlContext/graphqlContext.cjs +7 -7
- package/dist/contexts/graphqlContext/graphqlContext.d.cts +1 -1
- package/dist/contexts/graphqlContext/graphqlContext.d.ts +1 -1
- package/dist/contexts/graphqlContext/graphqlContext.js +3 -3
- package/dist/contexts/graphqlContext/mockV3Config.cjs +10 -4
- package/dist/contexts/graphqlContext/mockV3Config.js +10 -4
- package/dist/contexts/hardcopyContext/hardcopyContext.cjs +50 -5
- package/dist/contexts/hardcopyContext/hardcopyContext.d.cts +4 -1
- package/dist/contexts/hardcopyContext/hardcopyContext.d.ts +4 -1
- package/dist/contexts/hardcopyContext/hardcopyContext.js +48 -3
- package/dist/contexts/newOrgConfigContext/newOrgConfigContext.cjs +4 -4
- package/dist/contexts/newOrgConfigContext/newOrgConfigContext.js +1 -1
- package/dist/contexts/pageContext/mapping.cjs +6 -2
- package/dist/contexts/pageContext/mapping.js +6 -2
- package/dist/contexts/pageContext/pageContext.cjs +4 -3
- package/dist/contexts/pageContext/pageContext.js +2 -1
- package/dist/contexts/pageContext/types.d.cts +7 -3
- package/dist/contexts/pageContext/types.d.ts +7 -3
- package/dist/contexts/salesAgentContext/chatAPI.cjs +103 -13
- package/dist/contexts/salesAgentContext/chatAPI.d.cts +8 -4
- package/dist/contexts/salesAgentContext/chatAPI.d.ts +8 -4
- package/dist/contexts/salesAgentContext/chatAPI.js +105 -15
- package/dist/contexts/salesAgentContext/salesAgentContext.cjs +1 -1
- package/dist/contexts/salesAgentContext/salesAgentContext.js +1 -1
- package/dist/contexts/salesAgentContext/salesAgentService.cjs +53 -11
- package/dist/contexts/salesAgentContext/salesAgentService.js +51 -9
- package/dist/contexts/searchContext/searchContext.cjs +4 -4
- package/dist/contexts/searchContext/searchContext.js +1 -1
- package/dist/contexts/systemSettingsContext/systemSettingsContext.cjs +3 -3
- package/dist/contexts/systemSettingsContext/systemSettingsContext.d.cts +2 -2
- package/dist/contexts/systemSettingsContext/systemSettingsContext.d.ts +2 -2
- package/dist/contexts/types.cjs +1 -1
- package/dist/contexts/types.d.cts +7 -2
- package/dist/contexts/types.d.ts +7 -2
- package/dist/contexts/types.js +1 -1
- package/dist/contexts/typesV3.cjs +2 -1
- package/dist/contexts/typesV3.d.cts +12 -7
- package/dist/contexts/typesV3.d.ts +10 -5
- package/dist/contexts/typesV3.js +2 -1
- package/dist/contexts/userIdentityContext/userIdentityContext.cjs +3 -3
- package/dist/contexts/widgetConfigContext/widgetConfigContext.cjs +1 -1
- package/dist/contexts/widgetConfigContext/widgetConfigContext.js +1 -1
- package/dist/hooks/AppDetails/useAppDetails.cjs +6 -6
- package/dist/hooks/GrabAndScroll/useGrabAndScroll.d.cts +2 -2
- package/dist/hooks/GrabAndScroll/useGrabAndScroll.d.ts +2 -2
- package/dist/hooks/ImageResolver/useImageResolver.cjs +3 -3
- package/dist/hooks/Search/useSearch.cjs +6 -6
- package/dist/hooks/Search/useSearch.d.cts +1 -1
- package/dist/hooks/Search/useSearch.d.ts +1 -1
- package/dist/hooks/Search/useSearch.js +3 -3
- package/dist/hooks/Search/useSearchInput.cjs +2 -2
- package/dist/hooks/Search/useSearchInput.js +2 -2
- package/dist/services/amplitudeService/amplitudeService.cjs +49 -11
- package/dist/services/amplitudeService/amplitudeService.d.cts +18 -3
- package/dist/services/amplitudeService/amplitudeService.d.ts +18 -3
- package/dist/services/amplitudeService/amplitudeService.js +49 -12
- package/dist/services/amplitudeService/index.cjs +1 -0
- package/dist/services/amplitudeService/index.d.cts +2 -2
- package/dist/services/amplitudeService/index.d.ts +2 -2
- package/dist/services/amplitudeService/index.js +2 -2
- package/dist/types/customerService.d.cts +3 -1
- package/dist/types/customerService.d.ts +1 -1
- package/package.json +10 -2
- package/src/application/models/api/widgetText.ts +2 -0
- package/src/application/models/api/widgetTextRequest.ts +1 -0
- package/src/application/models/chatElementDisplayLocationV3.ts +1 -0
- package/src/application/models/variantInfo/variantInfo.ts +5 -0
- package/src/application/utils/widgetTextFromApiWidgetTextResponse.ts +2 -0
- package/src/application/utils/widgetTextRequestToApiRequest.ts +4 -0
- package/src/atoms/app/variant.ts +56 -2
- package/src/contexts/amplitudeContext/__tests__/amplitudeContext.test.tsx +2 -0
- package/src/contexts/amplitudeContext/amplitudeContext.tsx +6 -1
- package/src/contexts/enviveConfigContext/enviveConfigContext.tsx +3 -0
- package/src/contexts/enviveContext/WindowChatToggleBinder.tsx +5 -1
- package/src/contexts/enviveContext/enviveContext.tsx +5 -1
- package/src/contexts/featureFlagServiceContext/featureFlagServiceContext.tsx +36 -0
- package/src/contexts/graphqlContext/mockV3Config.ts +13 -7
- package/src/contexts/hardcopyContext/hardcopyContext.tsx +48 -3
- package/src/contexts/pageContext/mapping.ts +13 -1
- package/src/contexts/pageContext/pageContext.tsx +1 -0
- package/src/contexts/pageContext/types.ts +7 -1
- package/src/contexts/salesAgentContext/chatAPI.ts +106 -13
- package/src/contexts/salesAgentContext/salesAgentService.ts +66 -4
- package/src/contexts/types.ts +8 -1
- package/src/contexts/typesV3.ts +24 -16
- package/src/services/amplitudeService/__tests__/amplitudeService.test.ts +80 -0
- 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: (
|
|
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+
|
|
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: (
|
|
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+
|
|
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 {
|
|
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
|
|
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
|
-
}, [
|
|
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
|
-
}, [
|
|
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
|
|
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
|
-
}, [
|
|
61
|
-
|
|
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
|
-
}, [
|
|
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
|
|
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
|
-
|
|
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
|
|
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
|
|
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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"}
|