@envive-ai/react-hooks 0.3.2 → 0.3.3
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 +6 -4
- package/dist/application/commerce-api.js +6 -4
- package/dist/application/models/api/nextMessageRequest.d.cts +3 -1
- package/dist/application/models/api/nextMessageRequest.d.ts +3 -1
- package/dist/application/models/variantInfo/variantInfo.cjs +4 -1
- package/dist/application/models/variantInfo/variantInfo.d.cts +4 -1
- package/dist/application/models/variantInfo/variantInfo.d.ts +4 -1
- package/dist/application/models/variantInfo/variantInfo.js +4 -1
- package/dist/application/utils/elementObserver.d.cts +2 -2
- package/dist/application/utils/elementObserver.d.ts +2 -2
- package/dist/atoms/app/index.cjs +12 -1
- package/dist/atoms/app/index.d.cts +3 -2
- package/dist/atoms/app/index.d.ts +3 -2
- package/dist/atoms/app/index.js +13 -3
- package/dist/atoms/app/variant.cjs +2 -2
- package/dist/atoms/app/variant.d.cts +69 -5
- package/dist/atoms/app/variant.d.ts +69 -5
- package/dist/atoms/app/variant.js +2 -2
- package/dist/atoms/chat/chatState.d.cts +15 -15
- package/dist/atoms/chat/chatState.d.ts +15 -15
- package/dist/atoms/chat/form.d.cts +2 -2
- package/dist/atoms/chat/form.d.ts +2 -2
- package/dist/atoms/chat/index.d.cts +2 -2
- package/dist/atoms/chat/index.d.ts +2 -2
- package/dist/atoms/chat/lastMessage.d.cts +2 -2
- package/dist/atoms/chat/lastMessage.d.ts +2 -2
- package/dist/atoms/chat/messageQueue.cjs +1 -35
- package/dist/atoms/chat/messageQueue.js +2 -34
- 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 +2 -2
- package/dist/atoms/chat/replies.d.ts +2 -2
- package/dist/atoms/chat/suggestions.d.cts +2 -2
- package/dist/atoms/chat/suggestions.d.ts +2 -2
- package/dist/atoms/globalSearch/globalSearch.cjs +1 -3
- package/dist/atoms/globalSearch/globalSearch.d.cts +7 -10
- package/dist/atoms/globalSearch/globalSearch.d.ts +7 -10
- package/dist/atoms/globalSearch/globalSearch.js +2 -3
- package/dist/atoms/globalSearch/index.cjs +0 -1
- package/dist/atoms/globalSearch/index.d.cts +2 -2
- package/dist/atoms/globalSearch/index.d.ts +2 -2
- package/dist/atoms/globalSearch/index.js +2 -2
- 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 +5 -5
- package/dist/atoms/org/graphqlConfig.d.ts +5 -5
- package/dist/atoms/org/newOrgConfigAtom.d.cts +2 -2
- package/dist/atoms/org/newOrgConfigAtom.d.ts +2 -2
- package/dist/atoms/org/orgAnalyticsConfig.d.cts +5 -5
- package/dist/atoms/org/orgAnalyticsConfig.d.ts +5 -5
- package/dist/atoms/search/chatSearch.d.cts +17 -17
- package/dist/atoms/search/chatSearch.d.ts +17 -17
- package/dist/atoms/search/searchAPI.cjs +1 -1
- package/dist/atoms/search/searchAPI.d.cts +14 -14
- package/dist/atoms/search/searchAPI.d.ts +14 -14
- package/dist/atoms/search/searchAPI.js +1 -1
- package/dist/contexts/amplitudeContext/amplitudeContext.cjs +5 -5
- package/dist/contexts/amplitudeContext/amplitudeContext.js +5 -5
- package/dist/contexts/enviveContext/enviveContext.cjs +7 -3
- package/dist/contexts/enviveContext/enviveContext.js +7 -3
- package/dist/contexts/graphqlContext/graphqlContext.cjs +1 -1
- package/dist/contexts/graphqlContext/graphqlContext.js +1 -1
- package/dist/contexts/hardcopyContext/hardcopyContext.cjs +62 -0
- package/dist/contexts/hardcopyContext/hardcopyContext.d.cts +23 -0
- package/dist/contexts/hardcopyContext/hardcopyContext.d.ts +23 -0
- package/dist/contexts/hardcopyContext/hardcopyContext.js +60 -0
- package/dist/contexts/hardcopyContext/index.cjs +4 -0
- package/dist/contexts/hardcopyContext/index.d.cts +2 -0
- package/dist/contexts/hardcopyContext/index.d.ts +2 -0
- package/dist/contexts/hardcopyContext/index.js +3 -0
- package/dist/contexts/pageContext/index.cjs +6 -0
- package/dist/contexts/pageContext/index.d.cts +3 -0
- package/dist/contexts/pageContext/index.d.ts +3 -0
- package/dist/contexts/pageContext/index.js +4 -0
- package/dist/contexts/pageContext/mapping.cjs +25 -0
- package/dist/contexts/pageContext/mapping.d.cts +8 -0
- package/dist/contexts/pageContext/mapping.d.ts +8 -0
- package/dist/contexts/pageContext/mapping.js +25 -0
- package/dist/contexts/pageContext/pageContext.cjs +66 -0
- package/dist/contexts/pageContext/pageContext.d.cts +11 -0
- package/dist/contexts/pageContext/pageContext.d.ts +11 -0
- package/dist/contexts/pageContext/pageContext.js +64 -0
- package/dist/contexts/pageContext/types.cjs +0 -0
- package/dist/contexts/pageContext/types.d.cts +27 -0
- package/dist/contexts/pageContext/types.d.ts +27 -0
- package/dist/contexts/pageContext/types.js +1 -0
- package/dist/contexts/salesAgentContext/chatAPI.cjs +45 -0
- package/dist/contexts/salesAgentContext/chatAPI.d.cts +35 -0
- package/dist/contexts/salesAgentContext/chatAPI.d.ts +35 -0
- package/dist/contexts/salesAgentContext/chatAPI.js +44 -0
- package/dist/contexts/salesAgentContext/index.cjs +20 -0
- package/dist/contexts/salesAgentContext/index.d.cts +4 -0
- package/dist/contexts/salesAgentContext/index.d.ts +4 -0
- package/dist/contexts/salesAgentContext/index.js +5 -0
- package/dist/contexts/salesAgentContext/salesAgentContext.cjs +90 -0
- package/dist/contexts/salesAgentContext/salesAgentContext.d.cts +22 -0
- package/dist/contexts/salesAgentContext/salesAgentContext.d.ts +22 -0
- package/dist/contexts/salesAgentContext/salesAgentContext.js +88 -0
- package/dist/contexts/salesAgentContext/salesAgentService.cjs +99 -0
- package/dist/contexts/salesAgentContext/salesAgentService.js +98 -0
- package/dist/contexts/searchContext/searchContext.cjs +1 -1
- package/dist/contexts/searchContext/searchContext.js +1 -1
- package/dist/contexts/systemSettingsContext/systemSettingsContext.d.cts +2 -2
- package/dist/contexts/types.cjs +11 -1
- package/dist/contexts/types.d.cts +10 -2
- package/dist/contexts/types.d.ts +10 -2
- package/dist/contexts/types.js +11 -2
- package/dist/contexts/userIdentityContext/userIdentityContext.cjs +1 -1
- package/dist/contexts/userIdentityContext/userIdentityContext.js +1 -1
- package/dist/hooks/GrabAndScroll/useGrabAndScroll.d.cts +2 -2
- package/dist/hooks/GrabAndScroll/useGrabAndScroll.d.ts +2 -2
- package/dist/hooks/GraphQLConfig/useGraphQLConfig.cjs +9 -3
- package/dist/hooks/GraphQLConfig/useGraphQLConfig.js +9 -3
- package/dist/hooks/ImageResolver/useImageResolver.cjs +1 -1
- package/dist/hooks/ImageResolver/useImageResolver.js +1 -1
- package/dist/hooks/Search/useSearch.cjs +5 -8
- package/dist/hooks/Search/useSearch.d.cts +1 -3
- package/dist/hooks/Search/useSearch.d.ts +1 -3
- package/dist/hooks/Search/useSearch.js +6 -9
- package/dist/hooks/utils.d.cts +1 -1
- package/dist/hooks/utils.d.ts +1 -1
- package/package.json +5 -1
- package/dist/contexts/chatContext/chatContext.cjs +0 -306
- package/dist/contexts/chatContext/chatContext.d.cts +0 -15
- package/dist/contexts/chatContext/chatContext.d.ts +0 -15
- package/dist/contexts/chatContext/chatContext.js +0 -304
- package/dist/contexts/chatContext/index.cjs +0 -4
- package/dist/contexts/chatContext/index.d.cts +0 -2
- package/dist/contexts/chatContext/index.d.ts +0 -2
- package/dist/contexts/chatContext/index.js +0 -3
|
@@ -1,304 +0,0 @@
|
|
|
1
|
-
import { createAppLoadedEvent, createVisitUserEvent } from "../../hooks/utils.js";
|
|
2
|
-
import { useMessageInterceptor } from "../../interceptors/useMessageInterceptor.js";
|
|
3
|
-
import logger_default from "../../application/logging/logger.js";
|
|
4
|
-
import { MessageRole, MessageType } from "../../application/models/message.js";
|
|
5
|
-
import "../../application/models/index.js";
|
|
6
|
-
import { getAtomStore } from "../../atoms/atomStore/atomStore.js";
|
|
7
|
-
import "../../atoms/atomStore/index.js";
|
|
8
|
-
import { supportedEventAtom, variantInfoAtom } from "../../atoms/app/variant.js";
|
|
9
|
-
import { chatIdAtom, userIdAtom } from "../../atoms/app/index.js";
|
|
10
|
-
import { SpiffyMetricsEventName } from "../amplitudeContext/amplitudeContext.js";
|
|
11
|
-
import { messagesAtom, requestFailureAtom, responseStreamingAtom, suggestionsAtom, suggestionsLoadingAtom, userEventsAtom, userHasRepliedAtom } from "../../atoms/chat/chatState.js";
|
|
12
|
-
import { clearUserEventAtom, createResponsePayload, processUserEventAtom, userEventQueueAtom, userQueueEventCountAtom } from "../../atoms/chat/messageQueue.js";
|
|
13
|
-
import { PerfMetricsEvents, logPerfMetricAtom } from "../../atoms/chat/performanceMetrics.js";
|
|
14
|
-
import { chatAtom } from "../../atoms/chat/index.js";
|
|
15
|
-
import { chatSearchIsLoadingAtom, chatSearchProductSortingAtom, chatSearchProducts, chatSearchStateAtom, handleSearchResultsAtom } from "../../atoms/search/chatSearch.js";
|
|
16
|
-
import { messageFromResponse } from "../../application/utils/messageFromResponse.js";
|
|
17
|
-
import "../../application/utils/index.js";
|
|
18
|
-
import { SessionRestartRequired } from "../../types/exceptions/sessionExceptions.js";
|
|
19
|
-
import commerce_api_default from "../../application/commerce-api.js";
|
|
20
|
-
import { useAmplitudeTracking } from "../../hooks/AmplitudeOperations/useAmplitudeOperations.js";
|
|
21
|
-
import { useSystemSettingsContext } from "../../hooks/SystemSettingsContext/useSystemSettingsContext.js";
|
|
22
|
-
import "../../hooks/SystemSettingsContext/index.js";
|
|
23
|
-
import { UserEventCategory } from "@spiffy-ai/commerce-api-client";
|
|
24
|
-
import { v4 } from "uuid";
|
|
25
|
-
import { createContext, useCallback, useEffect, useMemo, useState } from "react";
|
|
26
|
-
import { useAtom, useAtomValue, useSetAtom } from "jotai";
|
|
27
|
-
import { jsx } from "react/jsx-runtime";
|
|
28
|
-
|
|
29
|
-
//#region src/contexts/chatContext/chatContext.tsx
|
|
30
|
-
/**
|
|
31
|
-
* Record the chat assistant response in Amplitude
|
|
32
|
-
*
|
|
33
|
-
* @param startTimeMs The start time of the assistant response
|
|
34
|
-
* @param payload The payload used to generate the response
|
|
35
|
-
*/
|
|
36
|
-
const recordAssistantResponse = (startTimeMs, payload, track) => {
|
|
37
|
-
const atomStore = getAtomStore();
|
|
38
|
-
const chatState = atomStore.get(chatAtom);
|
|
39
|
-
const chatSearchState = atomStore.get(chatSearchStateAtom);
|
|
40
|
-
const searchProducts = atomStore.get(chatSearchProducts);
|
|
41
|
-
const searchProductsSort = atomStore.get(chatSearchProductSortingAtom);
|
|
42
|
-
const assistantResponseTimeMs = {
|
|
43
|
-
start: startTimeMs,
|
|
44
|
-
end: Date.now()
|
|
45
|
-
};
|
|
46
|
-
let userQueryProperty;
|
|
47
|
-
if (chatState.replyEventCategory === UserEventCategory.SuggestionClicked && chatState.suggestion) userQueryProperty = chatState.suggestion.content;
|
|
48
|
-
else if (chatState.userQuery && chatState.userQuery.length > 0) userQueryProperty = chatState.userQuery;
|
|
49
|
-
const eventProps = {
|
|
50
|
-
response_time_ms: assistantResponseTimeMs.end - assistantResponseTimeMs.start,
|
|
51
|
-
user_event_type: chatState.replyEventCategory,
|
|
52
|
-
user_query: userQueryProperty
|
|
53
|
-
};
|
|
54
|
-
if (chatState.replyEventCategory === UserEventCategory.FormSubmitted) {
|
|
55
|
-
const lastAssistantTurn = chatState.messages.filter((turn) => turn.length > 0 && turn[0].role === MessageRole.Assistant).pop();
|
|
56
|
-
eventProps.form_submitted_attributes = {
|
|
57
|
-
form_type: payload.userEvents?.find((event) => event.category === UserEventCategory.FormSubmitted)?.attributes.formType,
|
|
58
|
-
status: lastAssistantTurn?.some((response) => response.type === MessageType.Order) ? "success" : "failed"
|
|
59
|
-
};
|
|
60
|
-
}
|
|
61
|
-
if (chatSearchState === "product-page") {
|
|
62
|
-
eventProps.search_products_returned = searchProducts.length;
|
|
63
|
-
eventProps.search_products_sort_type = searchProductsSort;
|
|
64
|
-
}
|
|
65
|
-
track(SpiffyMetricsEventName.ChatAssistantResponse, { eventProps });
|
|
66
|
-
};
|
|
67
|
-
const ChatContext = createContext(void 0);
|
|
68
|
-
const updateMessageState = (message, lastMessage, setMessages) => {
|
|
69
|
-
if (lastMessage == null) {
|
|
70
|
-
setMessages((prev) => [...prev, [message]]);
|
|
71
|
-
return message;
|
|
72
|
-
}
|
|
73
|
-
if (lastMessage.type === MessageType.Text && message.type === MessageType.Text) {
|
|
74
|
-
const newMessage = {
|
|
75
|
-
...lastMessage,
|
|
76
|
-
metadata: {
|
|
77
|
-
...lastMessage.metadata,
|
|
78
|
-
content: lastMessage.metadata.content + message.metadata.content
|
|
79
|
-
}
|
|
80
|
-
};
|
|
81
|
-
setMessages((prev) => {
|
|
82
|
-
const lastTurn = prev[prev.length - 1];
|
|
83
|
-
return [...prev.slice(0, prev.length - 1), [...lastTurn.slice(0, lastTurn.length - 1), newMessage]];
|
|
84
|
-
});
|
|
85
|
-
return newMessage;
|
|
86
|
-
}
|
|
87
|
-
setMessages((prev) => [...prev.slice(0, prev.length - 1), [...prev[prev.length - 1], message]]);
|
|
88
|
-
return message;
|
|
89
|
-
};
|
|
90
|
-
const handleStreamingError = (_error, setRequestFailure, setMessages) => {
|
|
91
|
-
setRequestFailure(true);
|
|
92
|
-
setMessages((prev) => [...prev, [{
|
|
93
|
-
id: v4(),
|
|
94
|
-
role: MessageRole.Assistant,
|
|
95
|
-
type: MessageType.Text,
|
|
96
|
-
createdAt: (/* @__PURE__ */ new Date()).toISOString(),
|
|
97
|
-
metadata: { content: "I'm sorry! I'm having trouble right now. Please refresh the page or try again in a moment." }
|
|
98
|
-
}]]);
|
|
99
|
-
};
|
|
100
|
-
const processStreamingResponse = async (stream, messageInterceptor, handleSearchResults, setMessages, setSearchIsLoading, chatId) => {
|
|
101
|
-
let lastMessage;
|
|
102
|
-
let hasSearchResults = false;
|
|
103
|
-
for await (const response of stream) try {
|
|
104
|
-
if (messageInterceptor.intercept(response)) return { hasSearchResults };
|
|
105
|
-
const message = messageFromResponse(response);
|
|
106
|
-
if (!message) throw new Error("Failed to transform API response to client message");
|
|
107
|
-
if (message.type === MessageType.ProductSearch) {
|
|
108
|
-
handleSearchResults(message);
|
|
109
|
-
hasSearchResults = true;
|
|
110
|
-
setSearchIsLoading(false);
|
|
111
|
-
}
|
|
112
|
-
lastMessage = updateMessageState(message, lastMessage, setMessages);
|
|
113
|
-
} catch (error) {
|
|
114
|
-
logger_default.logWarn(`[spiffy-ai] Failed to generate responses from stream chat_id=${chatId}`, error, {
|
|
115
|
-
lastResponse: lastMessage,
|
|
116
|
-
response
|
|
117
|
-
});
|
|
118
|
-
}
|
|
119
|
-
return { hasSearchResults };
|
|
120
|
-
};
|
|
121
|
-
const ChatContextProvider = ({ children }) => {
|
|
122
|
-
const logPerfMetric = useSetAtom(logPerfMetricAtom);
|
|
123
|
-
const [widgetInitialized, setWidgetInitialized] = useState(false);
|
|
124
|
-
const setUserHasReplied = useSetAtom(userHasRepliedAtom);
|
|
125
|
-
const [messages, setMessages] = useAtom(messagesAtom);
|
|
126
|
-
const setUserEvents = useSetAtom(userEventsAtom);
|
|
127
|
-
const setSuggestions = useSetAtom(suggestionsAtom);
|
|
128
|
-
const [suggestionsLoading, setSuggestionsLoading] = useAtom(suggestionsLoadingAtom);
|
|
129
|
-
const [responseStreaming, setResponseStreaming] = useAtom(responseStreamingAtom);
|
|
130
|
-
const setRequestFailure = useSetAtom(requestFailureAtom);
|
|
131
|
-
const userEvents = useAtomValue(userEventQueueAtom);
|
|
132
|
-
const userQueueEventCount = useAtomValue(userQueueEventCountAtom);
|
|
133
|
-
const markUserEventsProcessed = useSetAtom(processUserEventAtom);
|
|
134
|
-
const clearUserEventQueue = useSetAtom(clearUserEventAtom);
|
|
135
|
-
const userId = useAtomValue(userIdAtom);
|
|
136
|
-
const chatId = useAtomValue(chatIdAtom);
|
|
137
|
-
const supportedEvent = useAtomValue(supportedEventAtom);
|
|
138
|
-
const orgId = "mock-org-id";
|
|
139
|
-
const variantInfo = useAtomValue(variantInfoAtom);
|
|
140
|
-
const settingsContext = useSystemSettingsContext();
|
|
141
|
-
const messageInterceptor = useMessageInterceptor();
|
|
142
|
-
const handleSearchResults = useSetAtom(handleSearchResultsAtom);
|
|
143
|
-
const setSearchIsLoading = useSetAtom(chatSearchIsLoadingAtom);
|
|
144
|
-
const { track } = useAmplitudeTracking();
|
|
145
|
-
const getStreamingResponses = useCallback(async (payload) => {
|
|
146
|
-
logPerfMetric(PerfMetricsEvents.FirstResponseStarted);
|
|
147
|
-
const stream = commerce_api_default.getNextResponseStreaming(payload);
|
|
148
|
-
try {
|
|
149
|
-
setRequestFailure(false);
|
|
150
|
-
const { hasSearchResults } = await processStreamingResponse(stream, messageInterceptor, handleSearchResults, setMessages, setSearchIsLoading, chatId);
|
|
151
|
-
return { hasSearchResults };
|
|
152
|
-
} catch (e) {
|
|
153
|
-
handleStreamingError(e, setRequestFailure, setMessages);
|
|
154
|
-
throw e;
|
|
155
|
-
} finally {
|
|
156
|
-
logPerfMetric(PerfMetricsEvents.FirstResponseCompleted);
|
|
157
|
-
}
|
|
158
|
-
}, [
|
|
159
|
-
logPerfMetric,
|
|
160
|
-
setRequestFailure,
|
|
161
|
-
messageInterceptor,
|
|
162
|
-
handleSearchResults,
|
|
163
|
-
setMessages,
|
|
164
|
-
setSearchIsLoading,
|
|
165
|
-
chatId
|
|
166
|
-
]);
|
|
167
|
-
const getSuggestions = useCallback(async () => {
|
|
168
|
-
logPerfMetric(PerfMetricsEvents.FirstSuggestionsStarted);
|
|
169
|
-
setSuggestionsLoading(true);
|
|
170
|
-
setSuggestions([]);
|
|
171
|
-
const payloadWithoutAppLoaded = createResponsePayload({
|
|
172
|
-
userEvents: [],
|
|
173
|
-
generationParams: settingsContext.generationParams
|
|
174
|
-
});
|
|
175
|
-
setSuggestions((await commerce_api_default.getNextSuggestions(payloadWithoutAppLoaded)).sort((a, b) => a.content.length - b.content.length));
|
|
176
|
-
setSuggestionsLoading(false);
|
|
177
|
-
logPerfMetric(PerfMetricsEvents.FirstSuggestionsCompleted);
|
|
178
|
-
}, [
|
|
179
|
-
logPerfMetric,
|
|
180
|
-
setSuggestionsLoading,
|
|
181
|
-
setSuggestions,
|
|
182
|
-
settingsContext.generationParams
|
|
183
|
-
]);
|
|
184
|
-
const getResponses = useCallback(async (payload) => {
|
|
185
|
-
try {
|
|
186
|
-
const requestPayload = payload ?? createResponsePayload({
|
|
187
|
-
userEvents,
|
|
188
|
-
generationParams: settingsContext.generationParams
|
|
189
|
-
});
|
|
190
|
-
setResponseStreaming(true);
|
|
191
|
-
setSuggestions([]);
|
|
192
|
-
const startTimeMs = Date.now();
|
|
193
|
-
await getStreamingResponses(requestPayload);
|
|
194
|
-
recordAssistantResponse(startTimeMs, requestPayload, track);
|
|
195
|
-
await getSuggestions();
|
|
196
|
-
} catch (error) {
|
|
197
|
-
logger_default.logError("[spiffy-ai] getResponses error", error);
|
|
198
|
-
} finally {
|
|
199
|
-
markUserEventsProcessed(userEvents.map(({ eventId }) => eventId));
|
|
200
|
-
setUserHasReplied(false);
|
|
201
|
-
setResponseStreaming(false);
|
|
202
|
-
}
|
|
203
|
-
}, [
|
|
204
|
-
userEvents,
|
|
205
|
-
settingsContext.generationParams,
|
|
206
|
-
setResponseStreaming,
|
|
207
|
-
setSuggestions,
|
|
208
|
-
getStreamingResponses,
|
|
209
|
-
markUserEventsProcessed,
|
|
210
|
-
getSuggestions,
|
|
211
|
-
setUserHasReplied,
|
|
212
|
-
track
|
|
213
|
-
]);
|
|
214
|
-
useEffect(() => {
|
|
215
|
-
const processUserEvents = async () => {
|
|
216
|
-
if (responseStreaming || !widgetInitialized) return;
|
|
217
|
-
if (variantInfo.variant === "pdp" && !variantInfo.productId || variantInfo.variant === "plp" && !variantInfo.plpId || variantInfo.variant === "page_visit" && !variantInfo.url) {
|
|
218
|
-
logger_default.logDebug("[spiffy-ai] variantInfo has invalid values, skipping...", {
|
|
219
|
-
variantInfo,
|
|
220
|
-
supportedEvent
|
|
221
|
-
});
|
|
222
|
-
return;
|
|
223
|
-
}
|
|
224
|
-
logger_default.logDebug(`Assistants Turn is_currently_streaming=${responseStreaming} initialized=${widgetInitialized}`);
|
|
225
|
-
try {
|
|
226
|
-
await getResponses();
|
|
227
|
-
logger_default.logInfo(`Assistants Turn [finished]`);
|
|
228
|
-
} catch (error) {
|
|
229
|
-
logger_default.logError("[spiffy-ai] Assistants Turn error", error);
|
|
230
|
-
}
|
|
231
|
-
};
|
|
232
|
-
if (userQueueEventCount > 0) processUserEvents();
|
|
233
|
-
}, [
|
|
234
|
-
getResponses,
|
|
235
|
-
responseStreaming,
|
|
236
|
-
userQueueEventCount,
|
|
237
|
-
widgetInitialized,
|
|
238
|
-
variantInfo,
|
|
239
|
-
supportedEvent
|
|
240
|
-
]);
|
|
241
|
-
useEffect(() => {
|
|
242
|
-
if (widgetInitialized || responseStreaming) {
|
|
243
|
-
logger_default.logDebug(`[spiffy-ai] initializeWidget [skipped] is_currently_streaming=${responseStreaming} is_initialized=${widgetInitialized}`);
|
|
244
|
-
return;
|
|
245
|
-
}
|
|
246
|
-
const hydrateChat = async () => {
|
|
247
|
-
try {
|
|
248
|
-
logger_default.logDebug(`[spiffy-ai] initializeWidget is_currently_streaming=${responseStreaming} is_initialized=${widgetInitialized}`);
|
|
249
|
-
const { messages: existingMessages, userEvents: userEvents$1 } = await commerce_api_default.getResponses(orgId, chatId, userId);
|
|
250
|
-
setMessages([...existingMessages]);
|
|
251
|
-
setUserEvents([...userEvents$1]);
|
|
252
|
-
getResponses();
|
|
253
|
-
} catch (error) {
|
|
254
|
-
logger_default.logInfo(`Init chat [exception] chat_id=${chatId} error=${error}`, error);
|
|
255
|
-
if (error instanceof SessionRestartRequired) {
|
|
256
|
-
const appLoadedEvent = createAppLoadedEvent();
|
|
257
|
-
const visitEvent = createVisitUserEvent({ variantInfo });
|
|
258
|
-
setMessages([]);
|
|
259
|
-
clearUserEventQueue();
|
|
260
|
-
if (visitEvent) getResponses(createResponsePayload({
|
|
261
|
-
userEvents: [appLoadedEvent, visitEvent],
|
|
262
|
-
generationParams: settingsContext.generationParams
|
|
263
|
-
}));
|
|
264
|
-
}
|
|
265
|
-
} finally {
|
|
266
|
-
setWidgetInitialized(true);
|
|
267
|
-
}
|
|
268
|
-
};
|
|
269
|
-
hydrateChat();
|
|
270
|
-
}, []);
|
|
271
|
-
const onFocus = useCallback(async () => {
|
|
272
|
-
try {
|
|
273
|
-
if (!responseStreaming && !suggestionsLoading && orgId) {
|
|
274
|
-
const { messages: existingMessages } = await commerce_api_default.getResponses(orgId, chatId, userId);
|
|
275
|
-
if (existingMessages.length > messages.length) setMessages([...existingMessages]);
|
|
276
|
-
}
|
|
277
|
-
} catch (error) {
|
|
278
|
-
logger_default.logError("[spiffy-ai] onFocus error", error);
|
|
279
|
-
}
|
|
280
|
-
}, [
|
|
281
|
-
responseStreaming,
|
|
282
|
-
suggestionsLoading,
|
|
283
|
-
orgId,
|
|
284
|
-
chatId,
|
|
285
|
-
userId,
|
|
286
|
-
messages.length,
|
|
287
|
-
setMessages
|
|
288
|
-
]);
|
|
289
|
-
useEffect(() => {
|
|
290
|
-
window.addEventListener("focus", onFocus);
|
|
291
|
-
return () => {
|
|
292
|
-
window.removeEventListener("focus", onFocus);
|
|
293
|
-
};
|
|
294
|
-
}, [onFocus]);
|
|
295
|
-
const chatContext = useMemo(() => ({}), []);
|
|
296
|
-
return /* @__PURE__ */ jsx(ChatContext.Provider, {
|
|
297
|
-
value: chatContext,
|
|
298
|
-
children
|
|
299
|
-
});
|
|
300
|
-
};
|
|
301
|
-
|
|
302
|
-
//#endregion
|
|
303
|
-
export { ChatContext, ChatContextProvider };
|
|
304
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"chatContext.js","names":["userQueryProperty: string | undefined","eventProps: Record<string, unknown>","uuid","lastMessage: Message | undefined","error: unknown","CommerceApiClient","userEvents"],"sources":["../../../src/contexts/chatContext/chatContext.tsx"],"sourcesContent":["import { UserEventCategory } from '@spiffy-ai/commerce-api-client';\nimport { useAtom, useAtomValue, useSetAtom } from 'jotai';\nimport { ReactNode, createContext, useCallback, useEffect, useMemo, useState } from 'react';\nimport CommerceApiClient from 'src/application/commerce-api';\nimport { v4 as uuid } from 'uuid';\nimport { SessionRestartRequired } from 'src/types/exceptions/sessionExceptions';\nimport Logger from 'src/application/logging/logger';\nimport {\n  Message,\n  MessageRole,\n  MessageType,\n  NextMessageRequest,\n  Response,\n} from 'src/application/models';\nimport { SpiffyMetricsEventName } from 'src/contexts/amplitudeContext/amplitudeContext';\nimport { messageFromResponse } from 'src/application/utils';\nimport { chatIdAtom, userIdAtom, variantInfoAtom } from 'src/atoms/app';\nimport {\n  PerfMetricsEvents,\n  chatAtom,\n  logPerfMetricAtom,\n  messagesAtom,\n  requestFailureAtom,\n  responseStreamingAtom,\n  suggestionsAtom,\n  suggestionsLoadingAtom,\n  userEventsAtom,\n  userHasRepliedAtom,\n} from 'src/atoms/chat';\nimport {\n  chatSearchIsLoadingAtom,\n  chatSearchProductSortingAtom,\n  chatSearchProducts,\n  chatSearchStateAtom,\n  handleSearchResultsAtom,\n} from 'src/atoms/search/chatSearch';\nimport { createAppLoadedEvent, createVisitUserEvent } from 'src/hooks/utils';\nimport { useMessageInterceptor } from 'src/interceptors/useMessageInterceptor';\nimport { supportedEventAtom } from 'src/atoms/app/variant';\nimport { getAtomStore } from 'src/atoms/atomStore';\nimport {\n  clearUserEventAtom,\n  createResponsePayload,\n  processUserEventAtom,\n  userEventQueueAtom,\n  userQueueEventCountAtom,\n} from 'src/atoms/chat/messageQueue';\nimport { useAmplitudeTracking } from 'src/hooks/AmplitudeOperations/useAmplitudeOperations';\nimport { useSystemSettingsContext } from 'src/hooks/SystemSettingsContext';\n\n/**\n * Record the chat assistant response in Amplitude\n *\n * @param startTimeMs The start time of the assistant response\n * @param payload The payload used to generate the response\n */\nconst recordAssistantResponse = (\n  startTimeMs: number,\n  payload: NextMessageRequest,\n  track: (eventName: SpiffyMetricsEventName, eventProps?: Record<string, unknown>) => void,\n) => {\n  const atomStore = getAtomStore();\n  const chatState = atomStore.get(chatAtom);\n  const chatSearchState = atomStore.get(chatSearchStateAtom);\n  const searchProducts = atomStore.get(chatSearchProducts);\n  const searchProductsSort = atomStore.get(chatSearchProductSortingAtom);\n  const assistantResponseTimeMs = { start: startTimeMs, end: Date.now() };\n  let userQueryProperty: string | undefined;\n\n  if (\n    chatState.replyEventCategory === UserEventCategory.SuggestionClicked &&\n    chatState.suggestion\n  ) {\n    userQueryProperty = chatState.suggestion.content;\n  } else if (chatState.userQuery && chatState.userQuery.length > 0) {\n    userQueryProperty = chatState.userQuery;\n  }\n\n  const eventProps: Record<string, unknown> = {\n    response_time_ms: assistantResponseTimeMs.end - assistantResponseTimeMs.start,\n    user_event_type: chatState.replyEventCategory,\n    user_query: userQueryProperty,\n  };\n\n  if (chatState.replyEventCategory === UserEventCategory.FormSubmitted) {\n    const lastAssistantTurn = chatState.messages\n      .filter(turn => turn.length > 0 && turn[0].role === MessageRole.Assistant)\n      .pop();\n    const formType = payload.userEvents?.find(\n      event => event.category === UserEventCategory.FormSubmitted,\n    )?.attributes.formType;\n    const formStatus = lastAssistantTurn?.some(response => response.type === MessageType.Order);\n    eventProps.form_submitted_attributes = {\n      form_type: formType,\n      status: formStatus ? 'success' : 'failed',\n    };\n  }\n\n  if (chatSearchState === 'product-page') {\n    eventProps.search_products_returned = searchProducts.length;\n    eventProps.search_products_sort_type = searchProductsSort;\n  }\n\n  track(SpiffyMetricsEventName.ChatAssistantResponse, {\n    eventProps,\n  });\n};\n\ninterface ChatContextParams {}\n\nconst ChatContext = createContext<ChatContextParams | undefined>(undefined);\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 handleStreamingError = (\n  _error: unknown,\n  setRequestFailure: (failed: boolean) => void,\n  setMessages: (updater: (prev: Message[][]) => Message[][]) => void,\n) => {\n  setRequestFailure(true);\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:\n            \"I'm sorry! I'm having trouble right now. Please refresh the page or try again in a moment.\",\n        },\n      },\n    ],\n  ]);\n};\n\nconst processStreamingResponse = async (\n  stream: AsyncIterable<Response>,\n  messageInterceptor: {\n    intercept: (response?: Response) => boolean | undefined;\n  },\n  handleSearchResults: (message: Message) => void,\n  setMessages: (updater: (prev: Message[][]) => Message[][]) => void,\n  setSearchIsLoading: (loading: boolean) => void,\n  chatId: string,\n): Promise<{ hasSearchResults: boolean }> => {\n  let lastMessage: Message | undefined;\n  let hasSearchResults = false;\n\n  for await (const response of stream) {\n    try {\n      if (messageInterceptor.intercept(response)) {\n        return { hasSearchResults };\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      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(\n        `[spiffy-ai] Failed to generate responses from stream chat_id=${chatId}`,\n        error,\n        {\n          lastResponse: lastMessage,\n          response,\n        },\n      );\n    }\n  }\n\n  return { hasSearchResults };\n};\n\nconst ChatContextProvider = ({ children }: { children: ReactNode }) => {\n  const logPerfMetric = useSetAtom(logPerfMetricAtom);\n  const [widgetInitialized, setWidgetInitialized] = useState(false);\n  const setUserHasReplied = useSetAtom(userHasRepliedAtom);\n  // TODO: create atoms for setting/getting the last message turn\n  const [messages, setMessages] = useAtom<Message[][]>(messagesAtom);\n  const setUserEvents = useSetAtom(userEventsAtom);\n  const setSuggestions = useSetAtom(suggestionsAtom);\n  const [suggestionsLoading, setSuggestionsLoading] = useAtom<boolean>(suggestionsLoadingAtom);\n  const [responseStreaming, setResponseStreaming] = useAtom<boolean>(responseStreamingAtom);\n  const setRequestFailure = useSetAtom(requestFailureAtom);\n  const userEvents = useAtomValue(userEventQueueAtom);\n  const userQueueEventCount = useAtomValue(userQueueEventCountAtom);\n  const markUserEventsProcessed = useSetAtom(processUserEventAtom);\n  const clearUserEventQueue = useSetAtom(clearUserEventAtom);\n  const userId = useAtomValue(userIdAtom);\n  const chatId = useAtomValue(chatIdAtom);\n  const supportedEvent = useAtomValue(supportedEventAtom);\n  // TODO: Replace with actual orgId from useEnviveConfig or NewOrgConfigContext when available\n  const orgId = 'mock-org-id';\n\n  const variantInfo = useAtomValue(variantInfoAtom);\n  const settingsContext = useSystemSettingsContext();\n  const messageInterceptor = useMessageInterceptor();\n  const handleSearchResults = useSetAtom(handleSearchResultsAtom);\n  const setSearchIsLoading = useSetAtom(chatSearchIsLoadingAtom);\n  const { track } = useAmplitudeTracking();\n\n  const getStreamingResponses = useCallback(\n    async (payload: NextMessageRequest): Promise<{ hasSearchResults: boolean }> => {\n      logPerfMetric(PerfMetricsEvents.FirstResponseStarted);\n      const stream = CommerceApiClient.getNextResponseStreaming(payload);\n\n      try {\n        setRequestFailure(false);\n\n        const { hasSearchResults } = await processStreamingResponse(\n          stream,\n          messageInterceptor,\n          handleSearchResults,\n          setMessages,\n          setSearchIsLoading,\n          chatId,\n        );\n\n        return { hasSearchResults };\n      } catch (e) {\n        handleStreamingError(e, setRequestFailure, setMessages);\n        throw e;\n      } finally {\n        logPerfMetric(PerfMetricsEvents.FirstResponseCompleted);\n      }\n    },\n    [\n      logPerfMetric,\n      setRequestFailure,\n      messageInterceptor,\n      handleSearchResults,\n      setMessages,\n      setSearchIsLoading,\n      chatId,\n    ],\n  );\n\n  const getSuggestions = useCallback(async () => {\n    logPerfMetric(PerfMetricsEvents.FirstSuggestionsStarted);\n    setSuggestionsLoading(true);\n    setSuggestions([]);\n\n    const payloadWithoutAppLoaded = createResponsePayload({\n      userEvents: [],\n      generationParams: settingsContext.generationParams,\n    });\n    const response = await CommerceApiClient.getNextSuggestions(payloadWithoutAppLoaded);\n\n    // sort the suggestions by shortest length so the pills can be stacked horizontally\n    setSuggestions(response.sort((a, b) => a.content.length - b.content.length));\n    setSuggestionsLoading(false);\n    logPerfMetric(PerfMetricsEvents.FirstSuggestionsCompleted);\n  }, [logPerfMetric, setSuggestionsLoading, setSuggestions, settingsContext.generationParams]);\n\n  const getResponses = useCallback(\n    async (payload?: NextMessageRequest) => {\n      try {\n        const requestPayload =\n          payload ??\n          createResponsePayload({\n            userEvents,\n            generationParams: settingsContext.generationParams,\n          });\n\n        setResponseStreaming(true);\n        setSuggestions([]);\n        const startTimeMs = Date.now();\n\n        await getStreamingResponses(requestPayload);\n\n        recordAssistantResponse(startTimeMs, requestPayload, track);\n        await getSuggestions();\n      } catch (error) {\n        Logger.logError('[spiffy-ai] getResponses error', error);\n      } finally {\n        // Remove search loading management from here - it's now handled independently\n        // in the processStreamingResponse function when search results are detected\n        markUserEventsProcessed(userEvents.map(({ eventId }) => eventId));\n        setUserHasReplied(false);\n        setResponseStreaming(false);\n      }\n    },\n    [\n      userEvents,\n      settingsContext.generationParams,\n      setResponseStreaming,\n      setSuggestions,\n      getStreamingResponses,\n      markUserEventsProcessed,\n      getSuggestions,\n      setUserHasReplied,\n      track,\n    ],\n  );\n\n  useEffect(() => {\n    const processUserEvents = async () => {\n      if (responseStreaming || !widgetInitialized) {\n        return;\n      }\n\n      if (\n        (variantInfo.variant === 'pdp' && !variantInfo.productId) ||\n        (variantInfo.variant === 'plp' && !variantInfo.plpId) ||\n        (variantInfo.variant === 'page_visit' && !variantInfo.url)\n      ) {\n        Logger.logDebug('[spiffy-ai] variantInfo has invalid values, skipping...', {\n          variantInfo,\n          supportedEvent,\n        });\n        return;\n      }\n\n      Logger.logDebug(\n        `Assistants Turn is_currently_streaming=${responseStreaming} initialized=${widgetInitialized}`,\n      );\n      try {\n        await getResponses();\n        Logger.logInfo(`Assistants Turn [finished]`);\n      } catch (error: unknown) {\n        Logger.logError('[spiffy-ai] Assistants Turn error', error);\n      }\n    };\n    if (userQueueEventCount > 0) {\n      processUserEvents();\n    }\n  }, [\n    getResponses,\n    responseStreaming,\n    userQueueEventCount,\n    widgetInitialized,\n    variantInfo,\n    supportedEvent,\n  ]);\n\n  useEffect(() => {\n    if (widgetInitialized || responseStreaming) {\n      Logger.logDebug(\n        `[spiffy-ai] initializeWidget [skipped] is_currently_streaming=${responseStreaming} is_initialized=${widgetInitialized}`,\n      );\n      return;\n    }\n\n    const hydrateChat = async () => {\n      try {\n        Logger.logDebug(\n          `[spiffy-ai] initializeWidget is_currently_streaming=${responseStreaming} is_initialized=${widgetInitialized}`,\n        );\n        // on mount, try to get the responses from an active session if one exists\n        if (!orgId) {\n          throw new Error('orgId is not available');\n        }\n        const { messages: existingMessages, userEvents } = await CommerceApiClient.getResponses(\n          orgId,\n          chatId,\n          userId,\n        );\n        setMessages([...existingMessages]);\n        setUserEvents([...userEvents]);\n        getResponses();\n      } catch (error) {\n        // no active chat session was found, start a new one\n        Logger.logInfo(`Init chat [exception] chat_id=${chatId} error=${error}`, error);\n        if (error instanceof SessionRestartRequired) {\n          const appLoadedEvent = createAppLoadedEvent();\n          const visitEvent = createVisitUserEvent({ variantInfo });\n          setMessages([]);\n          clearUserEventQueue();\n          if (visitEvent) {\n            const payload = createResponsePayload({\n              userEvents: [appLoadedEvent, visitEvent],\n              generationParams: settingsContext.generationParams,\n            });\n            getResponses(payload);\n          }\n        }\n      } finally {\n        setWidgetInitialized(true);\n      }\n    };\n\n    hydrateChat();\n  }, []);\n\n  const onFocus = useCallback(async () => {\n    try {\n      if (!responseStreaming && !suggestionsLoading && orgId) {\n        const { messages: existingMessages } = await CommerceApiClient.getResponses(\n          orgId,\n          chatId,\n          userId,\n        );\n\n        if (existingMessages.length > messages.length) {\n          setMessages([...existingMessages]);\n          // TODO: Is this bug hack still necessary?\n          // If it is, come up with a better solution for it\n          // } else if (existingMessages.length === 0) {\n          //   // if there was an error during the initialization of a new session, the session would be\n          //   // created in the backend but without any messages. Retry the next_responses request with\n          //   // the current set of parameters to \"jumpstart\" the session\n          //   triggerGetResponseCall('onFocus');\n        }\n      }\n    } catch (error: unknown) {\n      Logger.logError('[spiffy-ai] onFocus error', error);\n    }\n  }, [responseStreaming, suggestionsLoading, orgId, chatId, userId, messages.length, setMessages]);\n\n  // listen for page focus to get latest messages from the server\n  useEffect(() => {\n    window.addEventListener('focus', onFocus);\n\n    return () => {\n      window.removeEventListener('focus', onFocus);\n    };\n  }, [onFocus]);\n\n  const chatContext = useMemo(() => ({}), []);\n\n  return <ChatContext.Provider value={chatContext}>{children}</ChatContext.Provider>;\n};\n\nexport { ChatContext, ChatContextProvider };\nexport type { ChatContextParams };\n"],"mappings":"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"}
|