@envive-ai/react-hooks 0.2.6-alpha-3 → 0.2.6-alpha-4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (144) hide show
  1. package/dist/{AmplitudeOperations-DcuLn1Zk.cjs → AmplitudeOperations-BEhMCOtx.cjs} +1 -1
  2. package/dist/{AmplitudeOperations-C3i2q2Bn.js → AmplitudeOperations-Dpp91XWV.js} +1 -1
  3. package/dist/{NewOrgConfig-BtvujFC8.js → NewOrgConfig-Cx3R7FM6.js} +2 -2
  4. package/dist/{NewOrgConfig-CHt7Xg4t.cjs → NewOrgConfig-DRs9Sk4E.cjs} +2 -2
  5. package/dist/{TrackComponentVisibleEvent-Be7wfXXY.js → TrackComponentVisibleEvent-CIfRZtFq.js} +2 -2
  6. package/dist/{TrackComponentVisibleEvent-KUihr_y4.cjs → TrackComponentVisibleEvent-OAFP4CZ6.cjs} +2 -2
  7. package/dist/{amplitudeContext-CRFxGj_x.d.ts → amplitudeContext-CMTvCsJC.d.ts} +1 -1
  8. package/dist/amplitudeContext-ZkerrMUa.d.cts +1 -1
  9. package/dist/amplitudeTrackEventAtom-BBkB1LPQ.cjs +15 -0
  10. package/dist/amplitudeTrackEventAtom-yP38BuDB.js +8 -0
  11. package/dist/application/models/graphql/index.d.ts +1 -1
  12. package/dist/application/models/guards/api/index.d.ts +2 -2
  13. package/dist/application/models/guards/utils.d.ts +1 -1
  14. package/dist/application/models/index.d.cts +1 -1
  15. package/dist/application/models/index.d.ts +4 -4
  16. package/dist/application/utils/index.d.ts +5 -5
  17. package/dist/atoms/app/index.d.cts +7 -7
  18. package/dist/atoms/app/index.d.ts +6 -6
  19. package/dist/atoms/chat/index.cjs +3 -2
  20. package/dist/atoms/chat/index.d.cts +32 -32
  21. package/dist/atoms/chat/index.d.ts +34 -34
  22. package/dist/atoms/chat/index.js +3 -2
  23. package/dist/atoms/globalSearch/index.d.cts +6 -6
  24. package/dist/atoms/globalSearch/index.d.ts +1 -1
  25. package/dist/atoms/org/index.d.cts +16 -16
  26. package/dist/atoms/org/index.d.ts +17 -17
  27. package/dist/atoms/search/index.cjs +3 -3
  28. package/dist/atoms/search/index.d.cts +15 -15
  29. package/dist/atoms/search/index.d.ts +14 -14
  30. package/dist/atoms/search/index.js +3 -3
  31. package/dist/atoms/search/utils.cjs +1 -1
  32. package/dist/atoms/search/utils.js +1 -1
  33. package/dist/{cdnContext-B2-NiEiW.js → cdnContext-BJZSYlsP.js} +1 -1
  34. package/dist/{cdnContext-CdhppNeT.cjs → cdnContext-KMAzFHrZ.cjs} +1 -1
  35. package/dist/chat-DYD6B6em.js +227 -0
  36. package/dist/chat-yICI8tzz.cjs +327 -0
  37. package/dist/{chatState-DqWSgmjC.cjs → chatState-BRYkF2g2.cjs} +1 -1
  38. package/dist/{chatState-DxG0t4fK.js → chatState-DRKB9bqM.js} +1 -1
  39. package/dist/config/index.d.ts +4 -4
  40. package/dist/config/locators/components/chat/index.d.ts +1 -1
  41. package/dist/config/locators/components/common/index.d.ts +1 -1
  42. package/dist/config/locators/components/index.d.ts +1 -1
  43. package/dist/config/locators/index.d.ts +4 -4
  44. package/dist/contexts/amplitudeContext/index.d.ts +1 -1
  45. package/dist/contexts/cdnContext/index.cjs +1 -1
  46. package/dist/contexts/cdnContext/index.d.cts +1 -1
  47. package/dist/contexts/cdnContext/index.js +1 -1
  48. package/dist/contexts/enviveConfigContext/index.d.cts +1 -1
  49. package/dist/contexts/enviveCssContext/index.cjs +5 -5
  50. package/dist/contexts/enviveCssContext/index.js +5 -5
  51. package/dist/contexts/featureFlagServiceContext/index.cjs +1 -1
  52. package/dist/contexts/featureFlagServiceContext/index.d.ts +1 -1
  53. package/dist/contexts/featureFlagServiceContext/index.js +1 -1
  54. package/dist/contexts/graphqlContext/index.cjs +1 -1
  55. package/dist/contexts/graphqlContext/index.js +1 -1
  56. package/dist/contexts/newOrgConfigContext/index.cjs +4 -4
  57. package/dist/contexts/newOrgConfigContext/index.js +4 -4
  58. package/dist/contexts/searchContext/index.cjs +5 -5
  59. package/dist/contexts/searchContext/index.js +5 -5
  60. package/dist/contexts/shopifyUrlContext/index.cjs +1 -1
  61. package/dist/contexts/shopifyUrlContext/index.js +1 -1
  62. package/dist/contexts/systemSettingsContext/index.cjs +1 -1
  63. package/dist/contexts/systemSettingsContext/index.d.cts +2 -2
  64. package/dist/contexts/systemSettingsContext/index.d.ts +2 -2
  65. package/dist/contexts/systemSettingsContext/index.js +1 -1
  66. package/dist/exceptions/index.d.cts +1 -1
  67. package/dist/{featureFlagServiceContext-ahy3jou0.d.ts → featureFlagServiceContext-DiYIv0jI.d.ts} +1 -1
  68. package/dist/{graphqlContext-CU7NIJo7.cjs → graphqlContext-BVFRNIdi.cjs} +1 -1
  69. package/dist/{graphqlContext-BuXJ5DiW.js → graphqlContext-D_Wp0LEl.js} +1 -1
  70. package/dist/hooks/AmplitudeOperations/index.cjs +1 -1
  71. package/dist/hooks/AmplitudeOperations/index.js +1 -1
  72. package/dist/hooks/AppDetails/index.cjs +3 -3
  73. package/dist/hooks/AppDetails/index.d.cts +1 -1
  74. package/dist/hooks/AppDetails/index.js +3 -3
  75. package/dist/hooks/CdnOperations/index.cjs +1 -1
  76. package/dist/hooks/CdnOperations/index.d.cts +1 -1
  77. package/dist/hooks/CdnOperations/index.d.ts +1 -1
  78. package/dist/hooks/CdnOperations/index.js +1 -1
  79. package/dist/hooks/ChatToggle/index.cjs +4 -3
  80. package/dist/hooks/ChatToggle/index.js +4 -3
  81. package/dist/hooks/ChatToggleAnalytics/index.cjs +2 -2
  82. package/dist/hooks/ChatToggleAnalytics/index.js +2 -2
  83. package/dist/hooks/ElementObserver/index.d.ts +1 -1
  84. package/dist/hooks/GraphQLConfig/index.cjs +2 -2
  85. package/dist/hooks/GraphQLConfig/index.js +2 -2
  86. package/dist/hooks/IdentifyUser/index.d.ts +1 -1
  87. package/dist/hooks/Intersection/index.cjs +1 -1
  88. package/dist/hooks/Intersection/index.js +1 -1
  89. package/dist/hooks/NewOrgConfig/index.cjs +5 -5
  90. package/dist/hooks/NewOrgConfig/index.d.cts +2 -2
  91. package/dist/hooks/NewOrgConfig/index.js +5 -5
  92. package/dist/hooks/Search/index.cjs +32 -58
  93. package/dist/hooks/Search/index.js +33 -59
  94. package/dist/hooks/SearchOperations/index.cjs +5 -5
  95. package/dist/hooks/SearchOperations/index.js +5 -5
  96. package/dist/hooks/ShopifyUrlOperations/index.cjs +1 -1
  97. package/dist/hooks/ShopifyUrlOperations/index.d.cts +2 -2
  98. package/dist/hooks/ShopifyUrlOperations/index.d.ts +2 -2
  99. package/dist/hooks/ShopifyUrlOperations/index.js +1 -1
  100. package/dist/hooks/SystemSettingsContext/index.cjs +1 -1
  101. package/dist/hooks/SystemSettingsContext/index.d.cts +4 -4
  102. package/dist/hooks/SystemSettingsContext/index.d.ts +2 -2
  103. package/dist/hooks/SystemSettingsContext/index.js +1 -1
  104. package/dist/hooks/TrackComponentVisibleEvent/index.cjs +2 -2
  105. package/dist/hooks/TrackComponentVisibleEvent/index.js +2 -2
  106. package/dist/{index-dngXrfnT.d.ts → index-BO-ZLYtR.d.ts} +1 -1
  107. package/dist/{index-B4cSHxVN.d.ts → index-BR1G8yyg.d.ts} +1 -1
  108. package/dist/index-BVZbvpx_.d.cts +1 -1
  109. package/dist/{index-DeQte6mb.d.ts → index-CAhGZxMI.d.ts} +1 -1
  110. package/dist/{index-CYGpI6hE.d.ts → index-DNccxbJi.d.ts} +1 -1
  111. package/dist/{index-CEmUfoZa.d.ts → index-Dgu085Lu.d.ts} +2 -2
  112. package/dist/{index-Clf4wYaJ.d.ts → index-DiIHuPq2.d.ts} +1 -1
  113. package/dist/{index-Dzv6WwSZ.d.ts → index-ZQMda2Iu.d.ts} +1 -1
  114. package/dist/index-ca7Qn8o0.d.cts +1 -1
  115. package/dist/interceptors/index.d.cts +1 -1
  116. package/dist/interceptors/index.d.ts +1 -1
  117. package/dist/{newOrgConfigContext-lw85mMS-.cjs → newOrgConfigContext-DkBvmY0p.cjs} +2 -2
  118. package/dist/{newOrgConfigContext-DPUxIF2G.js → newOrgConfigContext-Dn9l1f9w.js} +2 -2
  119. package/dist/{nodeSelector-CbWcUbuc.d.ts → nodeSelector-0gJ8Xayf.d.ts} +1 -1
  120. package/dist/{search-D1cU29CV.cjs → search-D-BGpGJt.cjs} +3 -3
  121. package/dist/{search-BzXoTgRx.js → search-jNTQGSyl.js} +3 -3
  122. package/dist/{searchContext-R_XuFzYe.cjs → searchContext-C75BHjlY.cjs} +3 -3
  123. package/dist/{searchContext-C8DpjnqD.js → searchContext-DjcqQsxp.js} +3 -3
  124. package/dist/{searchServiceAdapter-CKc7UXKi.js → searchServiceAdapter-BSPZOg1r.js} +1 -1
  125. package/dist/{searchServiceAdapter-pROGm3Rf.cjs → searchServiceAdapter-DrjFCiw8.cjs} +1 -1
  126. package/dist/{shopifyUrlContext-WAm0pnPh.js → shopifyUrlContext-BI3fVtA5.js} +1 -1
  127. package/dist/{shopifyUrlContext-G9eqIcsL.cjs → shopifyUrlContext-Ba6MQdNV.cjs} +1 -1
  128. package/dist/{systemSettingsContext-J1sny8W8.cjs → systemSettingsContext-CSXsgBiG.cjs} +1 -1
  129. package/dist/{systemSettingsContext-B-FZ8UAg.js → systemSettingsContext-DTdzjE_O.js} +1 -1
  130. package/dist/types/index.d.ts +1 -1
  131. package/dist/{useAppDetails-Bw2jJ39H.cjs → useAppDetails-Cq8TbvkX.cjs} +2 -2
  132. package/dist/{useAppDetails-BtK9ZZJs.js → useAppDetails-DjdYWTqz.js} +2 -2
  133. package/dist/{useGraphQLConfig-Dc8h2yyN.cjs → useGraphQLConfig-Dy0ml9xw.cjs} +2 -2
  134. package/dist/{useGraphQLConfig-DuLB3tdv.js → useGraphQLConfig-nY5CE8kj.js} +2 -2
  135. package/dist/{useIntersection-DZHz44BY.cjs → useIntersection-BQMfiS4x.cjs} +1 -1
  136. package/dist/{useIntersection-BWiHkpKL.js → useIntersection-D-ol9eH8.js} +1 -1
  137. package/dist/{utils-BivlGkeU.cjs → utils-BBICrPjW.cjs} +1 -1
  138. package/dist/{utils-wWki3L1d.d.ts → utils-sosM0bEk.d.ts} +1 -1
  139. package/dist/{utils-CExht7Uj.js → utils-w4-xONRA.js} +1 -1
  140. package/package.json +1 -1
  141. package/dist/chat-C9KdnT5V.js +0 -230
  142. package/dist/chat-bImNRcnQ.cjs +0 -330
  143. /package/dist/{featureFlagServiceContext-CbUOAw5l.js → featureFlagServiceContext-DgoR6euC.js} +0 -0
  144. /package/dist/{featureFlagServiceContext-DSo2brno.cjs → featureFlagServiceContext-b-rYgf0u.cjs} +0 -0
@@ -1,230 +0,0 @@
1
- import { MessageType as MessageType$1 } from "./dist-CtkINi1R.js";
2
- import { MessageRole, MessageType } from "./types-D5du68Vp.js";
3
- import { logger_default } from "./logger-BMVdhQOV.js";
4
- import { getAtomStore, sessionStorageUtil } from "./atomStore-BuopbV9k.js";
5
- import { SpiffyMetricsEventName } from "./utils-IogJwDB9.js";
6
- import { askQuestionBtnClickedAtom, chatIsOpenAtom, chatOnToggleAtom, formSubmitAtom, messagesAtom, replyEventCategoryAtom, responseStreamingAtom, suggestionAtom, suggestionsAtom, suggestionsLoadingAtom, userEventsAtom, userHasRepliedAtom, userQueryAtom } from "./chatState-DxG0t4fK.js";
7
- import { ContextEnvEnum, ContextSourceEnum, UserEventCategory } from "@spiffy-ai/commerce-api-client";
8
- import { v4 } from "uuid";
9
- import { atom } from "jotai";
10
- import { atomWithStorage } from "jotai/utils";
11
-
12
- //#region src/atoms/chat/messageQueue.ts
13
- const internalUserEventQueueAtom = atom([]);
14
- const userEventQueueAtom = atom((get) => {
15
- const queue = get(internalUserEventQueueAtom);
16
- return queue === void 0 ? [] : queue.filter((v) => v !== void 0);
17
- });
18
- /**
19
- * This atom is used to queue a new message for processing on `next_responses`
20
- * It receives a single `userEvent` that is added to the processing queue.
21
- * If the event has the same eventId as an existing message in the queue the NEW
22
- * event is ignored
23
- */
24
- const queueUserEventAtom = atom(null, (get, set, userEvent) => {
25
- if (userEvent === void 0) return;
26
- set(internalUserEventQueueAtom, [...get(internalUserEventQueueAtom), userEvent]);
27
- });
28
- /**
29
- * This atom exposes a function to reset the entire queue. All messages in the queue will be purged
30
- */
31
- const clearUserEventAtom = atom(null, (_, set) => {
32
- set(internalUserEventQueueAtom, []);
33
- });
34
- /**
35
- * This atom is used to mark events as processed and remove them from the queue
36
- * It accepts a list of eventId values and will remove all events with those eventIds from the queue.
37
- */
38
- const processUserEventAtom = atom(null, (get, set, eventIds) => {
39
- const remaining = get(internalUserEventQueueAtom)?.filter((event) => !eventIds.includes(event.eventId));
40
- set(internalUserEventQueueAtom, remaining);
41
- });
42
- const userQueueEventCountAtom = atom((get) => get(userEventQueueAtom).length);
43
-
44
- //#endregion
45
- //#region src/atoms/amplitude/amplitudeTrackEventAtom.ts
46
- const amplitudeTrackEventAtom = atom(null);
47
-
48
- //#endregion
49
- //#region src/atoms/chat/replies.ts
50
- const handleReplyAtom = atom(null, (get, set, { message, userTyped }) => {
51
- if (message.type !== MessageType$1.QueryTyped) return;
52
- const trackEvent = get(amplitudeTrackEventAtom);
53
- const queryTyped = message.metadata.content;
54
- set(replyEventCategoryAtom, UserEventCategory.QueryTyped);
55
- set(userQueryAtom, queryTyped);
56
- set(messagesAtom, [...get(messagesAtom), [message]]);
57
- set(userHasRepliedAtom, true);
58
- set(queueUserEventAtom, {
59
- eventId: message.id,
60
- createdAt: message.createdAt,
61
- category: UserEventCategory.QueryTyped,
62
- attributes: { query: queryTyped }
63
- });
64
- if (trackEvent) trackEvent({
65
- eventName: SpiffyMetricsEventName.ChatUserMessageInput,
66
- eventProps: {
67
- message_id: message.id,
68
- message_role: message.role,
69
- message_type: message.type,
70
- message_metadata: {
71
- content: message?.metadata?.content,
72
- created_at: message.createdAt,
73
- user_typed: userTyped
74
- }
75
- },
76
- alsoSendToGoogleAnalytics: true
77
- });
78
- });
79
-
80
- //#endregion
81
- //#region src/atoms/chat/performanceMetrics.ts
82
- const APP_INITIAL_START_TIME_KEY = "spiffy-app-initial-start-time";
83
- const PAGE_LOAD_OFFSET_TIME_KEY = "spiffy-page-load-offset-time";
84
- /**
85
- * The different performance metrics that can be logged. All times are relative to the
86
- * initial start time of the app and are stored in milliseconds.
87
- */
88
- let PerfMetricsEvents = /* @__PURE__ */ function(PerfMetricsEvents$1) {
89
- PerfMetricsEvents$1["PageLoadOffset"] = "page_load_offset_ms";
90
- PerfMetricsEvents$1["MainBundleLoaded"] = "main_bundle_loaded_ms";
91
- PerfMetricsEvents$1["OrgConfigLoadStarted"] = "org_config_load_started_ms";
92
- PerfMetricsEvents$1["OrgConfigLoadEnded"] = "org_config_load_ended_ms";
93
- PerfMetricsEvents$1["FirstResponseStarted"] = "first_response_started_ms";
94
- PerfMetricsEvents$1["FirstResponseCompleted"] = "first_response_completed_ms";
95
- PerfMetricsEvents$1["FirstSuggestionsStarted"] = "first_suggestions_started_ms";
96
- PerfMetricsEvents$1["FirstSuggestionsCompleted"] = "first_suggestions_completed_ms";
97
- PerfMetricsEvents$1["EmbeddedWidgetRendered"] = "embedded_widget_rendered_ms";
98
- PerfMetricsEvents$1["FloatingButtonRendered"] = "floating_button_rendered_ms";
99
- PerfMetricsEvents$1["TopSuggestionsBarRendered"] = "top_suggestions_bar_rendered_ms";
100
- PerfMetricsEvents$1["BottomSuggestionsBarRendered"] = "bottom_suggestions_bar_rendered_ms";
101
- PerfMetricsEvents$1["SearchPromptRendered"] = "search_prompt_rendered_ms";
102
- return PerfMetricsEvents$1;
103
- }({});
104
- const internalPerfMetricsAtom = atom(/* @__PURE__ */ new Map());
105
- const appInitialStartTimeMsAtom = atomWithStorage(APP_INITIAL_START_TIME_KEY, sessionStorage.getItem(APP_INITIAL_START_TIME_KEY) ?? void 0, sessionStorageUtil, { getOnInit: true });
106
- const pageLoadOffsetTimeAtom = atomWithStorage(PAGE_LOAD_OFFSET_TIME_KEY, sessionStorage.getItem(PAGE_LOAD_OFFSET_TIME_KEY) ?? void 0, sessionStorageUtil, { getOnInit: true });
107
- /**
108
- * Resets the performance metrics atom to an empty map. This should be called after the performance
109
- * metrics have been reported to amplitude. On SPA, it ensures that we can still capture metrics as
110
- * the user navigates around the app (although there is more work to be done to fully enable this).
111
- * On non-SPA, it ensures that previously captured metrics are not reported again.
112
- */
113
- const resetPerformanceMetricsAtom = () => {
114
- getAtomStore().set(internalPerfMetricsAtom, /* @__PURE__ */ new Map());
115
- };
116
- const performanceMetricsAtom = atom((get) => get(internalPerfMetricsAtom));
117
- const hasReportedPerformanceMetricsAtom = atom(false);
118
- /**
119
- * Logs a performance metric by capturing the delta between the initial app start time
120
- * and the current time. If the metric has already been logged, it will not be logged again.
121
- *
122
- * @param value The performance metric name to log.
123
- */
124
- const logPerfMetricAtom = atom(null, (get, set, value) => {
125
- const initialTimeStorageValue = get(appInitialStartTimeMsAtom);
126
- const initialTimeMs = initialTimeStorageValue ? parseInt(initialTimeStorageValue, 10) : void 0;
127
- if (initialTimeMs == null) {
128
- logger_default.logWarn(`[spiffy-ai] No initial app start time found. Skipping...`, void 0);
129
- return;
130
- }
131
- const currentPerfMetrics = get(internalPerfMetricsAtom);
132
- if (currentPerfMetrics.has(value)) return;
133
- const deltaMs = Date.now() - initialTimeMs;
134
- currentPerfMetrics.set(value, deltaMs);
135
- set(internalPerfMetricsAtom, currentPerfMetrics);
136
- });
137
-
138
- //#endregion
139
- //#region src/atoms/chat/form.ts
140
- const handleFormSubmittedAtom = atom(null, (_, set, form) => {
141
- set(replyEventCategoryAtom, UserEventCategory.FormSubmitted);
142
- set(formSubmitAtom, form);
143
- const formUserEvent = {
144
- eventId: v4(),
145
- category: UserEventCategory.FormSubmitted,
146
- createdAt: (/* @__PURE__ */ new Date()).toISOString(),
147
- attributes: form
148
- };
149
- set(queueUserEventAtom, formUserEvent);
150
- });
151
-
152
- //#endregion
153
- //#region src/atoms/chat/suggestions.ts
154
- const handleSuggestionAtom = atom(null, (get, set, suggestion) => {
155
- const newMessage = {
156
- id: suggestion.id,
157
- role: MessageRole.User,
158
- type: MessageType.SuggestionClicked,
159
- createdAt: (/* @__PURE__ */ new Date()).toISOString(),
160
- metadata: {
161
- suggestionId: suggestion.id,
162
- suggestionContent: suggestion.content
163
- }
164
- };
165
- set(replyEventCategoryAtom, UserEventCategory.SuggestionClicked);
166
- set(suggestionAtom, suggestion);
167
- set(messagesAtom, [...get(messagesAtom), [newMessage]]);
168
- set(userHasRepliedAtom, true);
169
- set(queueUserEventAtom, {
170
- eventId: suggestion.id,
171
- category: UserEventCategory.SuggestionClicked,
172
- createdAt: newMessage.createdAt,
173
- attributes: { suggestionId: suggestion.id }
174
- });
175
- });
176
-
177
- //#endregion
178
- //#region src/atoms/chat/lastMessage.ts
179
- const lastAssistantMessageAtom = atom((get) => {
180
- const messages = get(messagesAtom);
181
- const userHasReplied = get(userHasRepliedAtom);
182
- if (messages.length > 0 && !userHasReplied) return messages[messages.length - 1];
183
- return null;
184
- });
185
-
186
- //#endregion
187
- //#region src/atoms/chat/renderedWidgetRefs.ts
188
- const internalWidgetArrayAtom = atom([]);
189
- const widgetArrayAtom = atom((get) => get(internalWidgetArrayAtom));
190
- /**
191
- * This function call is used to create a list of the Spiffy widgets
192
- * that are rendering on the page.
193
- *
194
- * It is used by the FloatingButton widget to slide out of view when it
195
- * overlaps with another Spiffy widget.
196
- *
197
- * The spiffy widgets should be added to the this array for the floating button
198
- * to know about them.
199
- *
200
- * Today the "SuggestionBar" widget is not added, but the other widgets are.
201
- *
202
- */
203
- const addWidget = (ref, idx) => {
204
- const atomStore = getAtomStore();
205
- const val = atomStore.get(internalWidgetArrayAtom);
206
- const insertIdx = idx ?? val.length;
207
- val[insertIdx] = ref;
208
- atomStore.set(internalWidgetArrayAtom, val);
209
- };
210
-
211
- //#endregion
212
- //#region src/atoms/chat/index.ts
213
- const chatAtom = atom((get) => ({
214
- userHasReplied: get(userHasRepliedAtom),
215
- replyEventCategory: get(replyEventCategoryAtom),
216
- userQuery: get(userQueryAtom),
217
- suggestion: get(suggestionAtom),
218
- askQuestionBtnClicked: get(askQuestionBtnClickedAtom),
219
- messages: get(messagesAtom),
220
- userEvents: get(userEventsAtom),
221
- suggestions: get(suggestionsAtom),
222
- suggestionsLoading: get(suggestionsLoadingAtom),
223
- responseStreaming: get(responseStreamingAtom),
224
- isOpen: get(chatIsOpenAtom),
225
- onToggle: get(chatOnToggleAtom)
226
- }));
227
-
228
- //#endregion
229
- export { APP_INITIAL_START_TIME_KEY, PAGE_LOAD_OFFSET_TIME_KEY, PerfMetricsEvents, addWidget, appInitialStartTimeMsAtom, chatAtom, handleFormSubmittedAtom, handleReplyAtom, handleSuggestionAtom, hasReportedPerformanceMetricsAtom, lastAssistantMessageAtom, logPerfMetricAtom, pageLoadOffsetTimeAtom, performanceMetricsAtom, resetPerformanceMetricsAtom, widgetArrayAtom };
230
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdC1DOUtkblQ1Vi5qcyIsIm5hbWVzIjpbIk1lc3NhZ2VUeXBlIiwidXVpZCIsIm5ld01lc3NhZ2U6IE1lc3NhZ2UiXSwic291cmNlcyI6WyIuLi9zcmMvYXRvbXMvY2hhdC9tZXNzYWdlUXVldWUudHMiLCIuLi9zcmMvYXRvbXMvYW1wbGl0dWRlL2FtcGxpdHVkZVRyYWNrRXZlbnRBdG9tLnRzIiwiLi4vc3JjL2F0b21zL2NoYXQvcmVwbGllcy50cyIsIi4uL3NyYy9hdG9tcy9jaGF0L3BlcmZvcm1hbmNlTWV0cmljcy50cyIsIi4uL3NyYy9hdG9tcy9jaGF0L2Zvcm0udHMiLCIuLi9zcmMvYXRvbXMvY2hhdC9zdWdnZXN0aW9ucy50cyIsIi4uL3NyYy9hdG9tcy9jaGF0L2xhc3RNZXNzYWdlLnRzIiwiLi4vc3JjL2F0b21zL2NoYXQvcmVuZGVyZWRXaWRnZXRSZWZzLnRzIiwiLi4vc3JjL2F0b21zL2NoYXQvaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgYXRvbSB9IGZyb20gJ2pvdGFpJztcbmltcG9ydCB7IHY0IGFzIHV1aWQgfSBmcm9tICd1dWlkJztcbmltcG9ydCB7IEdlbmVyYXRpb25QYXJhbXMsIE5leHRNZXNzYWdlUmVxdWVzdCwgVXNlckV2ZW50IH0gZnJvbSAnQGVudml2ZS1haS90eXBlcyc7XG5pbXBvcnQgeyBnZXRBdG9tU3RvcmUgfSBmcm9tICdzcmMvYXRvbXMvYXRvbVN0b3JlL2F0b21TdG9yZSc7XG5pbXBvcnQgeyB1c2VySWRBdG9tLCBjaGF0SWRBdG9tIH0gZnJvbSAnc3JjL2F0b21zL2FwcCc7XG5pbXBvcnQgeyBvcmdTaG9ydE5hbWVBdG9tLCBjb250ZXh0U291cmNlQXRvbSwgZW52QXRvbSB9IGZyb20gJ3NyYy9hdG9tcy9lbnZpdmUvZW52aXZlQ29uZmlnJztcbmltcG9ydCB7IG9yZ0lkQXRvbSwgZmVhdHVyZUZsYWdTZXJ2aWNlQXRvbSB9IGZyb20gJ3NyYy9hdG9tcy9vcmcvZ3JhcGhxbENvbmZpZyc7XG5cbmNvbnN0IGludGVybmFsVXNlckV2ZW50UXVldWVBdG9tID0gYXRvbTxVc2VyRXZlbnRbXT4oW10pO1xuXG5leHBvcnQgY29uc3QgdXNlckV2ZW50UXVldWVBdG9tID0gYXRvbSgoZ2V0KSA9PiB7XG4gIGNvbnN0IHF1ZXVlID0gZ2V0KGludGVybmFsVXNlckV2ZW50UXVldWVBdG9tKTtcbiAgcmV0dXJuIHF1ZXVlID09PSB1bmRlZmluZWQgPyBbXSA6IHF1ZXVlLmZpbHRlcigodikgPT4gdiAhPT0gdW5kZWZpbmVkKTtcbn0pO1xuXG4vKipcbiAqIFRoaXMgYXRvbSBpcyB1c2VkIHRvIHF1ZXVlIGEgbmV3IG1lc3NhZ2UgZm9yIHByb2Nlc3Npbmcgb24gYG5leHRfcmVzcG9uc2VzYFxuICogSXQgcmVjZWl2ZXMgYSBzaW5nbGUgYHVzZXJFdmVudGAgdGhhdCBpcyBhZGRlZCB0byB0aGUgcHJvY2Vzc2luZyBxdWV1ZS5cbiAqIElmIHRoZSBldmVudCBoYXMgdGhlIHNhbWUgZXZlbnRJZCBhcyBhbiBleGlzdGluZyBtZXNzYWdlIGluIHRoZSBxdWV1ZSB0aGUgTkVXXG4gKiBldmVudCBpcyBpZ25vcmVkXG4gKi9cbmV4cG9ydCBjb25zdCBxdWV1ZVVzZXJFdmVudEF0b20gPSBhdG9tKG51bGwsIChnZXQsIHNldCwgdXNlckV2ZW50OiBVc2VyRXZlbnQpID0+IHtcbiAgaWYgKHVzZXJFdmVudCA9PT0gdW5kZWZpbmVkKSB7XG4gICAgcmV0dXJuO1xuICB9XG4gIHNldChpbnRlcm5hbFVzZXJFdmVudFF1ZXVlQXRvbSwgWy4uLmdldChpbnRlcm5hbFVzZXJFdmVudFF1ZXVlQXRvbSksIHVzZXJFdmVudF0pO1xufSk7XG5cbi8qKlxuICogVGhpcyBhdG9tIGV4cG9zZXMgYSBmdW5jdGlvbiB0byByZXNldCB0aGUgZW50aXJlIHF1ZXVlLiBBbGwgbWVzc2FnZXMgaW4gdGhlIHF1ZXVlIHdpbGwgYmUgcHVyZ2VkXG4gKi9cbmV4cG9ydCBjb25zdCBjbGVhclVzZXJFdmVudEF0b20gPSBhdG9tKG51bGwsIChfLCBzZXQpID0+IHtcbiAgc2V0KGludGVybmFsVXNlckV2ZW50UXVldWVBdG9tLCBbXSk7XG59KTtcblxuLyoqXG4gKiBUaGlzIGF0b20gaXMgdXNlZCB0byBtYXJrIGV2ZW50cyBhcyBwcm9jZXNzZWQgYW5kIHJlbW92ZSB0aGVtIGZyb20gdGhlIHF1ZXVlXG4gKiBJdCBhY2NlcHRzIGEgbGlzdCBvZiBldmVudElkIHZhbHVlcyBhbmQgd2lsbCByZW1vdmUgYWxsIGV2ZW50cyB3aXRoIHRob3NlIGV2ZW50SWRzIGZyb20gdGhlIHF1ZXVlLlxuICovXG5leHBvcnQgY29uc3QgcHJvY2Vzc1VzZXJFdmVudEF0b20gPSBhdG9tKG51bGwsIChnZXQsIHNldCwgZXZlbnRJZHM6IHN0cmluZ1tdKSA9PiB7XG4gIGNvbnN0IGN1cnIgPSBnZXQoaW50ZXJuYWxVc2VyRXZlbnRRdWV1ZUF0b20pO1xuICBjb25zdCByZW1haW5pbmcgPSBjdXJyPy5maWx0ZXIoKGV2ZW50KSA9PiAhZXZlbnRJZHMuaW5jbHVkZXMoZXZlbnQuZXZlbnRJZCkpO1xuICBzZXQoaW50ZXJuYWxVc2VyRXZlbnRRdWV1ZUF0b20sIHJlbWFpbmluZyk7XG59KTtcblxuZXhwb3J0IGNvbnN0IHVzZXJRdWV1ZUV2ZW50Q291bnRBdG9tID0gYXRvbSgoZ2V0KSA9PiBnZXQodXNlckV2ZW50UXVldWVBdG9tKS5sZW5ndGgpO1xuXG5pbXBvcnQgeyBDb250ZXh0U291cmNlRW51bSwgQ29udGV4dEVudkVudW0gfSBmcm9tICdAc3BpZmZ5LWFpL2NvbW1lcmNlLWFwaS1jbGllbnQnOyAvLyBJbXBvcnQgbmVjZXNzYXJ5IGVudW1zXG4vLyBpbXBvcnQgdHlwZSB7IENvbnRleHQgfSBmcm9tIFwiQHNwaWZmeS1haS9jb21tZXJjZS1hcGktY2xpZW50L2Rpc3QvbW9kZWxzL0NvbnRleHRcIjsgLy8gSW1wb3J0IENvbnRleHQgdHlwZVxuXG5leHBvcnQgY29uc3QgY3JlYXRlUmVzcG9uc2VQYXlsb2FkID0gKHtcbiAgdXNlckV2ZW50cyxcbiAgZ2VuZXJhdGlvblBhcmFtcyxcbn06IHtcbiAgdXNlckV2ZW50czogVXNlckV2ZW50W107XG4gIGdlbmVyYXRpb25QYXJhbXM/OiBHZW5lcmF0aW9uUGFyYW1zO1xufSk6IE5leHRNZXNzYWdlUmVxdWVzdCA9PiB7XG4gIGNvbnN0IGF0b21TdG9yZSA9IGdldEF0b21TdG9yZSgpO1xuICBjb25zdCBvcmdTaG9ydE5hbWUgPSBhdG9tU3RvcmUuZ2V0KG9yZ1Nob3J0TmFtZUF0b20pO1xuICBjb25zdCBvcmdJZCA9IGF0b21TdG9yZS5nZXQob3JnSWRBdG9tKTtcbiAgY29uc3QgdXNlcklkID0gYXRvbVN0b3JlLmdldCh1c2VySWRBdG9tKTtcbiAgY29uc3QgY2hhdElkID0gYXRvbVN0b3JlLmdldChjaGF0SWRBdG9tKTtcbiAgY29uc3Qgc291cmNlID0gYXRvbVN0b3JlLmdldChjb250ZXh0U291cmNlQXRvbSk7XG4gIGNvbnN0IGVudiA9IGF0b21TdG9yZS5nZXQoZW52QXRvbSk7XG5cbiAgY29uc3QgZmVhdHVyZUZsYWdTZXJ2aWNlID0gYXRvbVN0b3JlLmdldChmZWF0dXJlRmxhZ1NlcnZpY2VBdG9tKTtcblxuICBjb25zdCBjb250ZXh0ID0ge1xuICAgIHVzZXJJZDogdXNlcklkID8/ICcnLFxuICAgIG9yZ19pZDogb3JnSWQgPz8gJycsXG4gICAgb3JnX3Nob3J0X25hbWU6IG9yZ1Nob3J0TmFtZSA/PyAnJyxcbiAgICBjaGF0X2lkOiBjaGF0SWQgPz8gJycsXG4gICAgc291cmNlOiBzb3VyY2UgPz8gQ29udGV4dFNvdXJjZUVudW0uQXBwLFxuICAgIGVudjogKGVudiBhcyBDb250ZXh0RW52RW51bSkgPz8gQ29udGV4dEVudkVudW0uRGV2LFxuICB9O1xuXG4gIGNvbnN0IGZlYXR1cmVGbGFncyA9IGZlYXR1cmVGbGFnU2VydmljZT8uZmVhdHVyZUZsYWdTZXJ2aWNlPy5nZXRGZWF0dXJlRmxhZ3MoKSB8fCB7fTtcblxuICByZXR1cm4ge1xuICAgIGlkOiB1dWlkKCksXG4gICAgY29udGV4dCxcbiAgICB1c2VyRXZlbnRzLFxuICAgIGZlYXR1cmVGbGFnczogT2JqZWN0LmZyb21FbnRyaWVzKFxuICAgICAgT2JqZWN0LmVudHJpZXMoZmVhdHVyZUZsYWdzKS5maWx0ZXIoKFssIGlzRW5hYmxlZF0pID0+IGlzRW5hYmxlZCksXG4gICAgKSwgLy8gQ29udmVydCBiYWNrIHRvIFJlY29yZDxzdHJpbmcsIGJvb2xlYW4+XG4gICAgZ2VuZXJhdGlvblBhcmFtcyxcbiAgfTtcbn07XG4iLCJpbXBvcnQgeyBhdG9tIH0gZnJvbSBcImpvdGFpXCI7XG5pbXBvcnQgeyBTcGlmZnlNZXRyaWNzRXZlbnROYW1lIH0gZnJvbSBcInNyYy9jb250ZXh0cy9hbXBsaXR1ZGVDb250ZXh0L2FtcGxpdHVkZUNvbnRleHRcIjtcblxuaW50ZXJmYWNlIFRyYWNrRXZlbnRQYXJhbXMge1xuICBldmVudE5hbWU6IFNwaWZmeU1ldHJpY3NFdmVudE5hbWU7XG4gIGV2ZW50UHJvcHM/OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgZXZlbnRHcm91cHM/OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgYWxzb1NlbmRUb0dvb2dsZUFuYWx5dGljcz86IGJvb2xlYW47XG59XG5cbi8vIFRoaXMgYXRvbSB3aWxsIGhvbGQgdGhlIHRyYWNrRXZlbnQgZnVuY3Rpb24gZnJvbSB0aGUgQW1wbGl0dWRlIGNvbnRleHQuXG4vLyBJdCBuZWVkcyB0byBiZSBpbml0aWFsaXplZCBieSBhIFJlYWN0IGNvbXBvbmVudCB0aGF0IGhhcyBhY2Nlc3MgdG8gdGhlIHVzZUFtcGxpdHVkZSBob29rLlxuZXhwb3J0IGNvbnN0IGFtcGxpdHVkZVRyYWNrRXZlbnRBdG9tID0gYXRvbTxcbiAgKChwYXJhbXM6IFRyYWNrRXZlbnRQYXJhbXMpID0+IFByb21pc2U8dm9pZD4pIHwgbnVsbFxuPihudWxsKTtcbiIsImltcG9ydCB7IGF0b20gfSBmcm9tICdqb3RhaSc7XG5pbXBvcnQgeyBNZXNzYWdlLCBNZXNzYWdlVHlwZSB9IGZyb20gJ0BlbnZpdmUtYWkvdHlwZXMnO1xuaW1wb3J0IHtcbiAgdXNlckhhc1JlcGxpZWRBdG9tLFxuICBtZXNzYWdlc0F0b20sXG4gIHVzZXJRdWVyeUF0b20sXG4gIHJlcGx5RXZlbnRDYXRlZ29yeUF0b20sXG59IGZyb20gJ3NyYy9hdG9tcy9jaGF0JztcbmltcG9ydCB7IFVzZXJFdmVudENhdGVnb3J5IH0gZnJvbSAnQHNwaWZmeS1haS9jb21tZXJjZS1hcGktY2xpZW50JztcbmltcG9ydCB7IHF1ZXVlVXNlckV2ZW50QXRvbSB9IGZyb20gJy4vbWVzc2FnZVF1ZXVlJztcbmltcG9ydCB7IFNwaWZmeU1ldHJpY3NFdmVudE5hbWUgfSBmcm9tICdzcmMvY29udGV4dHMvYW1wbGl0dWRlQ29udGV4dC9hbXBsaXR1ZGVDb250ZXh0JztcbmltcG9ydCB7IGFtcGxpdHVkZVRyYWNrRXZlbnRBdG9tIH0gZnJvbSAnc3JjL2F0b21zL2FtcGxpdHVkZS9hbXBsaXR1ZGVUcmFja0V2ZW50QXRvbSc7XG5cbnR5cGUgSGFuZGxlUmVwbHlQYXJhbXMgPSB7XG4gIG1lc3NhZ2U6IE1lc3NhZ2U7XG4gIHVzZXJUeXBlZDogYm9vbGVhbjtcbn07XG5cbmV4cG9ydCBjb25zdCBoYW5kbGVSZXBseUF0b20gPSBhdG9tKG51bGwsIChnZXQsIHNldCwgeyBtZXNzYWdlLCB1c2VyVHlwZWQgfTogSGFuZGxlUmVwbHlQYXJhbXMpID0+IHtcbiAgaWYgKG1lc3NhZ2UudHlwZSAhPT0gTWVzc2FnZVR5cGUuUXVlcnlUeXBlZCkge1xuICAgIHJldHVybjtcbiAgfVxuXG4gIGNvbnN0IHRyYWNrRXZlbnQgPSBnZXQoYW1wbGl0dWRlVHJhY2tFdmVudEF0b20pO1xuICBjb25zdCBxdWVyeVR5cGVkID0gbWVzc2FnZS5tZXRhZGF0YS5jb250ZW50O1xuXG4gIHNldChyZXBseUV2ZW50Q2F0ZWdvcnlBdG9tLCBVc2VyRXZlbnRDYXRlZ29yeS5RdWVyeVR5cGVkKTtcbiAgc2V0KHVzZXJRdWVyeUF0b20sIHF1ZXJ5VHlwZWQpO1xuICBzZXQobWVzc2FnZXNBdG9tLCBbLi4uZ2V0KG1lc3NhZ2VzQXRvbSksIFttZXNzYWdlXV0pO1xuICBzZXQodXNlckhhc1JlcGxpZWRBdG9tLCB0cnVlKTtcbiAgc2V0KHF1ZXVlVXNlckV2ZW50QXRvbSwge1xuICAgIGV2ZW50SWQ6IG1lc3NhZ2UuaWQsXG4gICAgY3JlYXRlZEF0OiBtZXNzYWdlLmNyZWF0ZWRBdCxcbiAgICBjYXRlZ29yeTogVXNlckV2ZW50Q2F0ZWdvcnkuUXVlcnlUeXBlZCxcbiAgICBhdHRyaWJ1dGVzOiB7XG4gICAgICBxdWVyeTogcXVlcnlUeXBlZCxcbiAgICB9LFxuICB9KTtcblxuICBpZiAodHJhY2tFdmVudCkge1xuICAgIHRyYWNrRXZlbnQoe1xuICAgICAgZXZlbnROYW1lOiBTcGlmZnlNZXRyaWNzRXZlbnROYW1lLkNoYXRVc2VyTWVzc2FnZUlucHV0LFxuICAgICAgZXZlbnRQcm9wczoge1xuICAgICAgICBtZXNzYWdlX2lkOiBtZXNzYWdlLmlkLFxuICAgICAgICBtZXNzYWdlX3JvbGU6IG1lc3NhZ2Uucm9sZSxcbiAgICAgICAgbWVzc2FnZV90eXBlOiBtZXNzYWdlLnR5cGUsXG4gICAgICAgIG1lc3NhZ2VfbWV0YWRhdGE6IHtcbiAgICAgICAgICBjb250ZW50OiBtZXNzYWdlPy5tZXRhZGF0YT8uY29udGVudCwgLy8gUmVtb3ZlZCBhbXBsaXR1ZGVTYWZlU3RyaW5nXG4gICAgICAgICAgY3JlYXRlZF9hdDogbWVzc2FnZS5jcmVhdGVkQXQsXG4gICAgICAgICAgdXNlcl90eXBlZDogdXNlclR5cGVkLFxuICAgICAgICB9LFxuICAgICAgfSxcbiAgICAgIGFsc29TZW5kVG9Hb29nbGVBbmFseXRpY3M6IHRydWUsXG4gICAgfSk7XG4gIH1cbn0pO1xuIiwiaW1wb3J0IHsgYXRvbSB9IGZyb20gXCJqb3RhaVwiO1xuaW1wb3J0IHsgYXRvbVdpdGhTdG9yYWdlIH0gZnJvbSBcImpvdGFpL3V0aWxzXCI7XG5pbXBvcnQgTG9nZ2VyIGZyb20gXCJzcmMvYXBwbGljYXRpb24vbG9nZ2luZy9sb2dnZXJcIjtcbmltcG9ydCB7XG4gIGdldEF0b21TdG9yZSxcbiAgc2Vzc2lvblN0b3JhZ2VVdGlsLFxufSBmcm9tIFwic3JjL2F0b21zL2F0b21TdG9yZS9hdG9tU3RvcmVcIjtcblxuZXhwb3J0IGNvbnN0IEFQUF9JTklUSUFMX1NUQVJUX1RJTUVfS0VZID0gXCJzcGlmZnktYXBwLWluaXRpYWwtc3RhcnQtdGltZVwiO1xuZXhwb3J0IGNvbnN0IFBBR0VfTE9BRF9PRkZTRVRfVElNRV9LRVkgPSBcInNwaWZmeS1wYWdlLWxvYWQtb2Zmc2V0LXRpbWVcIjtcbi8qKlxuICogVGhlIGRpZmZlcmVudCBwZXJmb3JtYW5jZSBtZXRyaWNzIHRoYXQgY2FuIGJlIGxvZ2dlZC4gQWxsIHRpbWVzIGFyZSByZWxhdGl2ZSB0byB0aGVcbiAqIGluaXRpYWwgc3RhcnQgdGltZSBvZiB0aGUgYXBwIGFuZCBhcmUgc3RvcmVkIGluIG1pbGxpc2Vjb25kcy5cbiAqL1xuZXhwb3J0IGVudW0gUGVyZk1ldHJpY3NFdmVudHMge1xuICBQYWdlTG9hZE9mZnNldCA9IFwicGFnZV9sb2FkX29mZnNldF9tc1wiLFxuICBNYWluQnVuZGxlTG9hZGVkID0gXCJtYWluX2J1bmRsZV9sb2FkZWRfbXNcIixcbiAgT3JnQ29uZmlnTG9hZFN0YXJ0ZWQgPSBcIm9yZ19jb25maWdfbG9hZF9zdGFydGVkX21zXCIsXG4gIE9yZ0NvbmZpZ0xvYWRFbmRlZCA9IFwib3JnX2NvbmZpZ19sb2FkX2VuZGVkX21zXCIsXG4gIEZpcnN0UmVzcG9uc2VTdGFydGVkID0gXCJmaXJzdF9yZXNwb25zZV9zdGFydGVkX21zXCIsXG4gIEZpcnN0UmVzcG9uc2VDb21wbGV0ZWQgPSBcImZpcnN0X3Jlc3BvbnNlX2NvbXBsZXRlZF9tc1wiLFxuICBGaXJzdFN1Z2dlc3Rpb25zU3RhcnRlZCA9IFwiZmlyc3Rfc3VnZ2VzdGlvbnNfc3RhcnRlZF9tc1wiLFxuICBGaXJzdFN1Z2dlc3Rpb25zQ29tcGxldGVkID0gXCJmaXJzdF9zdWdnZXN0aW9uc19jb21wbGV0ZWRfbXNcIixcbiAgRW1iZWRkZWRXaWRnZXRSZW5kZXJlZCA9IFwiZW1iZWRkZWRfd2lkZ2V0X3JlbmRlcmVkX21zXCIsXG4gIEZsb2F0aW5nQnV0dG9uUmVuZGVyZWQgPSBcImZsb2F0aW5nX2J1dHRvbl9yZW5kZXJlZF9tc1wiLFxuICBUb3BTdWdnZXN0aW9uc0JhclJlbmRlcmVkID0gXCJ0b3Bfc3VnZ2VzdGlvbnNfYmFyX3JlbmRlcmVkX21zXCIsXG4gIEJvdHRvbVN1Z2dlc3Rpb25zQmFyUmVuZGVyZWQgPSBcImJvdHRvbV9zdWdnZXN0aW9uc19iYXJfcmVuZGVyZWRfbXNcIixcbiAgU2VhcmNoUHJvbXB0UmVuZGVyZWQgPSBcInNlYXJjaF9wcm9tcHRfcmVuZGVyZWRfbXNcIixcbn1cblxuY29uc3QgaW50ZXJuYWxQZXJmTWV0cmljc0F0b20gPSBhdG9tPE1hcDxQZXJmTWV0cmljc0V2ZW50cywgbnVtYmVyPj4obmV3IE1hcCgpKTtcbmV4cG9ydCBjb25zdCBhcHBJbml0aWFsU3RhcnRUaW1lTXNBdG9tID0gYXRvbVdpdGhTdG9yYWdlPHN0cmluZyB8IHVuZGVmaW5lZD4oXG4gIEFQUF9JTklUSUFMX1NUQVJUX1RJTUVfS0VZLFxuICBzZXNzaW9uU3RvcmFnZS5nZXRJdGVtKEFQUF9JTklUSUFMX1NUQVJUX1RJTUVfS0VZKSA/PyB1bmRlZmluZWQsXG4gIHNlc3Npb25TdG9yYWdlVXRpbCxcbiAge1xuICAgIGdldE9uSW5pdDogdHJ1ZSxcbiAgfVxuKTtcblxuZXhwb3J0IGNvbnN0IHBhZ2VMb2FkT2Zmc2V0VGltZUF0b20gPSBhdG9tV2l0aFN0b3JhZ2U8c3RyaW5nIHwgdW5kZWZpbmVkPihcbiAgUEFHRV9MT0FEX09GRlNFVF9USU1FX0tFWSxcbiAgc2Vzc2lvblN0b3JhZ2UuZ2V0SXRlbShQQUdFX0xPQURfT0ZGU0VUX1RJTUVfS0VZKSA/PyB1bmRlZmluZWQsXG4gIHNlc3Npb25TdG9yYWdlVXRpbCxcbiAge1xuICAgIGdldE9uSW5pdDogdHJ1ZSxcbiAgfVxuKTtcblxuLyoqXG4gKiBSZXNldHMgdGhlIHBlcmZvcm1hbmNlIG1ldHJpY3MgYXRvbSB0byBhbiBlbXB0eSBtYXAuIFRoaXMgc2hvdWxkIGJlIGNhbGxlZCBhZnRlciB0aGUgcGVyZm9ybWFuY2VcbiAqIG1ldHJpY3MgaGF2ZSBiZWVuIHJlcG9ydGVkIHRvIGFtcGxpdHVkZS4gT24gU1BBLCBpdCBlbnN1cmVzIHRoYXQgd2UgY2FuIHN0aWxsIGNhcHR1cmUgbWV0cmljcyBhc1xuICogdGhlIHVzZXIgbmF2aWdhdGVzIGFyb3VuZCB0aGUgYXBwIChhbHRob3VnaCB0aGVyZSBpcyBtb3JlIHdvcmsgdG8gYmUgZG9uZSB0byBmdWxseSBlbmFibGUgdGhpcykuXG4gKiBPbiBub24tU1BBLCBpdCBlbnN1cmVzIHRoYXQgcHJldmlvdXNseSBjYXB0dXJlZCBtZXRyaWNzIGFyZSBub3QgcmVwb3J0ZWQgYWdhaW4uXG4gKi9cbmV4cG9ydCBjb25zdCByZXNldFBlcmZvcm1hbmNlTWV0cmljc0F0b20gPSAoKSA9PiB7XG4gIGNvbnN0IGF0b21TdG9yZSA9IGdldEF0b21TdG9yZSgpO1xuICBhdG9tU3RvcmUuc2V0KGludGVybmFsUGVyZk1ldHJpY3NBdG9tLCBuZXcgTWFwKCkpO1xufTtcbmV4cG9ydCBjb25zdCBwZXJmb3JtYW5jZU1ldHJpY3NBdG9tID0gYXRvbSgoZ2V0KSA9PlxuICBnZXQoaW50ZXJuYWxQZXJmTWV0cmljc0F0b20pXG4pO1xuZXhwb3J0IGNvbnN0IGhhc1JlcG9ydGVkUGVyZm9ybWFuY2VNZXRyaWNzQXRvbSA9IGF0b208Ym9vbGVhbj4oZmFsc2UpO1xuXG4vKipcbiAqIExvZ3MgYSBwZXJmb3JtYW5jZSBtZXRyaWMgYnkgY2FwdHVyaW5nIHRoZSBkZWx0YSBiZXR3ZWVuIHRoZSBpbml0aWFsIGFwcCBzdGFydCB0aW1lXG4gKiBhbmQgdGhlIGN1cnJlbnQgdGltZS4gSWYgdGhlIG1ldHJpYyBoYXMgYWxyZWFkeSBiZWVuIGxvZ2dlZCwgaXQgd2lsbCBub3QgYmUgbG9nZ2VkIGFnYWluLlxuICpcbiAqIEBwYXJhbSB2YWx1ZSBUaGUgcGVyZm9ybWFuY2UgbWV0cmljIG5hbWUgdG8gbG9nLlxuICovXG5leHBvcnQgY29uc3QgbG9nUGVyZk1ldHJpY0F0b20gPSBhdG9tKFxuICBudWxsLFxuICAoZ2V0LCBzZXQsIHZhbHVlOiBQZXJmTWV0cmljc0V2ZW50cykgPT4ge1xuICAgIGNvbnN0IGluaXRpYWxUaW1lU3RvcmFnZVZhbHVlID0gZ2V0KGFwcEluaXRpYWxTdGFydFRpbWVNc0F0b20pO1xuICAgIGNvbnN0IGluaXRpYWxUaW1lTXMgPSBpbml0aWFsVGltZVN0b3JhZ2VWYWx1ZVxuICAgICAgPyBwYXJzZUludChpbml0aWFsVGltZVN0b3JhZ2VWYWx1ZSwgMTApXG4gICAgICA6IHVuZGVmaW5lZDtcblxuICAgIGlmIChpbml0aWFsVGltZU1zID09IG51bGwpIHtcbiAgICAgIExvZ2dlci5sb2dXYXJuKFxuICAgICAgICBgW3NwaWZmeS1haV0gTm8gaW5pdGlhbCBhcHAgc3RhcnQgdGltZSBmb3VuZC4gU2tpcHBpbmcuLi5gLFxuICAgICAgICB1bmRlZmluZWRcbiAgICAgICk7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgY29uc3QgY3VycmVudFBlcmZNZXRyaWNzID0gZ2V0KGludGVybmFsUGVyZk1ldHJpY3NBdG9tKTtcbiAgICBpZiAoY3VycmVudFBlcmZNZXRyaWNzLmhhcyh2YWx1ZSkpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBjb25zdCBjdXJyZW50VGltZU1zID0gRGF0ZS5ub3coKTtcbiAgICBjb25zdCBkZWx0YU1zID0gY3VycmVudFRpbWVNcyAtIGluaXRpYWxUaW1lTXM7XG4gICAgY3VycmVudFBlcmZNZXRyaWNzLnNldCh2YWx1ZSwgZGVsdGFNcyk7XG4gICAgc2V0KGludGVybmFsUGVyZk1ldHJpY3NBdG9tLCBjdXJyZW50UGVyZk1ldHJpY3MpO1xuICB9XG4pO1xuIiwiaW1wb3J0IHsgVXNlckV2ZW50Q2F0ZWdvcnkgfSBmcm9tICdAc3BpZmZ5LWFpL2NvbW1lcmNlLWFwaS1jbGllbnQnO1xuaW1wb3J0IHsgYXRvbSB9IGZyb20gJ2pvdGFpJztcbmltcG9ydCB7IEZvcm1TdWJtaXR0ZWRBdHRyaWJ1dGVzLCBVc2VyRXZlbnQgfSBmcm9tICdAZW52aXZlLWFpL3R5cGVzJztcbmltcG9ydCB7IHY0IGFzIHV1aWQgfSBmcm9tICd1dWlkJztcbmltcG9ydCB7IGZvcm1TdWJtaXRBdG9tLCByZXBseUV2ZW50Q2F0ZWdvcnlBdG9tIH0gZnJvbSAnc3JjL2F0b21zL2NoYXQnO1xuaW1wb3J0IHsgcXVldWVVc2VyRXZlbnRBdG9tIH0gZnJvbSAnc3JjL2F0b21zL2NoYXQvbWVzc2FnZVF1ZXVlJztcblxuZXhwb3J0IGNvbnN0IGhhbmRsZUZvcm1TdWJtaXR0ZWRBdG9tID0gYXRvbShudWxsLCAoXywgc2V0LCBmb3JtOiBGb3JtU3VibWl0dGVkQXR0cmlidXRlcykgPT4ge1xuICBzZXQocmVwbHlFdmVudENhdGVnb3J5QXRvbSwgVXNlckV2ZW50Q2F0ZWdvcnkuRm9ybVN1Ym1pdHRlZCk7XG4gIHNldChmb3JtU3VibWl0QXRvbSwgZm9ybSk7XG5cbiAgY29uc3QgZm9ybVVzZXJFdmVudCA9IHtcbiAgICBldmVudElkOiB1dWlkKCksXG4gICAgY2F0ZWdvcnk6IFVzZXJFdmVudENhdGVnb3J5LkZvcm1TdWJtaXR0ZWQsXG4gICAgY3JlYXRlZEF0OiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXG4gICAgYXR0cmlidXRlczogZm9ybSxcbiAgfSBzYXRpc2ZpZXMgVXNlckV2ZW50O1xuICBzZXQocXVldWVVc2VyRXZlbnRBdG9tLCBmb3JtVXNlckV2ZW50KTtcbn0pO1xuIiwiaW1wb3J0IHsgYXRvbSB9IGZyb20gXCJqb3RhaVwiO1xuaW1wb3J0IHsgTWVzc2FnZSwgTWVzc2FnZVJvbGUsIE1lc3NhZ2VUeXBlLCBTdWdnZXN0aW9uIH0gZnJvbSBcInNyYy90eXBlc1wiO1xuaW1wb3J0IHtcbiAgbWVzc2FnZXNBdG9tLFxuICByZXBseUV2ZW50Q2F0ZWdvcnlBdG9tLFxuICBzdWdnZXN0aW9uQXRvbSxcbiAgdXNlckhhc1JlcGxpZWRBdG9tLFxufSBmcm9tIFwic3JjL2F0b21zL2NoYXRcIjtcbmltcG9ydCB7IEZpbHRlckF0dHJpYnV0ZSwgUHJvZHVjdFJlc3BvbnNlQXR0cmlidXRlcyB9IGZyb20gXCJzcmMvdHlwZXNcIjtcblxuaW1wb3J0IHsgVXNlckV2ZW50Q2F0ZWdvcnkgfSBmcm9tIFwiQHNwaWZmeS1haS9jb21tZXJjZS1hcGktY2xpZW50XCI7XG5pbXBvcnQgeyBxdWV1ZVVzZXJFdmVudEF0b20gfSBmcm9tIFwiLi9tZXNzYWdlUXVldWVcIjtcblxuZXhwb3J0IGNvbnN0IGhhbmRsZVN1Z2dlc3Rpb25BdG9tID0gYXRvbShcbiAgbnVsbCxcbiAgKGdldCwgc2V0LCBzdWdnZXN0aW9uOiBTdWdnZXN0aW9uKSA9PiB7XG4gICAgY29uc3QgbmV3TWVzc2FnZTogTWVzc2FnZSA9IHtcbiAgICAgIGlkOiBzdWdnZXN0aW9uLmlkLFxuICAgICAgcm9sZTogTWVzc2FnZVJvbGUuVXNlcixcbiAgICAgIHR5cGU6IE1lc3NhZ2VUeXBlLlN1Z2dlc3Rpb25DbGlja2VkLFxuICAgICAgY3JlYXRlZEF0OiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXG4gICAgICBtZXRhZGF0YToge1xuICAgICAgICBzdWdnZXN0aW9uSWQ6IHN1Z2dlc3Rpb24uaWQsXG4gICAgICAgIHN1Z2dlc3Rpb25Db250ZW50OiBzdWdnZXN0aW9uLmNvbnRlbnQsXG4gICAgICB9LFxuICAgIH07XG5cbiAgICBzZXQocmVwbHlFdmVudENhdGVnb3J5QXRvbSwgVXNlckV2ZW50Q2F0ZWdvcnkuU3VnZ2VzdGlvbkNsaWNrZWQpO1xuICAgIHNldChzdWdnZXN0aW9uQXRvbSwgc3VnZ2VzdGlvbik7XG4gICAgc2V0KG1lc3NhZ2VzQXRvbSwgWy4uLmdldChtZXNzYWdlc0F0b20pLCBbbmV3TWVzc2FnZV1dKTtcbiAgICBzZXQodXNlckhhc1JlcGxpZWRBdG9tLCB0cnVlKTtcbiAgICBzZXQocXVldWVVc2VyRXZlbnRBdG9tLCB7XG4gICAgICBldmVudElkOiBzdWdnZXN0aW9uLmlkLFxuICAgICAgY2F0ZWdvcnk6IFVzZXJFdmVudENhdGVnb3J5LlN1Z2dlc3Rpb25DbGlja2VkLFxuICAgICAgY3JlYXRlZEF0OiBuZXdNZXNzYWdlLmNyZWF0ZWRBdCxcbiAgICAgIGF0dHJpYnV0ZXM6IHtcbiAgICAgICAgc3VnZ2VzdGlvbklkOiBzdWdnZXN0aW9uLmlkLFxuICAgICAgfSxcbiAgICB9KTtcbiAgfVxuKTtcbiIsImltcG9ydCB7IGF0b20gfSBmcm9tICdqb3RhaSc7XG5pbXBvcnQgeyBtZXNzYWdlc0F0b20sIHVzZXJIYXNSZXBsaWVkQXRvbSB9IGZyb20gJ3NyYy9hdG9tcy9jaGF0JztcblxuZXhwb3J0IGNvbnN0IGxhc3RBc3Npc3RhbnRNZXNzYWdlQXRvbSA9IGF0b20oKGdldCkgPT4ge1xuICBjb25zdCBtZXNzYWdlcyA9IGdldChtZXNzYWdlc0F0b20pO1xuICBjb25zdCB1c2VySGFzUmVwbGllZCA9IGdldCh1c2VySGFzUmVwbGllZEF0b20pO1xuICBpZiAobWVzc2FnZXMubGVuZ3RoID4gMCAmJiAhdXNlckhhc1JlcGxpZWQpIHtcbiAgICByZXR1cm4gbWVzc2FnZXNbbWVzc2FnZXMubGVuZ3RoIC0gMV07XG4gIH1cbiAgcmV0dXJuIG51bGw7XG59KTtcbiIsImltcG9ydCB7IGF0b20gfSBmcm9tIFwiam90YWlcIjtcbmltcG9ydCB7IGdldEF0b21TdG9yZSB9IGZyb20gXCJzcmMvYXRvbXMvYXRvbVN0b3JlL2F0b21TdG9yZVwiO1xuXG5jb25zdCBpbnRlcm5hbFdpZGdldEFycmF5QXRvbSA9IGF0b208SFRNTEVsZW1lbnRbXT4oW10pO1xuXG5leHBvcnQgY29uc3Qgd2lkZ2V0QXJyYXlBdG9tID0gYXRvbSgoZ2V0KSA9PiBnZXQoaW50ZXJuYWxXaWRnZXRBcnJheUF0b20pKTtcblxuLyoqXG4gKiBUaGlzIGZ1bmN0aW9uIGNhbGwgaXMgdXNlZCB0byBjcmVhdGUgYSBsaXN0IG9mIHRoZSBTcGlmZnkgd2lkZ2V0c1xuICogdGhhdCBhcmUgcmVuZGVyaW5nIG9uIHRoZSBwYWdlLlxuICpcbiAqIEl0IGlzIHVzZWQgYnkgdGhlIEZsb2F0aW5nQnV0dG9uIHdpZGdldCB0byBzbGlkZSBvdXQgb2YgdmlldyB3aGVuIGl0XG4gKiBvdmVybGFwcyB3aXRoIGFub3RoZXIgU3BpZmZ5IHdpZGdldC5cbiAqXG4gKiBUaGUgc3BpZmZ5IHdpZGdldHMgc2hvdWxkIGJlIGFkZGVkIHRvIHRoZSB0aGlzIGFycmF5IGZvciB0aGUgZmxvYXRpbmcgYnV0dG9uXG4gKiB0byBrbm93IGFib3V0IHRoZW0uXG4gKlxuICogVG9kYXkgdGhlIFwiU3VnZ2VzdGlvbkJhclwiIHdpZGdldCBpcyBub3QgYWRkZWQsIGJ1dCB0aGUgb3RoZXIgd2lkZ2V0cyBhcmUuXG4gKlxuICovXG5leHBvcnQgY29uc3QgYWRkV2lkZ2V0ID0gKHJlZjogSFRNTEVsZW1lbnQsIGlkeD86IG51bWJlcikgPT4ge1xuICBjb25zdCBhdG9tU3RvcmUgPSBnZXRBdG9tU3RvcmUoKTtcbiAgY29uc3QgdmFsID0gYXRvbVN0b3JlLmdldChpbnRlcm5hbFdpZGdldEFycmF5QXRvbSk7XG4gIGNvbnN0IGluc2VydElkeCA9IGlkeCA/PyB2YWwubGVuZ3RoO1xuICB2YWxbaW5zZXJ0SWR4XSA9IHJlZjtcbiAgYXRvbVN0b3JlLnNldChpbnRlcm5hbFdpZGdldEFycmF5QXRvbSwgdmFsKTtcbn07XG4iLCJpbXBvcnQgeyBhdG9tIH0gZnJvbSAnam90YWknO1xuaW1wb3J0IHtcbiAgcmVzcG9uc2VTdHJlYW1pbmdBdG9tLFxuICBzdWdnZXN0aW9uc0xvYWRpbmdBdG9tLFxuICBtZXNzYWdlc0F0b20sXG4gIHVzZXJRdWVyeUF0b20sXG4gIHJlcGx5RXZlbnRDYXRlZ29yeUF0b20sXG4gIHVzZXJIYXNSZXBsaWVkQXRvbSxcbiAgYXNrUXVlc3Rpb25CdG5DbGlja2VkQXRvbSxcbiAgc3VnZ2VzdGlvbnNBdG9tLFxuICBjaGF0SXNPcGVuQXRvbSxcbiAgY2hhdE9uVG9nZ2xlQXRvbSxcbiAgdXNlckV2ZW50c0F0b20sXG4gIHN1Z2dlc3Rpb25BdG9tLFxufSBmcm9tICcuL2NoYXRTdGF0ZSc7XG5cbmV4cG9ydCAqIGZyb20gJy4vY2hhdFN0YXRlJztcbmV4cG9ydCAqIGZyb20gJy4vcmVwbGllcyc7XG5leHBvcnQgKiBmcm9tICcuL3BlcmZvcm1hbmNlTWV0cmljcyc7XG5leHBvcnQgKiBmcm9tICcuL2Zvcm0nO1xuZXhwb3J0ICogZnJvbSAnLi9zdWdnZXN0aW9ucyc7XG5leHBvcnQgKiBmcm9tICcuL2xhc3RNZXNzYWdlJztcbmV4cG9ydCAqIGZyb20gJy4vcmVuZGVyZWRXaWRnZXRSZWZzJztcblxuZXhwb3J0IGNvbnN0IGNoYXRBdG9tID0gYXRvbSgoZ2V0KSA9PiAoe1xuICB1c2VySGFzUmVwbGllZDogZ2V0KHVzZXJIYXNSZXBsaWVkQXRvbSksXG4gIHJlcGx5RXZlbnRDYXRlZ29yeTogZ2V0KHJlcGx5RXZlbnRDYXRlZ29yeUF0b20pLFxuICB1c2VyUXVlcnk6IGdldCh1c2VyUXVlcnlBdG9tKSxcbiAgc3VnZ2VzdGlvbjogZ2V0KHN1Z2dlc3Rpb25BdG9tKSxcbiAgYXNrUXVlc3Rpb25CdG5DbGlja2VkOiBnZXQoYXNrUXVlc3Rpb25CdG5DbGlja2VkQXRvbSksXG4gIG1lc3NhZ2VzOiBnZXQobWVzc2FnZXNBdG9tKSxcbiAgdXNlckV2ZW50czogZ2V0KHVzZXJFdmVudHNBdG9tKSxcbiAgc3VnZ2VzdGlvbnM6IGdldChzdWdnZXN0aW9uc0F0b20pLFxuICBzdWdnZXN0aW9uc0xvYWRpbmc6IGdldChzdWdnZXN0aW9uc0xvYWRpbmdBdG9tKSxcbiAgcmVzcG9uc2VTdHJlYW1pbmc6IGdldChyZXNwb25zZVN0cmVhbWluZ0F0b20pLFxuICBpc09wZW46IGdldChjaGF0SXNPcGVuQXRvbSksXG4gIG9uVG9nZ2xlOiBnZXQoY2hhdE9uVG9nZ2xlQXRvbSksXG59KSk7XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7OztBQVFBLE1BQU0sNkJBQTZCLEtBQWtCLEVBQUUsQ0FBQztB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
@@ -1,330 +0,0 @@
1
- const require_chunk = require('./chunk-CUT6urMc.cjs');
2
- const require_dist = require('./dist-B7BErEyV.cjs');
3
- const require_types = require('./types-1iJ_FnQQ.cjs');
4
- const require_logger = require('./logger-BqHq67zN.cjs');
5
- const require_atomStore = require('./atomStore-CZKe3itM.cjs');
6
- const require_utils = require('./utils-DqNhRm2b.cjs');
7
- const require_chatState = require('./chatState-DqWSgmjC.cjs');
8
- let __spiffy_ai_commerce_api_client = require("@spiffy-ai/commerce-api-client");
9
- __spiffy_ai_commerce_api_client = require_chunk.__toESM(__spiffy_ai_commerce_api_client);
10
- let uuid = require("uuid");
11
- uuid = require_chunk.__toESM(uuid);
12
- let jotai = require("jotai");
13
- jotai = require_chunk.__toESM(jotai);
14
- let jotai_utils = require("jotai/utils");
15
- jotai_utils = require_chunk.__toESM(jotai_utils);
16
-
17
- //#region src/atoms/chat/messageQueue.ts
18
- const internalUserEventQueueAtom = (0, jotai.atom)([]);
19
- const userEventQueueAtom = (0, jotai.atom)((get) => {
20
- const queue = get(internalUserEventQueueAtom);
21
- return queue === void 0 ? [] : queue.filter((v) => v !== void 0);
22
- });
23
- /**
24
- * This atom is used to queue a new message for processing on `next_responses`
25
- * It receives a single `userEvent` that is added to the processing queue.
26
- * If the event has the same eventId as an existing message in the queue the NEW
27
- * event is ignored
28
- */
29
- const queueUserEventAtom = (0, jotai.atom)(null, (get, set, userEvent) => {
30
- if (userEvent === void 0) return;
31
- set(internalUserEventQueueAtom, [...get(internalUserEventQueueAtom), userEvent]);
32
- });
33
- /**
34
- * This atom exposes a function to reset the entire queue. All messages in the queue will be purged
35
- */
36
- const clearUserEventAtom = (0, jotai.atom)(null, (_, set) => {
37
- set(internalUserEventQueueAtom, []);
38
- });
39
- /**
40
- * This atom is used to mark events as processed and remove them from the queue
41
- * It accepts a list of eventId values and will remove all events with those eventIds from the queue.
42
- */
43
- const processUserEventAtom = (0, jotai.atom)(null, (get, set, eventIds) => {
44
- const remaining = get(internalUserEventQueueAtom)?.filter((event) => !eventIds.includes(event.eventId));
45
- set(internalUserEventQueueAtom, remaining);
46
- });
47
- const userQueueEventCountAtom = (0, jotai.atom)((get) => get(userEventQueueAtom).length);
48
-
49
- //#endregion
50
- //#region src/atoms/amplitude/amplitudeTrackEventAtom.ts
51
- const amplitudeTrackEventAtom = (0, jotai.atom)(null);
52
-
53
- //#endregion
54
- //#region src/atoms/chat/replies.ts
55
- const handleReplyAtom = (0, jotai.atom)(null, (get, set, { message, userTyped }) => {
56
- if (message.type !== require_dist.MessageType.QueryTyped) return;
57
- const trackEvent = get(amplitudeTrackEventAtom);
58
- const queryTyped = message.metadata.content;
59
- set(require_chatState.replyEventCategoryAtom, __spiffy_ai_commerce_api_client.UserEventCategory.QueryTyped);
60
- set(require_chatState.userQueryAtom, queryTyped);
61
- set(require_chatState.messagesAtom, [...get(require_chatState.messagesAtom), [message]]);
62
- set(require_chatState.userHasRepliedAtom, true);
63
- set(queueUserEventAtom, {
64
- eventId: message.id,
65
- createdAt: message.createdAt,
66
- category: __spiffy_ai_commerce_api_client.UserEventCategory.QueryTyped,
67
- attributes: { query: queryTyped }
68
- });
69
- if (trackEvent) trackEvent({
70
- eventName: require_utils.SpiffyMetricsEventName.ChatUserMessageInput,
71
- eventProps: {
72
- message_id: message.id,
73
- message_role: message.role,
74
- message_type: message.type,
75
- message_metadata: {
76
- content: message?.metadata?.content,
77
- created_at: message.createdAt,
78
- user_typed: userTyped
79
- }
80
- },
81
- alsoSendToGoogleAnalytics: true
82
- });
83
- });
84
-
85
- //#endregion
86
- //#region src/atoms/chat/performanceMetrics.ts
87
- const APP_INITIAL_START_TIME_KEY = "spiffy-app-initial-start-time";
88
- const PAGE_LOAD_OFFSET_TIME_KEY = "spiffy-page-load-offset-time";
89
- /**
90
- * The different performance metrics that can be logged. All times are relative to the
91
- * initial start time of the app and are stored in milliseconds.
92
- */
93
- let PerfMetricsEvents = /* @__PURE__ */ function(PerfMetricsEvents$1) {
94
- PerfMetricsEvents$1["PageLoadOffset"] = "page_load_offset_ms";
95
- PerfMetricsEvents$1["MainBundleLoaded"] = "main_bundle_loaded_ms";
96
- PerfMetricsEvents$1["OrgConfigLoadStarted"] = "org_config_load_started_ms";
97
- PerfMetricsEvents$1["OrgConfigLoadEnded"] = "org_config_load_ended_ms";
98
- PerfMetricsEvents$1["FirstResponseStarted"] = "first_response_started_ms";
99
- PerfMetricsEvents$1["FirstResponseCompleted"] = "first_response_completed_ms";
100
- PerfMetricsEvents$1["FirstSuggestionsStarted"] = "first_suggestions_started_ms";
101
- PerfMetricsEvents$1["FirstSuggestionsCompleted"] = "first_suggestions_completed_ms";
102
- PerfMetricsEvents$1["EmbeddedWidgetRendered"] = "embedded_widget_rendered_ms";
103
- PerfMetricsEvents$1["FloatingButtonRendered"] = "floating_button_rendered_ms";
104
- PerfMetricsEvents$1["TopSuggestionsBarRendered"] = "top_suggestions_bar_rendered_ms";
105
- PerfMetricsEvents$1["BottomSuggestionsBarRendered"] = "bottom_suggestions_bar_rendered_ms";
106
- PerfMetricsEvents$1["SearchPromptRendered"] = "search_prompt_rendered_ms";
107
- return PerfMetricsEvents$1;
108
- }({});
109
- const internalPerfMetricsAtom = (0, jotai.atom)(/* @__PURE__ */ new Map());
110
- const appInitialStartTimeMsAtom = (0, jotai_utils.atomWithStorage)(APP_INITIAL_START_TIME_KEY, sessionStorage.getItem(APP_INITIAL_START_TIME_KEY) ?? void 0, require_atomStore.sessionStorageUtil, { getOnInit: true });
111
- const pageLoadOffsetTimeAtom = (0, jotai_utils.atomWithStorage)(PAGE_LOAD_OFFSET_TIME_KEY, sessionStorage.getItem(PAGE_LOAD_OFFSET_TIME_KEY) ?? void 0, require_atomStore.sessionStorageUtil, { getOnInit: true });
112
- /**
113
- * Resets the performance metrics atom to an empty map. This should be called after the performance
114
- * metrics have been reported to amplitude. On SPA, it ensures that we can still capture metrics as
115
- * the user navigates around the app (although there is more work to be done to fully enable this).
116
- * On non-SPA, it ensures that previously captured metrics are not reported again.
117
- */
118
- const resetPerformanceMetricsAtom = () => {
119
- require_atomStore.getAtomStore().set(internalPerfMetricsAtom, /* @__PURE__ */ new Map());
120
- };
121
- const performanceMetricsAtom = (0, jotai.atom)((get) => get(internalPerfMetricsAtom));
122
- const hasReportedPerformanceMetricsAtom = (0, jotai.atom)(false);
123
- /**
124
- * Logs a performance metric by capturing the delta between the initial app start time
125
- * and the current time. If the metric has already been logged, it will not be logged again.
126
- *
127
- * @param value The performance metric name to log.
128
- */
129
- const logPerfMetricAtom = (0, jotai.atom)(null, (get, set, value) => {
130
- const initialTimeStorageValue = get(appInitialStartTimeMsAtom);
131
- const initialTimeMs = initialTimeStorageValue ? parseInt(initialTimeStorageValue, 10) : void 0;
132
- if (initialTimeMs == null) {
133
- require_logger.logger_default.logWarn(`[spiffy-ai] No initial app start time found. Skipping...`, void 0);
134
- return;
135
- }
136
- const currentPerfMetrics = get(internalPerfMetricsAtom);
137
- if (currentPerfMetrics.has(value)) return;
138
- const deltaMs = Date.now() - initialTimeMs;
139
- currentPerfMetrics.set(value, deltaMs);
140
- set(internalPerfMetricsAtom, currentPerfMetrics);
141
- });
142
-
143
- //#endregion
144
- //#region src/atoms/chat/form.ts
145
- const handleFormSubmittedAtom = (0, jotai.atom)(null, (_, set, form) => {
146
- set(require_chatState.replyEventCategoryAtom, __spiffy_ai_commerce_api_client.UserEventCategory.FormSubmitted);
147
- set(require_chatState.formSubmitAtom, form);
148
- const formUserEvent = {
149
- eventId: (0, uuid.v4)(),
150
- category: __spiffy_ai_commerce_api_client.UserEventCategory.FormSubmitted,
151
- createdAt: (/* @__PURE__ */ new Date()).toISOString(),
152
- attributes: form
153
- };
154
- set(queueUserEventAtom, formUserEvent);
155
- });
156
-
157
- //#endregion
158
- //#region src/atoms/chat/suggestions.ts
159
- const handleSuggestionAtom = (0, jotai.atom)(null, (get, set, suggestion) => {
160
- const newMessage = {
161
- id: suggestion.id,
162
- role: require_types.MessageRole.User,
163
- type: require_types.MessageType.SuggestionClicked,
164
- createdAt: (/* @__PURE__ */ new Date()).toISOString(),
165
- metadata: {
166
- suggestionId: suggestion.id,
167
- suggestionContent: suggestion.content
168
- }
169
- };
170
- set(require_chatState.replyEventCategoryAtom, __spiffy_ai_commerce_api_client.UserEventCategory.SuggestionClicked);
171
- set(require_chatState.suggestionAtom, suggestion);
172
- set(require_chatState.messagesAtom, [...get(require_chatState.messagesAtom), [newMessage]]);
173
- set(require_chatState.userHasRepliedAtom, true);
174
- set(queueUserEventAtom, {
175
- eventId: suggestion.id,
176
- category: __spiffy_ai_commerce_api_client.UserEventCategory.SuggestionClicked,
177
- createdAt: newMessage.createdAt,
178
- attributes: { suggestionId: suggestion.id }
179
- });
180
- });
181
-
182
- //#endregion
183
- //#region src/atoms/chat/lastMessage.ts
184
- const lastAssistantMessageAtom = (0, jotai.atom)((get) => {
185
- const messages = get(require_chatState.messagesAtom);
186
- const userHasReplied = get(require_chatState.userHasRepliedAtom);
187
- if (messages.length > 0 && !userHasReplied) return messages[messages.length - 1];
188
- return null;
189
- });
190
-
191
- //#endregion
192
- //#region src/atoms/chat/renderedWidgetRefs.ts
193
- const internalWidgetArrayAtom = (0, jotai.atom)([]);
194
- const widgetArrayAtom = (0, jotai.atom)((get) => get(internalWidgetArrayAtom));
195
- /**
196
- * This function call is used to create a list of the Spiffy widgets
197
- * that are rendering on the page.
198
- *
199
- * It is used by the FloatingButton widget to slide out of view when it
200
- * overlaps with another Spiffy widget.
201
- *
202
- * The spiffy widgets should be added to the this array for the floating button
203
- * to know about them.
204
- *
205
- * Today the "SuggestionBar" widget is not added, but the other widgets are.
206
- *
207
- */
208
- const addWidget = (ref, idx) => {
209
- const atomStore = require_atomStore.getAtomStore();
210
- const val = atomStore.get(internalWidgetArrayAtom);
211
- const insertIdx = idx ?? val.length;
212
- val[insertIdx] = ref;
213
- atomStore.set(internalWidgetArrayAtom, val);
214
- };
215
-
216
- //#endregion
217
- //#region src/atoms/chat/index.ts
218
- const chatAtom = (0, jotai.atom)((get) => ({
219
- userHasReplied: get(require_chatState.userHasRepliedAtom),
220
- replyEventCategory: get(require_chatState.replyEventCategoryAtom),
221
- userQuery: get(require_chatState.userQueryAtom),
222
- suggestion: get(require_chatState.suggestionAtom),
223
- askQuestionBtnClicked: get(require_chatState.askQuestionBtnClickedAtom),
224
- messages: get(require_chatState.messagesAtom),
225
- userEvents: get(require_chatState.userEventsAtom),
226
- suggestions: get(require_chatState.suggestionsAtom),
227
- suggestionsLoading: get(require_chatState.suggestionsLoadingAtom),
228
- responseStreaming: get(require_chatState.responseStreamingAtom),
229
- isOpen: get(require_chatState.chatIsOpenAtom),
230
- onToggle: get(require_chatState.chatOnToggleAtom)
231
- }));
232
-
233
- //#endregion
234
- Object.defineProperty(exports, 'APP_INITIAL_START_TIME_KEY', {
235
- enumerable: true,
236
- get: function () {
237
- return APP_INITIAL_START_TIME_KEY;
238
- }
239
- });
240
- Object.defineProperty(exports, 'PAGE_LOAD_OFFSET_TIME_KEY', {
241
- enumerable: true,
242
- get: function () {
243
- return PAGE_LOAD_OFFSET_TIME_KEY;
244
- }
245
- });
246
- Object.defineProperty(exports, 'PerfMetricsEvents', {
247
- enumerable: true,
248
- get: function () {
249
- return PerfMetricsEvents;
250
- }
251
- });
252
- Object.defineProperty(exports, 'addWidget', {
253
- enumerable: true,
254
- get: function () {
255
- return addWidget;
256
- }
257
- });
258
- Object.defineProperty(exports, 'appInitialStartTimeMsAtom', {
259
- enumerable: true,
260
- get: function () {
261
- return appInitialStartTimeMsAtom;
262
- }
263
- });
264
- Object.defineProperty(exports, 'chatAtom', {
265
- enumerable: true,
266
- get: function () {
267
- return chatAtom;
268
- }
269
- });
270
- Object.defineProperty(exports, 'handleFormSubmittedAtom', {
271
- enumerable: true,
272
- get: function () {
273
- return handleFormSubmittedAtom;
274
- }
275
- });
276
- Object.defineProperty(exports, 'handleReplyAtom', {
277
- enumerable: true,
278
- get: function () {
279
- return handleReplyAtom;
280
- }
281
- });
282
- Object.defineProperty(exports, 'handleSuggestionAtom', {
283
- enumerable: true,
284
- get: function () {
285
- return handleSuggestionAtom;
286
- }
287
- });
288
- Object.defineProperty(exports, 'hasReportedPerformanceMetricsAtom', {
289
- enumerable: true,
290
- get: function () {
291
- return hasReportedPerformanceMetricsAtom;
292
- }
293
- });
294
- Object.defineProperty(exports, 'lastAssistantMessageAtom', {
295
- enumerable: true,
296
- get: function () {
297
- return lastAssistantMessageAtom;
298
- }
299
- });
300
- Object.defineProperty(exports, 'logPerfMetricAtom', {
301
- enumerable: true,
302
- get: function () {
303
- return logPerfMetricAtom;
304
- }
305
- });
306
- Object.defineProperty(exports, 'pageLoadOffsetTimeAtom', {
307
- enumerable: true,
308
- get: function () {
309
- return pageLoadOffsetTimeAtom;
310
- }
311
- });
312
- Object.defineProperty(exports, 'performanceMetricsAtom', {
313
- enumerable: true,
314
- get: function () {
315
- return performanceMetricsAtom;
316
- }
317
- });
318
- Object.defineProperty(exports, 'resetPerformanceMetricsAtom', {
319
- enumerable: true,
320
- get: function () {
321
- return resetPerformanceMetricsAtom;
322
- }
323
- });
324
- Object.defineProperty(exports, 'widgetArrayAtom', {
325
- enumerable: true,
326
- get: function () {
327
- return widgetArrayAtom;
328
- }
329
- });
330
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdC1iSW1OUmNuUS5janMiLCJuYW1lcyI6WyJNZXNzYWdlVHlwZSIsInJlcGx5RXZlbnRDYXRlZ29yeUF0b20iLCJVc2VyRXZlbnRDYXRlZ29yeSIsInVzZXJRdWVyeUF0b20iLCJtZXNzYWdlc0F0b20iLCJ1c2VySGFzUmVwbGllZEF0b20iLCJTcGlmZnlNZXRyaWNzRXZlbnROYW1lIiwic2Vzc2lvblN0b3JhZ2VVdGlsIiwiZ2V0QXRvbVN0b3JlIiwicmVwbHlFdmVudENhdGVnb3J5QXRvbSIsIlVzZXJFdmVudENhdGVnb3J5IiwiZm9ybVN1Ym1pdEF0b20iLCJuZXdNZXNzYWdlOiBNZXNzYWdlIiwiTWVzc2FnZVJvbGUiLCJNZXNzYWdlVHlwZSIsInJlcGx5RXZlbnRDYXRlZ29yeUF0b20iLCJVc2VyRXZlbnRDYXRlZ29yeSIsInN1Z2dlc3Rpb25BdG9tIiwibWVzc2FnZXNBdG9tIiwidXNlckhhc1JlcGxpZWRBdG9tIiwibWVzc2FnZXNBdG9tIiwidXNlckhhc1JlcGxpZWRBdG9tIiwiZ2V0QXRvbVN0b3JlIiwidXNlckhhc1JlcGxpZWRBdG9tIiwicmVwbHlFdmVudENhdGVnb3J5QXRvbSIsInVzZXJRdWVyeUF0b20iLCJzdWdnZXN0aW9uQXRvbSIsImFza1F1ZXN0aW9uQnRuQ2xpY2tlZEF0b20iLCJtZXNzYWdlc0F0b20iLCJ1c2VyRXZlbnRzQXRvbSIsInN1Z2dlc3Rpb25zQXRvbSIsInN1Z2dlc3Rpb25zTG9hZGluZ0F0b20iLCJyZXNwb25zZVN0cmVhbWluZ0F0b20iLCJjaGF0SXNPcGVuQXRvbSIsImNoYXRPblRvZ2dsZUF0b20iXSwic291cmNlcyI6WyIuLi9zcmMvYXRvbXMvY2hhdC9tZXNzYWdlUXVldWUudHMiLCIuLi9zcmMvYXRvbXMvYW1wbGl0dWRlL2FtcGxpdHVkZVRyYWNrRXZlbnRBdG9tLnRzIiwiLi4vc3JjL2F0b21zL2NoYXQvcmVwbGllcy50cyIsIi4uL3NyYy9hdG9tcy9jaGF0L3BlcmZvcm1hbmNlTWV0cmljcy50cyIsIi4uL3NyYy9hdG9tcy9jaGF0L2Zvcm0udHMiLCIuLi9zcmMvYXRvbXMvY2hhdC9zdWdnZXN0aW9ucy50cyIsIi4uL3NyYy9hdG9tcy9jaGF0L2xhc3RNZXNzYWdlLnRzIiwiLi4vc3JjL2F0b21zL2NoYXQvcmVuZGVyZWRXaWRnZXRSZWZzLnRzIiwiLi4vc3JjL2F0b21zL2NoYXQvaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgYXRvbSB9IGZyb20gJ2pvdGFpJztcbmltcG9ydCB7IHY0IGFzIHV1aWQgfSBmcm9tICd1dWlkJztcbmltcG9ydCB7IEdlbmVyYXRpb25QYXJhbXMsIE5leHRNZXNzYWdlUmVxdWVzdCwgVXNlckV2ZW50IH0gZnJvbSAnQGVudml2ZS1haS90eXBlcyc7XG5pbXBvcnQgeyBnZXRBdG9tU3RvcmUgfSBmcm9tICdzcmMvYXRvbXMvYXRvbVN0b3JlL2F0b21TdG9yZSc7XG5pbXBvcnQgeyB1c2VySWRBdG9tLCBjaGF0SWRBdG9tIH0gZnJvbSAnc3JjL2F0b21zL2FwcCc7XG5pbXBvcnQgeyBvcmdTaG9ydE5hbWVBdG9tLCBjb250ZXh0U291cmNlQXRvbSwgZW52QXRvbSB9IGZyb20gJ3NyYy9hdG9tcy9lbnZpdmUvZW52aXZlQ29uZmlnJztcbmltcG9ydCB7IG9yZ0lkQXRvbSwgZmVhdHVyZUZsYWdTZXJ2aWNlQXRvbSB9IGZyb20gJ3NyYy9hdG9tcy9vcmcvZ3JhcGhxbENvbmZpZyc7XG5cbmNvbnN0IGludGVybmFsVXNlckV2ZW50UXVldWVBdG9tID0gYXRvbTxVc2VyRXZlbnRbXT4oW10pO1xuXG5leHBvcnQgY29uc3QgdXNlckV2ZW50UXVldWVBdG9tID0gYXRvbSgoZ2V0KSA9PiB7XG4gIGNvbnN0IHF1ZXVlID0gZ2V0KGludGVybmFsVXNlckV2ZW50UXVldWVBdG9tKTtcbiAgcmV0dXJuIHF1ZXVlID09PSB1bmRlZmluZWQgPyBbXSA6IHF1ZXVlLmZpbHRlcigodikgPT4gdiAhPT0gdW5kZWZpbmVkKTtcbn0pO1xuXG4vKipcbiAqIFRoaXMgYXRvbSBpcyB1c2VkIHRvIHF1ZXVlIGEgbmV3IG1lc3NhZ2UgZm9yIHByb2Nlc3Npbmcgb24gYG5leHRfcmVzcG9uc2VzYFxuICogSXQgcmVjZWl2ZXMgYSBzaW5nbGUgYHVzZXJFdmVudGAgdGhhdCBpcyBhZGRlZCB0byB0aGUgcHJvY2Vzc2luZyBxdWV1ZS5cbiAqIElmIHRoZSBldmVudCBoYXMgdGhlIHNhbWUgZXZlbnRJZCBhcyBhbiBleGlzdGluZyBtZXNzYWdlIGluIHRoZSBxdWV1ZSB0aGUgTkVXXG4gKiBldmVudCBpcyBpZ25vcmVkXG4gKi9cbmV4cG9ydCBjb25zdCBxdWV1ZVVzZXJFdmVudEF0b20gPSBhdG9tKG51bGwsIChnZXQsIHNldCwgdXNlckV2ZW50OiBVc2VyRXZlbnQpID0+IHtcbiAgaWYgKHVzZXJFdmVudCA9PT0gdW5kZWZpbmVkKSB7XG4gICAgcmV0dXJuO1xuICB9XG4gIHNldChpbnRlcm5hbFVzZXJFdmVudFF1ZXVlQXRvbSwgWy4uLmdldChpbnRlcm5hbFVzZXJFdmVudFF1ZXVlQXRvbSksIHVzZXJFdmVudF0pO1xufSk7XG5cbi8qKlxuICogVGhpcyBhdG9tIGV4cG9zZXMgYSBmdW5jdGlvbiB0byByZXNldCB0aGUgZW50aXJlIHF1ZXVlLiBBbGwgbWVzc2FnZXMgaW4gdGhlIHF1ZXVlIHdpbGwgYmUgcHVyZ2VkXG4gKi9cbmV4cG9ydCBjb25zdCBjbGVhclVzZXJFdmVudEF0b20gPSBhdG9tKG51bGwsIChfLCBzZXQpID0+IHtcbiAgc2V0KGludGVybmFsVXNlckV2ZW50UXVldWVBdG9tLCBbXSk7XG59KTtcblxuLyoqXG4gKiBUaGlzIGF0b20gaXMgdXNlZCB0byBtYXJrIGV2ZW50cyBhcyBwcm9jZXNzZWQgYW5kIHJlbW92ZSB0aGVtIGZyb20gdGhlIHF1ZXVlXG4gKiBJdCBhY2NlcHRzIGEgbGlzdCBvZiBldmVudElkIHZhbHVlcyBhbmQgd2lsbCByZW1vdmUgYWxsIGV2ZW50cyB3aXRoIHRob3NlIGV2ZW50SWRzIGZyb20gdGhlIHF1ZXVlLlxuICovXG5leHBvcnQgY29uc3QgcHJvY2Vzc1VzZXJFdmVudEF0b20gPSBhdG9tKG51bGwsIChnZXQsIHNldCwgZXZlbnRJZHM6IHN0cmluZ1tdKSA9PiB7XG4gIGNvbnN0IGN1cnIgPSBnZXQoaW50ZXJuYWxVc2VyRXZlbnRRdWV1ZUF0b20pO1xuICBjb25zdCByZW1haW5pbmcgPSBjdXJyPy5maWx0ZXIoKGV2ZW50KSA9PiAhZXZlbnRJZHMuaW5jbHVkZXMoZXZlbnQuZXZlbnRJZCkpO1xuICBzZXQoaW50ZXJuYWxVc2VyRXZlbnRRdWV1ZUF0b20sIHJlbWFpbmluZyk7XG59KTtcblxuZXhwb3J0IGNvbnN0IHVzZXJRdWV1ZUV2ZW50Q291bnRBdG9tID0gYXRvbSgoZ2V0KSA9PiBnZXQodXNlckV2ZW50UXVldWVBdG9tKS5sZW5ndGgpO1xuXG5pbXBvcnQgeyBDb250ZXh0U291cmNlRW51bSwgQ29udGV4dEVudkVudW0gfSBmcm9tICdAc3BpZmZ5LWFpL2NvbW1lcmNlLWFwaS1jbGllbnQnOyAvLyBJbXBvcnQgbmVjZXNzYXJ5IGVudW1zXG4vLyBpbXBvcnQgdHlwZSB7IENvbnRleHQgfSBmcm9tIFwiQHNwaWZmeS1haS9jb21tZXJjZS1hcGktY2xpZW50L2Rpc3QvbW9kZWxzL0NvbnRleHRcIjsgLy8gSW1wb3J0IENvbnRleHQgdHlwZVxuXG5leHBvcnQgY29uc3QgY3JlYXRlUmVzcG9uc2VQYXlsb2FkID0gKHtcbiAgdXNlckV2ZW50cyxcbiAgZ2VuZXJhdGlvblBhcmFtcyxcbn06IHtcbiAgdXNlckV2ZW50czogVXNlckV2ZW50W107XG4gIGdlbmVyYXRpb25QYXJhbXM/OiBHZW5lcmF0aW9uUGFyYW1zO1xufSk6IE5leHRNZXNzYWdlUmVxdWVzdCA9PiB7XG4gIGNvbnN0IGF0b21TdG9yZSA9IGdldEF0b21TdG9yZSgpO1xuICBjb25zdCBvcmdTaG9ydE5hbWUgPSBhdG9tU3RvcmUuZ2V0KG9yZ1Nob3J0TmFtZUF0b20pO1xuICBjb25zdCBvcmdJZCA9IGF0b21TdG9yZS5nZXQob3JnSWRBdG9tKTtcbiAgY29uc3QgdXNlcklkID0gYXRvbVN0b3JlLmdldCh1c2VySWRBdG9tKTtcbiAgY29uc3QgY2hhdElkID0gYXRvbVN0b3JlLmdldChjaGF0SWRBdG9tKTtcbiAgY29uc3Qgc291cmNlID0gYXRvbVN0b3JlLmdldChjb250ZXh0U291cmNlQXRvbSk7XG4gIGNvbnN0IGVudiA9IGF0b21TdG9yZS5nZXQoZW52QXRvbSk7XG5cbiAgY29uc3QgZmVhdHVyZUZsYWdTZXJ2aWNlID0gYXRvbVN0b3JlLmdldChmZWF0dXJlRmxhZ1NlcnZpY2VBdG9tKTtcblxuICBjb25zdCBjb250ZXh0ID0ge1xuICAgIHVzZXJJZDogdXNlcklkID8/ICcnLFxuICAgIG9yZ19pZDogb3JnSWQgPz8gJycsXG4gICAgb3JnX3Nob3J0X25hbWU6IG9yZ1Nob3J0TmFtZSA/PyAnJyxcbiAgICBjaGF0X2lkOiBjaGF0SWQgPz8gJycsXG4gICAgc291cmNlOiBzb3VyY2UgPz8gQ29udGV4dFNvdXJjZUVudW0uQXBwLFxuICAgIGVudjogKGVudiBhcyBDb250ZXh0RW52RW51bSkgPz8gQ29udGV4dEVudkVudW0uRGV2LFxuICB9O1xuXG4gIGNvbnN0IGZlYXR1cmVGbGFncyA9IGZlYXR1cmVGbGFnU2VydmljZT8uZmVhdHVyZUZsYWdTZXJ2aWNlPy5nZXRGZWF0dXJlRmxhZ3MoKSB8fCB7fTtcblxuICByZXR1cm4ge1xuICAgIGlkOiB1dWlkKCksXG4gICAgY29udGV4dCxcbiAgICB1c2VyRXZlbnRzLFxuICAgIGZlYXR1cmVGbGFnczogT2JqZWN0LmZyb21FbnRyaWVzKFxuICAgICAgT2JqZWN0LmVudHJpZXMoZmVhdHVyZUZsYWdzKS5maWx0ZXIoKFssIGlzRW5hYmxlZF0pID0+IGlzRW5hYmxlZCksXG4gICAgKSwgLy8gQ29udmVydCBiYWNrIHRvIFJlY29yZDxzdHJpbmcsIGJvb2xlYW4+XG4gICAgZ2VuZXJhdGlvblBhcmFtcyxcbiAgfTtcbn07XG4iLCJpbXBvcnQgeyBhdG9tIH0gZnJvbSBcImpvdGFpXCI7XG5pbXBvcnQgeyBTcGlmZnlNZXRyaWNzRXZlbnROYW1lIH0gZnJvbSBcInNyYy9jb250ZXh0cy9hbXBsaXR1ZGVDb250ZXh0L2FtcGxpdHVkZUNvbnRleHRcIjtcblxuaW50ZXJmYWNlIFRyYWNrRXZlbnRQYXJhbXMge1xuICBldmVudE5hbWU6IFNwaWZmeU1ldHJpY3NFdmVudE5hbWU7XG4gIGV2ZW50UHJvcHM/OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgZXZlbnRHcm91cHM/OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgYWxzb1NlbmRUb0dvb2dsZUFuYWx5dGljcz86IGJvb2xlYW47XG59XG5cbi8vIFRoaXMgYXRvbSB3aWxsIGhvbGQgdGhlIHRyYWNrRXZlbnQgZnVuY3Rpb24gZnJvbSB0aGUgQW1wbGl0dWRlIGNvbnRleHQuXG4vLyBJdCBuZWVkcyB0byBiZSBpbml0aWFsaXplZCBieSBhIFJlYWN0IGNvbXBvbmVudCB0aGF0IGhhcyBhY2Nlc3MgdG8gdGhlIHVzZUFtcGxpdHVkZSBob29rLlxuZXhwb3J0IGNvbnN0IGFtcGxpdHVkZVRyYWNrRXZlbnRBdG9tID0gYXRvbTxcbiAgKChwYXJhbXM6IFRyYWNrRXZlbnRQYXJhbXMpID0+IFByb21pc2U8dm9pZD4pIHwgbnVsbFxuPihudWxsKTtcbiIsImltcG9ydCB7IGF0b20gfSBmcm9tICdqb3RhaSc7XG5pbXBvcnQgeyBNZXNzYWdlLCBNZXNzYWdlVHlwZSB9IGZyb20gJ0BlbnZpdmUtYWkvdHlwZXMnO1xuaW1wb3J0IHtcbiAgdXNlckhhc1JlcGxpZWRBdG9tLFxuICBtZXNzYWdlc0F0b20sXG4gIHVzZXJRdWVyeUF0b20sXG4gIHJlcGx5RXZlbnRDYXRlZ29yeUF0b20sXG59IGZyb20gJ3NyYy9hdG9tcy9jaGF0JztcbmltcG9ydCB7IFVzZXJFdmVudENhdGVnb3J5IH0gZnJvbSAnQHNwaWZmeS1haS9jb21tZXJjZS1hcGktY2xpZW50JztcbmltcG9ydCB7IHF1ZXVlVXNlckV2ZW50QXRvbSB9IGZyb20gJy4vbWVzc2FnZVF1ZXVlJztcbmltcG9ydCB7IFNwaWZmeU1ldHJpY3NFdmVudE5hbWUgfSBmcm9tICdzcmMvY29udGV4dHMvYW1wbGl0dWRlQ29udGV4dC9hbXBsaXR1ZGVDb250ZXh0JztcbmltcG9ydCB7IGFtcGxpdHVkZVRyYWNrRXZlbnRBdG9tIH0gZnJvbSAnc3JjL2F0b21zL2FtcGxpdHVkZS9hbXBsaXR1ZGVUcmFja0V2ZW50QXRvbSc7XG5cbnR5cGUgSGFuZGxlUmVwbHlQYXJhbXMgPSB7XG4gIG1lc3NhZ2U6IE1lc3NhZ2U7XG4gIHVzZXJUeXBlZDogYm9vbGVhbjtcbn07XG5cbmV4cG9ydCBjb25zdCBoYW5kbGVSZXBseUF0b20gPSBhdG9tKG51bGwsIChnZXQsIHNldCwgeyBtZXNzYWdlLCB1c2VyVHlwZWQgfTogSGFuZGxlUmVwbHlQYXJhbXMpID0+IHtcbiAgaWYgKG1lc3NhZ2UudHlwZSAhPT0gTWVzc2FnZVR5cGUuUXVlcnlUeXBlZCkge1xuICAgIHJldHVybjtcbiAgfVxuXG4gIGNvbnN0IHRyYWNrRXZlbnQgPSBnZXQoYW1wbGl0dWRlVHJhY2tFdmVudEF0b20pO1xuICBjb25zdCBxdWVyeVR5cGVkID0gbWVzc2FnZS5tZXRhZGF0YS5jb250ZW50O1xuXG4gIHNldChyZXBseUV2ZW50Q2F0ZWdvcnlBdG9tLCBVc2VyRXZlbnRDYXRlZ29yeS5RdWVyeVR5cGVkKTtcbiAgc2V0KHVzZXJRdWVyeUF0b20sIHF1ZXJ5VHlwZWQpO1xuICBzZXQobWVzc2FnZXNBdG9tLCBbLi4uZ2V0KG1lc3NhZ2VzQXRvbSksIFttZXNzYWdlXV0pO1xuICBzZXQodXNlckhhc1JlcGxpZWRBdG9tLCB0cnVlKTtcbiAgc2V0KHF1ZXVlVXNlckV2ZW50QXRvbSwge1xuICAgIGV2ZW50SWQ6IG1lc3NhZ2UuaWQsXG4gICAgY3JlYXRlZEF0OiBtZXNzYWdlLmNyZWF0ZWRBdCxcbiAgICBjYXRlZ29yeTogVXNlckV2ZW50Q2F0ZWdvcnkuUXVlcnlUeXBlZCxcbiAgICBhdHRyaWJ1dGVzOiB7XG4gICAgICBxdWVyeTogcXVlcnlUeXBlZCxcbiAgICB9LFxuICB9KTtcblxuICBpZiAodHJhY2tFdmVudCkge1xuICAgIHRyYWNrRXZlbnQoe1xuICAgICAgZXZlbnROYW1lOiBTcGlmZnlNZXRyaWNzRXZlbnROYW1lLkNoYXRVc2VyTWVzc2FnZUlucHV0LFxuICAgICAgZXZlbnRQcm9wczoge1xuICAgICAgICBtZXNzYWdlX2lkOiBtZXNzYWdlLmlkLFxuICAgICAgICBtZXNzYWdlX3JvbGU6IG1lc3NhZ2Uucm9sZSxcbiAgICAgICAgbWVzc2FnZV90eXBlOiBtZXNzYWdlLnR5cGUsXG4gICAgICAgIG1lc3NhZ2VfbWV0YWRhdGE6IHtcbiAgICAgICAgICBjb250ZW50OiBtZXNzYWdlPy5tZXRhZGF0YT8uY29udGVudCwgLy8gUmVtb3ZlZCBhbXBsaXR1ZGVTYWZlU3RyaW5nXG4gICAgICAgICAgY3JlYXRlZF9hdDogbWVzc2FnZS5jcmVhdGVkQXQsXG4gICAgICAgICAgdXNlcl90eXBlZDogdXNlclR5cGVkLFxuICAgICAgICB9LFxuICAgICAgfSxcbiAgICAgIGFsc29TZW5kVG9Hb29nbGVBbmFseXRpY3M6IHRydWUsXG4gICAgfSk7XG4gIH1cbn0pO1xuIiwiaW1wb3J0IHsgYXRvbSB9IGZyb20gXCJqb3RhaVwiO1xuaW1wb3J0IHsgYXRvbVdpdGhTdG9yYWdlIH0gZnJvbSBcImpvdGFpL3V0aWxzXCI7XG5pbXBvcnQgTG9nZ2VyIGZyb20gXCJzcmMvYXBwbGljYXRpb24vbG9nZ2luZy9sb2dnZXJcIjtcbmltcG9ydCB7XG4gIGdldEF0b21TdG9yZSxcbiAgc2Vzc2lvblN0b3JhZ2VVdGlsLFxufSBmcm9tIFwic3JjL2F0b21zL2F0b21TdG9yZS9hdG9tU3RvcmVcIjtcblxuZXhwb3J0IGNvbnN0IEFQUF9JTklUSUFMX1NUQVJUX1RJTUVfS0VZID0gXCJzcGlmZnktYXBwLWluaXRpYWwtc3RhcnQtdGltZVwiO1xuZXhwb3J0IGNvbnN0IFBBR0VfTE9BRF9PRkZTRVRfVElNRV9LRVkgPSBcInNwaWZmeS1wYWdlLWxvYWQtb2Zmc2V0LXRpbWVcIjtcbi8qKlxuICogVGhlIGRpZmZlcmVudCBwZXJmb3JtYW5jZSBtZXRyaWNzIHRoYXQgY2FuIGJlIGxvZ2dlZC4gQWxsIHRpbWVzIGFyZSByZWxhdGl2ZSB0byB0aGVcbiAqIGluaXRpYWwgc3RhcnQgdGltZSBvZiB0aGUgYXBwIGFuZCBhcmUgc3RvcmVkIGluIG1pbGxpc2Vjb25kcy5cbiAqL1xuZXhwb3J0IGVudW0gUGVyZk1ldHJpY3NFdmVudHMge1xuICBQYWdlTG9hZE9mZnNldCA9IFwicGFnZV9sb2FkX29mZnNldF9tc1wiLFxuICBNYWluQnVuZGxlTG9hZGVkID0gXCJtYWluX2J1bmRsZV9sb2FkZWRfbXNcIixcbiAgT3JnQ29uZmlnTG9hZFN0YXJ0ZWQgPSBcIm9yZ19jb25maWdfbG9hZF9zdGFydGVkX21zXCIsXG4gIE9yZ0NvbmZpZ0xvYWRFbmRlZCA9IFwib3JnX2NvbmZpZ19sb2FkX2VuZGVkX21zXCIsXG4gIEZpcnN0UmVzcG9uc2VTdGFydGVkID0gXCJmaXJzdF9yZXNwb25zZV9zdGFydGVkX21zXCIsXG4gIEZpcnN0UmVzcG9uc2VDb21wbGV0ZWQgPSBcImZpcnN0X3Jlc3BvbnNlX2NvbXBsZXRlZF9tc1wiLFxuICBGaXJzdFN1Z2dlc3Rpb25zU3RhcnRlZCA9IFwiZmlyc3Rfc3VnZ2VzdGlvbnNfc3RhcnRlZF9tc1wiLFxuICBGaXJzdFN1Z2dlc3Rpb25zQ29tcGxldGVkID0gXCJmaXJzdF9zdWdnZXN0aW9uc19jb21wbGV0ZWRfbXNcIixcbiAgRW1iZWRkZWRXaWRnZXRSZW5kZXJlZCA9IFwiZW1iZWRkZWRfd2lkZ2V0X3JlbmRlcmVkX21zXCIsXG4gIEZsb2F0aW5nQnV0dG9uUmVuZGVyZWQgPSBcImZsb2F0aW5nX2J1dHRvbl9yZW5kZXJlZF9tc1wiLFxuICBUb3BTdWdnZXN0aW9uc0JhclJlbmRlcmVkID0gXCJ0b3Bfc3VnZ2VzdGlvbnNfYmFyX3JlbmRlcmVkX21zXCIsXG4gIEJvdHRvbVN1Z2dlc3Rpb25zQmFyUmVuZGVyZWQgPSBcImJvdHRvbV9zdWdnZXN0aW9uc19iYXJfcmVuZGVyZWRfbXNcIixcbiAgU2VhcmNoUHJvbXB0UmVuZGVyZWQgPSBcInNlYXJjaF9wcm9tcHRfcmVuZGVyZWRfbXNcIixcbn1cblxuY29uc3QgaW50ZXJuYWxQZXJmTWV0cmljc0F0b20gPSBhdG9tPE1hcDxQZXJmTWV0cmljc0V2ZW50cywgbnVtYmVyPj4obmV3IE1hcCgpKTtcbmV4cG9ydCBjb25zdCBhcHBJbml0aWFsU3RhcnRUaW1lTXNBdG9tID0gYXRvbVdpdGhTdG9yYWdlPHN0cmluZyB8IHVuZGVmaW5lZD4oXG4gIEFQUF9JTklUSUFMX1NUQVJUX1RJTUVfS0VZLFxuICBzZXNzaW9uU3RvcmFnZS5nZXRJdGVtKEFQUF9JTklUSUFMX1NUQVJUX1RJTUVfS0VZKSA/PyB1bmRlZmluZWQsXG4gIHNlc3Npb25TdG9yYWdlVXRpbCxcbiAge1xuICAgIGdldE9uSW5pdDogdHJ1ZSxcbiAgfVxuKTtcblxuZXhwb3J0IGNvbnN0IHBhZ2VMb2FkT2Zmc2V0VGltZUF0b20gPSBhdG9tV2l0aFN0b3JhZ2U8c3RyaW5nIHwgdW5kZWZpbmVkPihcbiAgUEFHRV9MT0FEX09GRlNFVF9USU1FX0tFWSxcbiAgc2Vzc2lvblN0b3JhZ2UuZ2V0SXRlbShQQUdFX0xPQURfT0ZGU0VUX1RJTUVfS0VZKSA/PyB1bmRlZmluZWQsXG4gIHNlc3Npb25TdG9yYWdlVXRpbCxcbiAge1xuICAgIGdldE9uSW5pdDogdHJ1ZSxcbiAgfVxuKTtcblxuLyoqXG4gKiBSZXNldHMgdGhlIHBlcmZvcm1hbmNlIG1ldHJpY3MgYXRvbSB0byBhbiBlbXB0eSBtYXAuIFRoaXMgc2hvdWxkIGJlIGNhbGxlZCBhZnRlciB0aGUgcGVyZm9ybWFuY2VcbiAqIG1ldHJpY3MgaGF2ZSBiZWVuIHJlcG9ydGVkIHRvIGFtcGxpdHVkZS4gT24gU1BBLCBpdCBlbnN1cmVzIHRoYXQgd2UgY2FuIHN0aWxsIGNhcHR1cmUgbWV0cmljcyBhc1xuICogdGhlIHVzZXIgbmF2aWdhdGVzIGFyb3VuZCB0aGUgYXBwIChhbHRob3VnaCB0aGVyZSBpcyBtb3JlIHdvcmsgdG8gYmUgZG9uZSB0byBmdWxseSBlbmFibGUgdGhpcykuXG4gKiBPbiBub24tU1BBLCBpdCBlbnN1cmVzIHRoYXQgcHJldmlvdXNseSBjYXB0dXJlZCBtZXRyaWNzIGFyZSBub3QgcmVwb3J0ZWQgYWdhaW4uXG4gKi9cbmV4cG9ydCBjb25zdCByZXNldFBlcmZvcm1hbmNlTWV0cmljc0F0b20gPSAoKSA9PiB7XG4gIGNvbnN0IGF0b21TdG9yZSA9IGdldEF0b21TdG9yZSgpO1xuICBhdG9tU3RvcmUuc2V0KGludGVybmFsUGVyZk1ldHJpY3NBdG9tLCBuZXcgTWFwKCkpO1xufTtcbmV4cG9ydCBjb25zdCBwZXJmb3JtYW5jZU1ldHJpY3NBdG9tID0gYXRvbSgoZ2V0KSA9PlxuICBnZXQoaW50ZXJuYWxQZXJmTWV0cmljc0F0b20pXG4pO1xuZXhwb3J0IGNvbnN0IGhhc1JlcG9ydGVkUGVyZm9ybWFuY2VNZXRyaWNzQXRvbSA9IGF0b208Ym9vbGVhbj4oZmFsc2UpO1xuXG4vKipcbiAqIExvZ3MgYSBwZXJmb3JtYW5jZSBtZXRyaWMgYnkgY2FwdHVyaW5nIHRoZSBkZWx0YSBiZXR3ZWVuIHRoZSBpbml0aWFsIGFwcCBzdGFydCB0aW1lXG4gKiBhbmQgdGhlIGN1cnJlbnQgdGltZS4gSWYgdGhlIG1ldHJpYyBoYXMgYWxyZWFkeSBiZWVuIGxvZ2dlZCwgaXQgd2lsbCBub3QgYmUgbG9nZ2VkIGFnYWluLlxuICpcbiAqIEBwYXJhbSB2YWx1ZSBUaGUgcGVyZm9ybWFuY2UgbWV0cmljIG5hbWUgdG8gbG9nLlxuICovXG5leHBvcnQgY29uc3QgbG9nUGVyZk1ldHJpY0F0b20gPSBhdG9tKFxuICBudWxsLFxuICAoZ2V0LCBzZXQsIHZhbHVlOiBQZXJmTWV0cmljc0V2ZW50cykgPT4ge1xuICAgIGNvbnN0IGluaXRpYWxUaW1lU3RvcmFnZVZhbHVlID0gZ2V0KGFwcEluaXRpYWxTdGFydFRpbWVNc0F0b20pO1xuICAgIGNvbnN0IGluaXRpYWxUaW1lTXMgPSBpbml0aWFsVGltZVN0b3JhZ2VWYWx1ZVxuICAgICAgPyBwYXJzZUludChpbml0aWFsVGltZVN0b3JhZ2VWYWx1ZSwgMTApXG4gICAgICA6IHVuZGVmaW5lZDtcblxuICAgIGlmIChpbml0aWFsVGltZU1zID09IG51bGwpIHtcbiAgICAgIExvZ2dlci5sb2dXYXJuKFxuICAgICAgICBgW3NwaWZmeS1haV0gTm8gaW5pdGlhbCBhcHAgc3RhcnQgdGltZSBmb3VuZC4gU2tpcHBpbmcuLi5gLFxuICAgICAgICB1bmRlZmluZWRcbiAgICAgICk7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgY29uc3QgY3VycmVudFBlcmZNZXRyaWNzID0gZ2V0KGludGVybmFsUGVyZk1ldHJpY3NBdG9tKTtcbiAgICBpZiAoY3VycmVudFBlcmZNZXRyaWNzLmhhcyh2YWx1ZSkpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBjb25zdCBjdXJyZW50VGltZU1zID0gRGF0ZS5ub3coKTtcbiAgICBjb25zdCBkZWx0YU1zID0gY3VycmVudFRpbWVNcyAtIGluaXRpYWxUaW1lTXM7XG4gICAgY3VycmVudFBlcmZNZXRyaWNzLnNldCh2YWx1ZSwgZGVsdGFNcyk7XG4gICAgc2V0KGludGVybmFsUGVyZk1ldHJpY3NBdG9tLCBjdXJyZW50UGVyZk1ldHJpY3MpO1xuICB9XG4pO1xuIiwiaW1wb3J0IHsgVXNlckV2ZW50Q2F0ZWdvcnkgfSBmcm9tICdAc3BpZmZ5LWFpL2NvbW1lcmNlLWFwaS1jbGllbnQnO1xuaW1wb3J0IHsgYXRvbSB9IGZyb20gJ2pvdGFpJztcbmltcG9ydCB7IEZvcm1TdWJtaXR0ZWRBdHRyaWJ1dGVzLCBVc2VyRXZlbnQgfSBmcm9tICdAZW52aXZlLWFpL3R5cGVzJztcbmltcG9ydCB7IHY0IGFzIHV1aWQgfSBmcm9tICd1dWlkJztcbmltcG9ydCB7IGZvcm1TdWJtaXRBdG9tLCByZXBseUV2ZW50Q2F0ZWdvcnlBdG9tIH0gZnJvbSAnc3JjL2F0b21zL2NoYXQnO1xuaW1wb3J0IHsgcXVldWVVc2VyRXZlbnRBdG9tIH0gZnJvbSAnc3JjL2F0b21zL2NoYXQvbWVzc2FnZVF1ZXVlJztcblxuZXhwb3J0IGNvbnN0IGhhbmRsZUZvcm1TdWJtaXR0ZWRBdG9tID0gYXRvbShudWxsLCAoXywgc2V0LCBmb3JtOiBGb3JtU3VibWl0dGVkQXR0cmlidXRlcykgPT4ge1xuICBzZXQocmVwbHlFdmVudENhdGVnb3J5QXRvbSwgVXNlckV2ZW50Q2F0ZWdvcnkuRm9ybVN1Ym1pdHRlZCk7XG4gIHNldChmb3JtU3VibWl0QXRvbSwgZm9ybSk7XG5cbiAgY29uc3QgZm9ybVVzZXJFdmVudCA9IHtcbiAgICBldmVudElkOiB1dWlkKCksXG4gICAgY2F0ZWdvcnk6IFVzZXJFdmVudENhdGVnb3J5LkZvcm1TdWJtaXR0ZWQsXG4gICAgY3JlYXRlZEF0OiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXG4gICAgYXR0cmlidXRlczogZm9ybSxcbiAgfSBzYXRpc2ZpZXMgVXNlckV2ZW50O1xuICBzZXQocXVldWVVc2VyRXZlbnRBdG9tLCBmb3JtVXNlckV2ZW50KTtcbn0pO1xuIiwiaW1wb3J0IHsgYXRvbSB9IGZyb20gXCJqb3RhaVwiO1xuaW1wb3J0IHsgTWVzc2FnZSwgTWVzc2FnZVJvbGUsIE1lc3NhZ2VUeXBlLCBTdWdnZXN0aW9uIH0gZnJvbSBcInNyYy90eXBlc1wiO1xuaW1wb3J0IHtcbiAgbWVzc2FnZXNBdG9tLFxuICByZXBseUV2ZW50Q2F0ZWdvcnlBdG9tLFxuICBzdWdnZXN0aW9uQXRvbSxcbiAgdXNlckhhc1JlcGxpZWRBdG9tLFxufSBmcm9tIFwic3JjL2F0b21zL2NoYXRcIjtcbmltcG9ydCB7IEZpbHRlckF0dHJpYnV0ZSwgUHJvZHVjdFJlc3BvbnNlQXR0cmlidXRlcyB9IGZyb20gXCJzcmMvdHlwZXNcIjtcblxuaW1wb3J0IHsgVXNlckV2ZW50Q2F0ZWdvcnkgfSBmcm9tIFwiQHNwaWZmeS1haS9jb21tZXJjZS1hcGktY2xpZW50XCI7XG5pbXBvcnQgeyBxdWV1ZVVzZXJFdmVudEF0b20gfSBmcm9tIFwiLi9tZXNzYWdlUXVldWVcIjtcblxuZXhwb3J0IGNvbnN0IGhhbmRsZVN1Z2dlc3Rpb25BdG9tID0gYXRvbShcbiAgbnVsbCxcbiAgKGdldCwgc2V0LCBzdWdnZXN0aW9uOiBTdWdnZXN0aW9uKSA9PiB7XG4gICAgY29uc3QgbmV3TWVzc2FnZTogTWVzc2FnZSA9IHtcbiAgICAgIGlkOiBzdWdnZXN0aW9uLmlkLFxuICAgICAgcm9sZTogTWVzc2FnZVJvbGUuVXNlcixcbiAgICAgIHR5cGU6IE1lc3NhZ2VUeXBlLlN1Z2dlc3Rpb25DbGlja2VkLFxuICAgICAgY3JlYXRlZEF0OiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXG4gICAgICBtZXRhZGF0YToge1xuICAgICAgICBzdWdnZXN0aW9uSWQ6IHN1Z2dlc3Rpb24uaWQsXG4gICAgICAgIHN1Z2dlc3Rpb25Db250ZW50OiBzdWdnZXN0aW9uLmNvbnRlbnQsXG4gICAgICB9LFxuICAgIH07XG5cbiAgICBzZXQocmVwbHlFdmVudENhdGVnb3J5QXRvbSwgVXNlckV2ZW50Q2F0ZWdvcnkuU3VnZ2VzdGlvbkNsaWNrZWQpO1xuICAgIHNldChzdWdnZXN0aW9uQXRvbSwgc3VnZ2VzdGlvbik7XG4gICAgc2V0KG1lc3NhZ2VzQXRvbSwgWy4uLmdldChtZXNzYWdlc0F0b20pLCBbbmV3TWVzc2FnZV1dKTtcbiAgICBzZXQodXNlckhhc1JlcGxpZWRBdG9tLCB0cnVlKTtcbiAgICBzZXQocXVldWVVc2VyRXZlbnRBdG9tLCB7XG4gICAgICBldmVudElkOiBzdWdnZXN0aW9uLmlkLFxuICAgICAgY2F0ZWdvcnk6IFVzZXJFdmVudENhdGVnb3J5LlN1Z2dlc3Rpb25DbGlja2VkLFxuICAgICAgY3JlYXRlZEF0OiBuZXdNZXNzYWdlLmNyZWF0ZWRBdCxcbiAgICAgIGF0dHJpYnV0ZXM6IHtcbiAgICAgICAgc3VnZ2VzdGlvbklkOiBzdWdnZXN0aW9uLmlkLFxuICAgICAgfSxcbiAgICB9KTtcbiAgfVxuKTtcbiIsImltcG9ydCB7IGF0b20gfSBmcm9tICdqb3RhaSc7XG5pbXBvcnQgeyBtZXNzYWdlc0F0b20sIHVzZXJIYXNSZXBsaWVkQXRvbSB9IGZyb20gJ3NyYy9hdG9tcy9jaGF0JztcblxuZXhwb3J0IGNvbnN0IGxhc3RBc3Npc3RhbnRNZXNzYWdlQXRvbSA9IGF0b20oKGdldCkgPT4ge1xuICBjb25zdCBtZXNzYWdlcyA9IGdldChtZXNzYWdlc0F0b20pO1xuICBjb25zdCB1c2VySGFzUmVwbGllZCA9IGdldCh1c2VySGFzUmVwbGllZEF0b20pO1xuICBpZiAobWVzc2FnZXMubGVuZ3RoID4gMCAmJiAhdXNlckhhc1JlcGxpZWQpIHtcbiAgICByZXR1cm4gbWVzc2FnZXNbbWVzc2FnZXMubGVuZ3RoIC0gMV07XG4gIH1cbiAgcmV0dXJuIG51bGw7XG59KTtcbiIsImltcG9ydCB7IGF0b20gfSBmcm9tIFwiam90YWlcIjtcbmltcG9ydCB7IGdldEF0b21TdG9yZSB9IGZyb20gXCJzcmMvYXRvbXMvYXRvbVN0b3JlL2F0b21TdG9yZVwiO1xuXG5jb25zdCBpbnRlcm5hbFdpZGdldEFycmF5QXRvbSA9IGF0b208SFRNTEVsZW1lbnRbXT4oW10pO1xuXG5leHBvcnQgY29uc3Qgd2lkZ2V0QXJyYXlBdG9tID0gYXRvbSgoZ2V0KSA9PiBnZXQoaW50ZXJuYWxXaWRnZXRBcnJheUF0b20pKTtcblxuLyoqXG4gKiBUaGlzIGZ1bmN0aW9uIGNhbGwgaXMgdXNlZCB0byBjcmVhdGUgYSBsaXN0IG9mIHRoZSBTcGlmZnkgd2lkZ2V0c1xuICogdGhhdCBhcmUgcmVuZGVyaW5nIG9uIHRoZSBwYWdlLlxuICpcbiAqIEl0IGlzIHVzZWQgYnkgdGhlIEZsb2F0aW5nQnV0dG9uIHdpZGdldCB0byBzbGlkZSBvdXQgb2YgdmlldyB3aGVuIGl0XG4gKiBvdmVybGFwcyB3aXRoIGFub3RoZXIgU3BpZmZ5IHdpZGdldC5cbiAqXG4gKiBUaGUgc3BpZmZ5IHdpZGdldHMgc2hvdWxkIGJlIGFkZGVkIHRvIHRoZSB0aGlzIGFycmF5IGZvciB0aGUgZmxvYXRpbmcgYnV0dG9uXG4gKiB0byBrbm93IGFib3V0IHRoZW0uXG4gKlxuICogVG9kYXkgdGhlIFwiU3VnZ2VzdGlvbkJhclwiIHdpZGdldCBpcyBub3QgYWRkZWQsIGJ1dCB0aGUgb3RoZXIgd2lkZ2V0cyBhcmUuXG4gKlxuICovXG5leHBvcnQgY29uc3QgYWRkV2lkZ2V0ID0gKHJlZjogSFRNTEVsZW1lbnQsIGlkeD86IG51bWJlcikgPT4ge1xuICBjb25zdCBhdG9tU3RvcmUgPSBnZXRBdG9tU3RvcmUoKTtcbiAgY29uc3QgdmFsID0gYXRvbVN0b3JlLmdldChpbnRlcm5hbFdpZGdldEFycmF5QXRvbSk7XG4gIGNvbnN0IGluc2VydElkeCA9IGlkeCA/PyB2YWwubGVuZ3RoO1xuICB2YWxbaW5zZXJ0SWR4XSA9IHJlZjtcbiAgYXRvbVN0b3JlLnNldChpbnRlcm5hbFdpZGdldEFycmF5QXRvbSwgdmFsKTtcbn07XG4iLCJpbXBvcnQgeyBhdG9tIH0gZnJvbSAnam90YWknO1xuaW1wb3J0IHtcbiAgcmVzcG9uc2VTdHJlYW1pbmdBdG9tLFxuICBzdWdnZXN0aW9uc0xvYWRpbmdBdG9tLFxuICBtZXNzYWdlc0F0b20sXG4gIHVzZXJRdWVyeUF0b20sXG4gIHJlcGx5RXZlbnRDYXRlZ29yeUF0b20sXG4gIHVzZXJIYXNSZXBsaWVkQXRvbSxcbiAgYXNrUXVlc3Rpb25CdG5DbGlja2VkQXRvbSxcbiAgc3VnZ2VzdGlvbnNBdG9tLFxuICBjaGF0SXNPcGVuQXRvbSxcbiAgY2hhdE9uVG9nZ2xlQXRvbSxcbiAgdXNlckV2ZW50c0F0b20sXG4gIHN1Z2dlc3Rpb25BdG9tLFxufSBmcm9tICcuL2NoYXRTdGF0ZSc7XG5cbmV4cG9ydCAqIGZyb20gJy4vY2hhdFN0YXRlJztcbmV4cG9ydCAqIGZyb20gJy4vcmVwbGllcyc7XG5leHBvcnQgKiBmcm9tICcuL3BlcmZvcm1hbmNlTWV0cmljcyc7XG5leHBvcnQgKiBmcm9tICcuL2Zvcm0nO1xuZXhwb3J0ICogZnJvbSAnLi9zdWdnZXN0aW9ucyc7XG5leHBvcnQgKiBmcm9tICcuL2xhc3RNZXNzYWdlJztcbmV4cG9ydCAqIGZyb20gJy4vcmVuZGVyZWRXaWRnZXRSZWZzJztcblxuZXhwb3J0IGNvbnN0IGNoYXRBdG9tID0gYXRvbSgoZ2V0KSA9PiAoe1xuICB1c2VySGFzUmVwbGllZDogZ2V0KHVzZXJIYXNSZXBsaWVkQXRvbSksXG4gIHJlcGx5RXZlbnRDYXRlZ29yeTogZ2V0KHJlcGx5RXZlbnRDYXRlZ29yeUF0b20pLFxuICB1c2VyUXVlcnk6IGdldCh1c2VyUXVlcnlBdG9tKSxcbiAgc3VnZ2VzdGlvbjogZ2V0KHN1Z2dlc3Rpb25BdG9tKSxcbiAgYXNrUXVlc3Rpb25CdG5DbGlja2VkOiBnZXQoYXNrUXVlc3Rpb25CdG5DbGlja2VkQXRvbSksXG4gIG1lc3NhZ2VzOiBnZXQobWVzc2FnZXNBdG9tKSxcbiAgdXNlckV2ZW50czogZ2V0KHVzZXJFdmVudHNBdG9tKSxcbiAgc3VnZ2VzdGlvbnM6IGdldChzdWdnZXN0aW9uc0F0b20pLFxuICBzdWdnZXN0aW9uc0xvYWRpbmc6IGdldChzdWdnZXN0aW9uc0xvYWRpbmdBdG9tKSxcbiAgcmVzcG9uc2VTdHJlYW1pbmc6IGdldChyZXNwb25zZVN0cmVhbWluZ0F0b20pLFxuICBpc09wZW46IGdldChjaGF0SXNPcGVuQXRvbSksXG4gIG9uVG9nZ2xlOiBnZXQoY2hhdE9uVG9nZ2xlQXRvbSksXG59KSk7XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBUUEsTUFBTSw2Q0FBK0MsRUFBRSxDQUFDO0FBRXhELE1BQWEsc0NBQTJCLFFBQVE7Q0FDOUMsTUFBTSxRQUFRLElBQUksMkJBQTJCO0FBQzdDLFFBQU8sVUFBVSxTQUFZLEVBQUUsR0FBRyxNQUFNLFFBQVEsTUFBTSxNQUFNLE9BQVU7RUFDdEU7Ozs7Ozs7QUFRRixNQUFhLHFDQUEwQixPQUFPLEtBQUssS0FBSyxjQUF5QjtBQUMvRSxLQUFJLGNBQWMsT0FDaEI7QUFFRixLQUFJLDRCQUE0QixDQUFDLEdBQUcsSUFBSSwyQkFBMkIsRUFBRSxVQUFVLENBQUM7RUFDaEY7Ozs7QUFLRixNQUFhLHFDQUEwQixPQUFPLEdBQUcs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