@envive-ai/react-hooks 0.2.6-alpha-10 → 0.2.6-alpha-arthur-1

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 (125) hide show
  1. package/dist/{AmplitudeOperations-BgeRn1-f.js → AmplitudeOperations-CXU93q_4.js} +2 -2
  2. package/dist/{AmplitudeOperations-D_Ryy56B.cjs → AmplitudeOperations-_uIb5QED.cjs} +2 -2
  3. package/dist/{NewOrgConfig-C9aQqGsb.js → NewOrgConfig-CUQGpmw9.js} +2 -2
  4. package/dist/{NewOrgConfig-CHkHRqHx.cjs → NewOrgConfig-DD0oloeK.cjs} +2 -2
  5. package/dist/{TrackComponentVisibleEvent-IUaBjwTi.cjs → TrackComponentVisibleEvent-D-hN3owz.cjs} +2 -2
  6. package/dist/{TrackComponentVisibleEvent-BDG4krXl.js → TrackComponentVisibleEvent-D-pvNVOP.js} +2 -2
  7. package/dist/amplitudeContext-DFXeRauX.js +233 -0
  8. package/dist/amplitudeContext-DrFGH7Q7.cjs +249 -0
  9. package/dist/{amplitudeTrackEventAtom-iEIu11Om.js → amplitudeTrackEventAtom-D66l5oFp.js} +1 -1
  10. package/dist/{amplitudeTrackEventAtom-D9vs8dpE.cjs → amplitudeTrackEventAtom-f22P2U0u.cjs} +1 -1
  11. package/dist/application/utils/index.d.cts +3 -3
  12. package/dist/atoms/app/index.d.cts +7 -7
  13. package/dist/atoms/app/index.d.ts +7 -7
  14. package/dist/atoms/atomStore/index.d.ts +1 -1
  15. package/dist/atoms/chat/index.cjs +3 -3
  16. package/dist/atoms/chat/index.d.cts +31 -31
  17. package/dist/atoms/chat/index.d.ts +32 -32
  18. package/dist/atoms/chat/index.js +3 -3
  19. package/dist/atoms/globalSearch/index.cjs +1 -1
  20. package/dist/atoms/globalSearch/index.d.cts +6 -6
  21. package/dist/atoms/globalSearch/index.d.ts +6 -6
  22. package/dist/atoms/globalSearch/index.js +1 -1
  23. package/dist/atoms/org/index.d.cts +15 -15
  24. package/dist/atoms/org/index.d.ts +16 -16
  25. package/dist/atoms/search/index.cjs +3 -3
  26. package/dist/atoms/search/index.d.cts +13 -13
  27. package/dist/atoms/search/index.d.ts +14 -14
  28. package/dist/atoms/search/index.js +3 -3
  29. package/dist/atoms/search/utils.cjs +1 -1
  30. package/dist/atoms/search/utils.d.ts +1 -1
  31. package/dist/atoms/search/utils.js +1 -1
  32. package/dist/{cdnContext-td-gAam0.js → cdnContext-D-qi0lub.js} +1 -1
  33. package/dist/{cdnContext-i5iYA8ig.cjs → cdnContext-DIVgBnsl.cjs} +1 -1
  34. package/dist/{chat-ByolnbMa.js → chat-ZaY3KY-R.js} +3 -3
  35. package/dist/{chat-EVIlt-8f.cjs → chat-rhEjVfrd.cjs} +3 -3
  36. package/dist/{chatState-D_UIPkjR.js → chatState-BbI93m6r.js} +1 -1
  37. package/dist/{chatState-CIhb4BWP.cjs → chatState-DlJpHAsW.cjs} +1 -1
  38. package/dist/config/locators/components/chat/variants/index.d.cts +1 -1
  39. package/dist/config/locators/components/search/index.d.cts +1 -1
  40. package/dist/contexts/amplitudeContext/index.cjs +3 -3
  41. package/dist/contexts/amplitudeContext/index.js +3 -3
  42. package/dist/contexts/cdnContext/index.cjs +1 -1
  43. package/dist/contexts/cdnContext/index.js +1 -1
  44. package/dist/contexts/enviveConfigContext/index.cjs +1 -1
  45. package/dist/contexts/enviveConfigContext/index.d.cts +1 -1
  46. package/dist/contexts/enviveConfigContext/index.js +1 -1
  47. package/dist/contexts/enviveCssContext/index.cjs +3 -3
  48. package/dist/contexts/enviveCssContext/index.d.cts +1 -1
  49. package/dist/contexts/enviveCssContext/index.js +3 -3
  50. package/dist/contexts/featureFlagContext/index.cjs +1 -1
  51. package/dist/contexts/featureFlagContext/index.js +1 -1
  52. package/dist/contexts/featureFlagServiceContext/index.cjs +1 -1
  53. package/dist/contexts/featureFlagServiceContext/index.js +1 -1
  54. package/dist/contexts/newOrgConfigContext/index.cjs +2 -2
  55. package/dist/contexts/newOrgConfigContext/index.js +2 -2
  56. package/dist/contexts/searchContext/index.cjs +3 -3
  57. package/dist/contexts/searchContext/index.js +3 -3
  58. package/dist/contexts/systemSettingsContext/index.d.cts +3 -3
  59. package/dist/contexts/userIdentityContext/index.d.cts +1 -1
  60. package/dist/{enviveConfigContext-De7lbzWa.js → enviveConfigContext-CIavPzcm.js} +2 -1
  61. package/dist/{enviveConfigContext-B2NpnOjc.cjs → enviveConfigContext-DHEs077z.cjs} +2 -1
  62. package/dist/exceptions/index.d.cts +1 -1
  63. package/dist/{featureFlagServiceContext-Cvp7NlpC.js → featureFlagServiceContext-Dr8Cw23n.js} +1 -1
  64. package/dist/{featureFlagServiceContext-DHtkQAtq.cjs → featureFlagServiceContext-rxZhpdEj.cjs} +1 -1
  65. package/dist/{globalSearch-DEKvpWV-.cjs → globalSearch-Cpc8egsM.cjs} +1 -1
  66. package/dist/{globalSearch-D_nMjrlf.js → globalSearch-nmrfGLOn.js} +1 -1
  67. package/dist/hooks/AmplitudeOperations/index.cjs +4 -4
  68. package/dist/hooks/AmplitudeOperations/index.js +4 -4
  69. package/dist/hooks/AppDetails/index.d.cts +1 -1
  70. package/dist/hooks/BlockBackButton/index.d.cts +1 -1
  71. package/dist/hooks/CdnOperations/index.cjs +1 -1
  72. package/dist/hooks/CdnOperations/index.js +1 -1
  73. package/dist/hooks/ChatToggle/index.cjs +6 -6
  74. package/dist/hooks/ChatToggle/index.d.cts +1 -1
  75. package/dist/hooks/ChatToggle/index.js +6 -6
  76. package/dist/hooks/ChatToggleAnalytics/index.cjs +5 -5
  77. package/dist/hooks/ChatToggleAnalytics/index.d.ts +1 -1
  78. package/dist/hooks/ChatToggleAnalytics/index.js +5 -5
  79. package/dist/hooks/Debounce/index.d.ts +1 -1
  80. package/dist/hooks/GrabAndScroll/index.d.cts +2 -2
  81. package/dist/hooks/GrabAndScroll/index.d.ts +2 -2
  82. package/dist/hooks/ImageResolver/index.d.ts +1 -1
  83. package/dist/hooks/IsSmallScreen/index.d.cts +1 -1
  84. package/dist/hooks/LocalStorageOperations/index.d.cts +1 -1
  85. package/dist/hooks/MessageFilter/index.d.cts +1 -1
  86. package/dist/hooks/NewOrgConfig/index.cjs +3 -3
  87. package/dist/hooks/NewOrgConfig/index.d.cts +2 -2
  88. package/dist/hooks/NewOrgConfig/index.d.ts +2 -2
  89. package/dist/hooks/NewOrgConfig/index.js +3 -3
  90. package/dist/hooks/Search/index.cjs +9 -9
  91. package/dist/hooks/Search/index.js +9 -9
  92. package/dist/hooks/SearchOperations/index.cjs +3 -3
  93. package/dist/hooks/SearchOperations/index.js +3 -3
  94. package/dist/hooks/SessionStorageOperations/index.d.cts +1 -1
  95. package/dist/hooks/ShopifyUrlOperations/index.d.cts +2 -2
  96. package/dist/hooks/ShopifyUrlOperations/index.d.ts +2 -2
  97. package/dist/hooks/SystemSettingsContext/index.d.cts +2 -2
  98. package/dist/hooks/TrackComponentVisibleEvent/index.cjs +4 -4
  99. package/dist/hooks/TrackComponentVisibleEvent/index.js +4 -4
  100. package/dist/hooks/UpdateAnalyticsProps/index.cjs +3 -3
  101. package/dist/hooks/UpdateAnalyticsProps/index.js +3 -3
  102. package/dist/hooks/utils.d.cts +1 -1
  103. package/dist/hooks/utils.d.ts +1 -1
  104. package/dist/index-CMZcE7pk.d.cts +1 -1
  105. package/dist/index-CsmO1rDH.d.ts +1 -1
  106. package/dist/index-DtzPIcQp.d.ts +1 -1
  107. package/dist/index-bEjLKG_Q.d.ts +1 -1
  108. package/dist/index-mv7KvWDq.d.ts +1 -1
  109. package/dist/index-npqPeJ1g.d.cts +1 -1
  110. package/dist/{newOrgConfigContext-Dt_W-wlK.js → newOrgConfigContext-BdzDnzUM.js} +2 -2
  111. package/dist/{newOrgConfigContext-Brsggue9.cjs → newOrgConfigContext-cWeJAT6E.cjs} +2 -2
  112. package/dist/nodeSelector-BYEAyrsj.d.cts +1 -1
  113. package/dist/{search-OWKKaQQj.js → search-CsXXamht.js} +3 -3
  114. package/dist/{search-CWkDtqQV.cjs → search-DHhDpeQU.cjs} +3 -3
  115. package/dist/{searchContext-DeNsXmHq.js → searchContext-B5ZNLtvd.js} +3 -3
  116. package/dist/{searchContext-CjB3sD6b.cjs → searchContext-SFvF_Y8J.cjs} +3 -3
  117. package/dist/{searchServiceAdapter-B0h7psvh.js → searchServiceAdapter-BkvDIhV8.js} +1 -1
  118. package/dist/{searchServiceAdapter-BclWy4fE.cjs → searchServiceAdapter-C4zzBUl0.cjs} +1 -1
  119. package/dist/{utils-DQhbbAnt.js → utils-BRNaQkCK.js} +1 -1
  120. package/dist/{utils-CqVRbvfN.cjs → utils-Buo4KU5l.cjs} +1 -1
  121. package/package.json +1 -1
  122. package/src/contexts/amplitudeContext/amplitudeContext.tsx +4 -0
  123. package/src/contexts/enviveConfigContext/enviveConfigContext.tsx +1 -0
  124. package/dist/amplitudeContext-BM6ctR3o.cjs +0 -248
  125. package/dist/amplitudeContext-DHCtfce5.js +0 -232
@@ -1,248 +0,0 @@
1
- const require_chunk = require('./chunk-CUT6urMc.cjs');
2
- const require_events = require('./events-CgFGtanE.cjs');
3
- const require_dist = require('./dist-VxRI6eQv.cjs');
4
- const require_logger = require('./logger-BqHq67zN.cjs');
5
- const require_enviveConfig = require('./enviveConfig-CQ_8mT-g.cjs');
6
- const require_org = require('./org-BNs_maoW.cjs');
7
- const require_app = require('./app-DxwdFQlv.cjs');
8
- const require_localStorageContext = require('./localStorageContext-CWc5xJ6U.cjs');
9
- const require_enviveConfigContext = require('./enviveConfigContext-B2NpnOjc.cjs');
10
- const require_featureFlagServiceContext = require('./featureFlagServiceContext-DHtkQAtq.cjs');
11
- let react = require("react");
12
- react = require_chunk.__toESM(react);
13
- let jotai = require("jotai");
14
- jotai = require_chunk.__toESM(jotai);
15
- let react_jsx_runtime = require("react/jsx-runtime");
16
- react_jsx_runtime = require_chunk.__toESM(react_jsx_runtime);
17
- let __amplitude_analytics_browser = require("@amplitude/analytics-browser");
18
- __amplitude_analytics_browser = require_chunk.__toESM(__amplitude_analytics_browser);
19
-
20
- //#region src/contexts/amplitudeContext/amplitudeContext.tsx
21
- const AmplitudeContext = (0, react.createContext)(null);
22
- const AmplitudeProvider = ({ children }) => {
23
- const userId = (0, jotai.useAtomValue)(require_app.userIdAtom);
24
- const amplitudeApiKey = (0, jotai.useAtomValue)(require_enviveConfig.amplitudeApiKeyAtom);
25
- const dataResidency = (0, jotai.useAtomValue)(require_enviveConfig.dataResidencyAtom);
26
- const orgGaConfig = (0, jotai.useAtomValue)(require_org.orgAnalyticsGoogleAnalyticsConfigAtom);
27
- const env = (0, jotai.useAtomValue)(require_enviveConfig.envAtom);
28
- const contextSource = (0, jotai.useAtomValue)(require_enviveConfig.contextSourceAtom);
29
- (0, jotai.useAtomValue)(require_enviveConfig.identifyingPrefixAtom);
30
- const { getItem } = require_localStorageContext.useLocalStorage();
31
- const { publicKey, featureOverrides, variantUrlOverride, variantInfoOverride, show, orgShortName, featureGates } = require_enviveConfigContext.useEnviveConfig();
32
- const { featureFlagService } = require_featureFlagServiceContext.useFeatureFlagService();
33
- const [amplitudeClient, setAmplitudeClient] = react.default.useState(void 0);
34
- const [internalEventTrackingEnrichment, setInternalEventTrackingEnrichment] = react.default.useState(void 0);
35
- const [supplementalDefaultProps, setSupplementalDefaultProps] = react.default.useState({});
36
- const isReady = Boolean(userId && featureFlagService && amplitudeApiKey && userId);
37
- const getDefaultTrackingProps = (0, react.useCallback)(() => {
38
- const gatesProps = featureGates ? featureGates.reduce((acc, curr) => {
39
- if (curr.name && curr.value != null) return {
40
- ...acc,
41
- [`feature_gate.${curr.name}`]: curr.value
42
- };
43
- return acc;
44
- }, {}) : {};
45
- const experimentProps = {};
46
- return {
47
- ...gatesProps,
48
- ...experimentProps,
49
- ...supplementalDefaultProps,
50
- app_id: "commerce-chat-react-component",
51
- chat_id: getItem(require_dist.LocalStorageKeys.ChatId),
52
- env: env || "unknown",
53
- app_source: contextSource,
54
- "org.short_name": orgShortName,
55
- "user.id": userId,
56
- "cdp.user_id": null,
57
- "cdp.provider": null,
58
- "event.source": "web-browser",
59
- "event.type": "user-activity",
60
- "event.id": null,
61
- "event.channel": "web",
62
- "event.timestamp": null
63
- };
64
- }, [
65
- featureGates,
66
- supplementalDefaultProps,
67
- env,
68
- contextSource,
69
- orgShortName,
70
- userId
71
- ]);
72
- const eventPropsToPrefixedEventProps = (0, react.useCallback)((eventName, eventProps) => {
73
- const prefix = eventName.toLowerCase().replace(/\s+/g, "_");
74
- return Object.entries(eventProps).reduce((acc, [key, value$1]) => {
75
- acc[`${prefix}.${key}`] = value$1;
76
- return acc;
77
- }, {});
78
- }, []);
79
- const amplitudeSessionReplayInit = (0, react.useCallback)(() => {
80
- const isEnabled = Boolean(orgShortName === require_dist.OrgShortName.UniqueVintage && featureFlagService?.isClientSessionEnabled() && featureFlagService?.isFeatureGateEnabled(require_dist.FeatureGates.IsNewFeatureEnabled));
81
- const sampleRate = 1;
82
- try {
83
- require_logger.logger_default.logDebug(`[spiffy-ai] amplitude session-replay initializing isEnabled=${isEnabled} sampleRate=${sampleRate}`);
84
- if (!isEnabled) return isEnabled;
85
- return isEnabled;
86
- } catch (e) {
87
- require_logger.logger_default.logError("[spiffy-ai] Error initializing amplitude session-replay", e);
88
- return false;
89
- }
90
- }, [orgShortName, featureFlagService]);
91
- const getEventTrackingEnrichment = (0, react.useCallback)(() => {
92
- if (internalEventTrackingEnrichment !== void 0) return internalEventTrackingEnrichment;
93
- const enrichment = {
94
- name: "page-view-tracking-enrichment",
95
- type: "enrichment",
96
- setup: async () => void 0,
97
- execute: async (event) => {
98
- let enrichedEvent;
99
- if (["[Amplitude] Page Viewed", `[Spiffy] ${require_dist.SpiffyMetricsEventName.BundleLoaded}`].includes(event.event_type)) {
100
- const globalProperties = {};
101
- if (publicKey) globalProperties["globalProperties.publicKey"] = publicKey;
102
- if (featureOverrides) Object.entries(featureOverrides).forEach(([key, value$1]) => {
103
- globalProperties[`globalProperties.featureOverrides.${key}`] = String(value$1);
104
- });
105
- if (variantUrlOverride) globalProperties["globalProperties.variantUrlOverride"] = variantUrlOverride;
106
- if (variantInfoOverride) globalProperties["globalProperties.variantInfoOverride"] = JSON.stringify(variantInfoOverride);
107
- if (show != null) globalProperties["globalProperties.show"] = String(show);
108
- const enabledFeatures = featureFlagService.getFeatureFlags();
109
- const enabledFeaturesProperties = Object.entries(enabledFeatures).reduce((acc, [key, value$1]) => ({
110
- ...acc,
111
- [`enabledFeatures.${key}`]: `${value$1}`
112
- }), {});
113
- const timingProperties = { "timing.enriched_at_ms": window.performance?.now() };
114
- enrichedEvent = {
115
- ...event,
116
- event_properties: {
117
- ...event.event_properties,
118
- ...getDefaultTrackingProps(),
119
- ...globalProperties,
120
- ...enabledFeaturesProperties,
121
- ...timingProperties
122
- }
123
- };
124
- } else enrichedEvent = event;
125
- require_events.EventsDispatcher.dispatch(require_dist.SpiffyEvent.AMPLITUDE_EVENT, enrichedEvent);
126
- return enrichedEvent;
127
- }
128
- };
129
- setInternalEventTrackingEnrichment(enrichment);
130
- return enrichment;
131
- }, [
132
- internalEventTrackingEnrichment,
133
- getDefaultTrackingProps,
134
- featureFlagService,
135
- publicKey,
136
- featureOverrides,
137
- variantUrlOverride,
138
- variantInfoOverride,
139
- show
140
- ]);
141
- (0, react.useEffect)(() => {
142
- if (isReady && !amplitudeClient) {
143
- const currentAmplitudeInstance = (0, __amplitude_analytics_browser.createInstance)();
144
- const isSessionsEnabled = amplitudeSessionReplayInit();
145
- currentAmplitudeInstance.add(getEventTrackingEnrichment());
146
- currentAmplitudeInstance.init(amplitudeApiKey, userId, {
147
- serverZone: dataResidency,
148
- trackingOptions: { ipAddress: true },
149
- autocapture: {
150
- attribution: true,
151
- pageViews: { trackHistoryChanges: "pathOnly" },
152
- sessions: isSessionsEnabled,
153
- formInteractions: false,
154
- fileDownloads: false
155
- }
156
- });
157
- setAmplitudeClient(currentAmplitudeInstance);
158
- }
159
- }, [
160
- isReady,
161
- amplitudeClient,
162
- amplitudeApiKey,
163
- userId,
164
- dataResidency,
165
- amplitudeSessionReplayInit,
166
- getEventTrackingEnrichment
167
- ]);
168
- const trackEvent = (0, react.useCallback)(async ({ eventName, eventProps, eventGroups, alsoSendToGoogleAnalytics = false }) => {
169
- require_logger.logger_default.logDebug("Submitting event", eventName);
170
- try {
171
- const decoratedEventName = `[Spiffy] ${eventName}`;
172
- if (!amplitudeClient) {
173
- require_logger.logger_default.logWarn("amplitude client undefined", void 0, { event_name: decoratedEventName });
174
- return;
175
- }
176
- const eventData = JSON.stringify({
177
- eventName,
178
- eventProps,
179
- created_at: (/* @__PURE__ */ new Date()).toISOString()
180
- });
181
- const data = new TextEncoder().encode(eventData);
182
- const hashBuffer = await crypto.subtle.digest("SHA-256", data);
183
- const currentInsertId = Array.from(new Uint8Array(hashBuffer)).map((b) => b.toString(16).padStart(2, "0")).join("");
184
- require_logger.logger_default.logDebug(`amplitude tracking ${decoratedEventName}`, null, {
185
- event_name: decoratedEventName,
186
- props: eventProps
187
- });
188
- amplitudeClient.track(decoratedEventName, {
189
- ...getDefaultTrackingProps(),
190
- ...eventProps,
191
- ...eventProps ? eventPropsToPrefixedEventProps(eventName, eventProps) : {}
192
- }, {
193
- ...eventGroups,
194
- insert_id: currentInsertId
195
- });
196
- if (alsoSendToGoogleAnalytics && orgGaConfig) {
197
- require_logger.logger_default.logDebug("[spiffy-ai] GA tracking", decoratedEventName);
198
- if (window.dataLayer) window.dataLayer.push({
199
- event: decoratedEventName,
200
- eventProps
201
- });
202
- }
203
- } catch (err) {
204
- require_logger.logger_default.logError("[spiffy-ai] Error tracking event", err, {
205
- eventName,
206
- eventProps
207
- });
208
- }
209
- }, [
210
- amplitudeClient,
211
- getDefaultTrackingProps,
212
- eventPropsToPrefixedEventProps,
213
- orgGaConfig
214
- ]);
215
- const value = (0, react.useMemo)(() => ({
216
- trackEvent,
217
- isReady,
218
- setSupplementalDefaultProps: (props) => setSupplementalDefaultProps(props)
219
- }), [
220
- trackEvent,
221
- isReady,
222
- setSupplementalDefaultProps
223
- ]);
224
- return /* @__PURE__ */ (0, react_jsx_runtime.jsx)(AmplitudeContext.Provider, {
225
- value,
226
- children
227
- });
228
- };
229
- const useAmplitude = () => {
230
- const context = (0, react.useContext)(AmplitudeContext);
231
- if (!context) throw new Error("useAmplitude must be used within AmplitudeProvider");
232
- return context;
233
- };
234
-
235
- //#endregion
236
- Object.defineProperty(exports, 'AmplitudeProvider', {
237
- enumerable: true,
238
- get: function () {
239
- return AmplitudeProvider;
240
- }
241
- });
242
- Object.defineProperty(exports, 'useAmplitude', {
243
- enumerable: true,
244
- get: function () {
245
- return useAmplitude;
246
- }
247
- });
248
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYW1wbGl0dWRlQ29udGV4dC1CTTZjdFIzby5janMiLCJuYW1lcyI6WyJBbXBsaXR1ZGVQcm92aWRlcjogUmVhY3QuRkM8eyBjaGlsZHJlbjogUmVhY3QuUmVhY3ROb2RlIH0+IiwidXNlcklkQXRvbSIsImFtcGxpdHVkZUFwaUtleUF0b20iLCJkYXRhUmVzaWRlbmN5QXRvbSIsIm9yZ0FuYWx5dGljc0dvb2dsZUFuYWx5dGljc0NvbmZpZ0F0b20iLCJlbnZBdG9tIiwiY29udGV4dFNvdXJjZUF0b20iLCJpZGVudGlmeWluZ1ByZWZpeEF0b20iLCJ1c2VMb2NhbFN0b3JhZ2UiLCJ1c2VFbnZpdmVDb25maWciLCJ1c2VGZWF0dXJlRmxhZ1NlcnZpY2UiLCJSZWFjdCIsIkxvY2FsU3RvcmFnZUtleXMiLCJ2YWx1ZSIsIk9yZ1Nob3J0TmFtZSIsIkZlYXR1cmVHYXRlcyIsImVucmljaG1lbnQ6IEVucmljaG1lbnRQbHVnaW4iLCJlbnJpY2hlZEV2ZW50OiBFdmVudCIsIlNwaWZmeU1ldHJpY3NFdmVudE5hbWUiLCJnbG9iYWxQcm9wZXJ0aWVzOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+IiwiU3BpZmZ5RXZlbnQiLCJjdXJyZW50QW1wbGl0dWRlSW5zdGFuY2U6IEJyb3dzZXJDbGllbnQiXSwic291cmNlcyI6WyIuLi9zcmMvY29udGV4dHMvYW1wbGl0dWRlQ29udGV4dC9hbXBsaXR1ZGVDb250ZXh0LnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QsIHtcbiAgY3JlYXRlQ29udGV4dCxcbiAgdXNlQ29udGV4dCxcbiAgdXNlQ2FsbGJhY2ssXG4gIHVzZU1lbW8sXG4gIHVzZUVmZmVjdCxcbn0gZnJvbSBcInJlYWN0XCI7XG5pbXBvcnQgeyB1c2VBdG9tVmFsdWUgfSBmcm9tIFwiam90YWlcIjtcbmltcG9ydCB7IGNyZWF0ZUluc3RhbmNlIH0gZnJvbSBcIkBhbXBsaXR1ZGUvYW5hbHl0aWNzLWJyb3dzZXJcIjtcbmltcG9ydCB7XG4gIEZlYXR1cmVHYXRlcyxcbiAgT3JnU2hvcnROYW1lLFxuICBMb2NhbFN0b3JhZ2VLZXlzLFxuICBTcGlmZnlFdmVudCxcbiAgQW1wbGl0dWRlQ29udGV4dFR5cGUsXG4gIFNwaWZmeU1ldHJpY3NFdmVudE5hbWUsXG4gIFRyYWNrRXZlbnRQYXJhbXMsXG59IGZyb20gXCJAZW52aXZlLWFpL3R5cGVzXCI7XG5pbXBvcnQge1xuICBhbXBsaXR1ZGVBcGlLZXlBdG9tLFxuICBjb250ZXh0U291cmNlQXRvbSxcbiAgZGF0YVJlc2lkZW5jeUF0b20sXG4gIGlkZW50aWZ5aW5nUHJlZml4QXRvbSxcbn0gZnJvbSBcInNyYy9hdG9tcy9lbnZpdmUvZW52aXZlQ29uZmlnXCI7XG5pbXBvcnQgeyB1c2VMb2NhbFN0b3JhZ2UgfSBmcm9tIFwic3JjL2NvbnRleHRzL2xvY2FsU3RvcmFnZUNvbnRleHRcIjtcbmltcG9ydCB7IG9yZ0FuYWx5dGljc0dvb2dsZUFuYWx5dGljc0NvbmZpZ0F0b20gfSBmcm9tIFwic3JjL2F0b21zL29yZy9vcmdBbmFseXRpY3NDb25maWdcIjtcbmltcG9ydCB7IHVzZXJJZEF0b20gfSBmcm9tIFwic3JjL2F0b21zL2FwcFwiO1xuaW1wb3J0IHsgZW52QXRvbSB9IGZyb20gXCJzcmMvYXRvbXMvZW52aXZlL2Vudml2ZUNvbmZpZ1wiO1xuaW1wb3J0IExvZ2dlciBmcm9tIFwic3JjL2FwcGxpY2F0aW9uL2xvZ2dpbmcvbG9nZ2VyXCI7XG5pbXBvcnQgdHlwZSB7XG4gIEJyb3dzZXJDbGllbnQsXG4gIEVucmljaG1lbnRQbHVnaW4sXG4gIEV2ZW50LFxuICBTZXJ2ZXJab25lVHlwZSxcbn0gZnJvbSBcIkBhbXBsaXR1ZGUvYW5hbHl0aWNzLXR5cGVzXCI7XG5pbXBvcnQgeyB1c2VFbnZpdmVDb25maWcgfSBmcm9tIFwic3JjL2NvbnRleHRzL2Vudml2ZUNvbmZpZ0NvbnRleHQvZW52aXZlQ29uZmlnQ29udGV4dFwiO1xuaW1wb3J0IHsgdXNlRmVhdHVyZUZsYWdTZXJ2aWNlIH0gZnJvbSBcInNyYy9jb250ZXh0cy9mZWF0dXJlRmxhZ1NlcnZpY2VDb250ZXh0L2ZlYXR1cmVGbGFnU2VydmljZUNvbnRleHRcIjtcbmltcG9ydCB7IEV2ZW50c0Rpc3BhdGNoZXIgfSBmcm9tIFwic3JjL2V2ZW50c1wiO1xuXG5jb25zdCBBbXBsaXR1ZGVDb250ZXh0ID0gY3JlYXRlQ29udGV4dDxBbXBsaXR1ZGVDb250ZXh0VHlwZSB8IG51bGw+KG51bGwpO1xuXG5leHBvcnQgY29uc3QgQW1wbGl0dWRlUHJvdmlkZXI6IFJlYWN0LkZDPHsgY2hpbGRyZW46IFJlYWN0LlJlYWN0Tm9kZSB9PiA9ICh7XG4gIGNoaWxkcmVuLFxufSkgPT4ge1xuICBjb25zdCB1c2VySWQgPSB1c2VBdG9tVmFsdWUodXNlcklkQXRvbSk7XG4gIGNvbnN0IGFtcGxpdHVkZUFwaUtleSA9IHVzZUF0b21WYWx1ZShhbXBsaXR1ZGVBcGlLZXlBdG9tKTtcbiAgY29uc3QgZGF0YVJlc2lkZW5jeSA9IHVzZUF0b21WYWx1ZShkYXRhUmVzaWRlbmN5QXRvbSk7XG4gIGNvbnN0IG9yZ0dhQ29uZmlnID0gdXNlQXRvbVZhbHVlKG9yZ0FuYWx5dGljc0dvb2dsZUFuYWx5dGljc0NvbmZpZ0F0b20pO1xuICBjb25zdCBlbnYgPSB1c2VBdG9tVmFsdWUoZW52QXRvbSk7XG4gIGNvbnN0IGNvbnRleHRTb3VyY2UgPSB1c2VBdG9tVmFsdWUoY29udGV4dFNvdXJjZUF0b20pO1xuICBjb25zdCBpZGVudGlmeWluZ1ByZWZpeCA9IHVzZUF0b21WYWx1ZShpZGVudGlmeWluZ1ByZWZpeEF0b20pO1xuICBjb25zdCB7IGdldEl0ZW0gfSA9IHVzZUxvY2FsU3RvcmFnZSgpO1xuICBjb25zdCB7XG4gICAgcHVibGljS2V5LFxuICAgIGZlYXR1cmVPdmVycmlkZXMsXG4gICAgdmFyaWFudFVybE92ZXJyaWRlLFxuICAgIHZhcmlhbnRJbmZvT3ZlcnJpZGUsXG4gICAgc2hvdyxcbiAgICBvcmdTaG9ydE5hbWUsXG4gICAgZmVhdHVyZUdhdGVzLFxuICB9ID0gdXNlRW52aXZlQ29uZmlnKCk7XG5cbiAgY29uc3QgeyBmZWF0dXJlRmxhZ1NlcnZpY2UgfSA9IHVzZUZlYXR1cmVGbGFnU2VydmljZSgpO1xuXG4gIGNvbnN0IFthbXBsaXR1ZGVDbGllbnQsIHNldEFtcGxpdHVkZUNsaWVudF0gPSBSZWFjdC51c2VTdGF0ZTxcbiAgICBCcm93c2VyQ2xpZW50IHwgdW5kZWZpbmVkXG4gID4odW5kZWZpbmVkKTtcbiAgY29uc3QgW2ludGVybmFsRXZlbnRUcmFja2luZ0VucmljaG1lbnQsIHNldEludGVybmFsRXZlbnRUcmFja2luZ0VucmljaG1lbnRdID1cbiAgICBSZWFjdC51c2VTdGF0ZTxFbnJpY2htZW50UGx1Z2luIHwgdW5kZWZpbmVkPih1bmRlZmluZWQpO1xuICBjb25zdCBbc3VwcGxlbWVudGFsRGVmYXVsdFByb3BzLCBzZXRTdXBwbGVtZW50YWxEZWZhdWx0UHJvcHNdID1cbiAgICBSZWFjdC51c2VTdGF0ZTxSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4oe30pO1xuXG4gIGNvbnN0IGlzUmVhZHkgPSBCb29sZWFuKFxuICAgIHVzZXJJZCAmJiBmZWF0dXJlRmxhZ1NlcnZpY2UgJiYgYW1wbGl0dWRlQXBpS2V5ICYmIHVzZXJJZFxuICApO1xuXG4gIGNvbnN0IGdldERlZmF1bHRUcmFja2luZ1Byb3BzID0gdXNlQ2FsbGJhY2soKCk6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgICBjb25zdCBnYXRlc1Byb3BzID0gZmVhdHVyZUdhdGVzXG4gICAgICA/IGZlYXR1cmVHYXRlcy5yZWR1Y2U8UmVjb3JkPHN0cmluZywgYm9vbGVhbj4+KChhY2MsIGN1cnIpID0+IHtcbiAgICAgICAgICBpZiAoY3Vyci5uYW1lICYmIGN1cnIudmFsdWUgIT0gbnVsbCkge1xuICAgICAgICAgICAgcmV0dXJuIHsgLi4uYWNjLCBbYGZlYXR1cmVfZ2F0ZS4ke2N1cnIubmFtZX1gXTogY3Vyci52YWx1ZSB9O1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm4gYWNjO1xuICAgICAgICB9LCB7fSlcbiAgICAgIDoge307XG4gICAgY29uc3QgZXhwZXJpbWVudFByb3BzID0ge307IC8vIE5vIGRpcmVjdCBlcXVpdmFsZW50IGZvciBleHBlcmltZW50cyBpbiBFbnZpdmVDb25maWcgeWV0XG5cbiAgICBjb25zdCBvcmdMZXZlbEFtcGxpdHVkZVRyYWNraW5nUHJvcHMgPSB7XG4gICAgICAuLi5nYXRlc1Byb3BzLFxuICAgICAgLi4uZXhwZXJpbWVudFByb3BzLFxuICAgIH07XG5cbiAgICByZXR1cm4ge1xuICAgICAgLi4ub3JnTGV2ZWxBbXBsaXR1ZGVUcmFja2luZ1Byb3BzLFxuICAgICAgLi4uc3VwcGxlbWVudGFsRGVmYXVsdFByb3BzLFxuICAgICAgLy8gVE9ETzogb3JnX2lkIGlzIG5vdCBkaXJlY3RseSBhdmFpbGFibGUgaW4gRW52aXZlQ29uZmlnLiBOZWVkIHRvIGZpbmQgYSBuZXcgc291cmNlIG9yIGRlcml2ZSBpdC5cbiAgICAgIC8vIG9yZ19pZDogb3JnQ29uZmlnPy5vcmc/Lm9yZz8uaWQsXG4gICAgICBhcHBfaWQ6IFwiY29tbWVyY2UtY2hhdC1yZWFjdC1jb21wb25lbnRcIixcbiAgICAgIGNoYXRfaWQ6IGdldEl0ZW0oTG9jYWxTdG9yYWdlS2V5cy5DaGF0SWQpLFxuICAgICAgZW52OiBlbnYgfHwgXCJ1bmtub3duXCIsXG4gICAgICBhcHBfc291cmNlOiBjb250ZXh0U291cmNlLFxuICAgICAgXCJvcmcuc2hvcnRfbmFtZVwiOiBvcmdTaG9ydE5hbWUsXG4gICAgICBcInVzZXIuaWRcIjogdXNlcklkLFxuICAgICAgXCJjZHAudXNlcl9pZFwiOiBudWxsLFxuICAgICAgXCJjZHAucHJvdmlkZXJcIjogbnVsbCxcbiAgICAgIFwiZXZlbnQuc291cmNlXCI6IFwid2ViLWJyb3dzZXJcIixcbiAgICAgIFwiZXZlbnQudHlwZVwiOiBcInVzZXItYWN0aXZpdHlcIixcbiAgICAgIFwiZXZlbnQuaWRcIjogbnVsbCxcbiAgICAgIFwiZXZlbnQuY2hhbm5lbFwiOiBcIndlYlwiLFxuICAgICAgXCJldmVudC50aW1lc3RhbXBcIjogbnVsbCxcbiAgICB9O1xuICB9LCBbXG4gICAgZmVhdHVyZUdhdGVzLFxuICAgIHN1cHBsZW1lbnRhbERlZmF1bHRQcm9wcyxcbiAgICBlbnYsXG4gICAgY29udGV4dFNvdXJjZSxcbiAgICBvcmdTaG9ydE5hbWUsXG4gICAgdXNlcklkLFxuICBdKTtcblxuICBjb25zdCBldmVudFByb3BzVG9QcmVmaXhlZEV2ZW50UHJvcHMgPSB1c2VDYWxsYmFjayhcbiAgICAoXG4gICAgICBldmVudE5hbWU6IFNwaWZmeU1ldHJpY3NFdmVudE5hbWUsXG4gICAgICBldmVudFByb3BzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPlxuICAgICk6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgICAgIGNvbnN0IHByZWZpeCA9IGV2ZW50TmFtZS50b0xvd2VyQ2FzZSgpLnJlcGxhY2UoL1xccysvZywgXCJfXCIpO1xuICAgICAgcmV0dXJuIE9iamVjdC5lbnRyaWVzKGV2ZW50UHJvcHMpLnJlZHVjZSgoYWNjLCBba2V5LCB2YWx1ZV0pID0+IHtcbiAgICAgICAgYWNjW2Ake3ByZWZpeH0uJHtrZXl9YF0gPSB2YWx1ZTtcbiAgICAgICAgcmV0dXJuIGFjYztcbiAgICAgIH0sIHt9IGFzIFJlY29yZDxzdHJpbmcsIHVua25vd24+KTtcbiAgICB9LFxuICAgIFtdXG4gICk7XG5cbiAgY29uc3QgYW1wbGl0dWRlU2Vzc2lvblJlcGxheUluaXQgPSB1c2VDYWxsYmFjaygoKTogYm9vbGVhbiA9PiB7XG4gICAgY29uc3QgaXNFbmFibGVkID0gQm9vbGVhbihcbiAgICAgIG9yZ1Nob3J0TmFtZSA9PT0gT3JnU2hvcnROYW1lLlVuaXF1ZVZpbnRhZ2UgJiZcbiAgICAgICAgZmVhdHVyZUZsYWdTZXJ2aWNlPy5pc0NsaWVudFNlc3Npb25FbmFibGVkKCkgJiZcbiAgICAgICAgZmVhdHVyZUZsYWdTZXJ2aWNlPy5pc0ZlYXR1cmVHYXRlRW5hYmxlZChcbiAgICAgICAgICBGZWF0dXJlR2F0ZXMuSXNOZXdGZWF0dXJlRW5hYmxlZFxuICAgICAgICApXG4gICAgKTtcbiAgICBjb25zdCBzYW1wbGVSYXRlID0gMTtcblxuICAgIHRyeSB7XG4gICAgICBMb2dnZXIubG9nRGVidWcoXG4gICAgICAgIGBbc3BpZmZ5LWFpXSBhbXBsaXR1ZGUgc2Vzc2lvbi1yZXBsYXkgaW5pdGlhbGl6aW5nIGlzRW5hYmxlZD0ke2lzRW5hYmxlZH0gc2FtcGxlUmF0ZT0ke3NhbXBsZVJhdGV9YFxuICAgICAgKTtcblxuICAgICAgaWYgKCFpc0VuYWJsZWQpIHtcbiAgICAgICAgcmV0dXJuIGlzRW5hYmxlZDtcbiAgICAgIH1cblxuICAgICAgLy8gYW1wbGl0dWRlSW5zdGFuY2UuYWRkKHNlc3Npb25SZXBsYXlQbHVnaW4oeyBzYW1wbGVSYXRlIH0pKTtcblxuICAgICAgcmV0dXJuIGlzRW5hYmxlZDtcbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICBMb2dnZXIubG9nRXJyb3IoXG4gICAgICAgIFwiW3NwaWZmeS1haV0gRXJyb3IgaW5pdGlhbGl6aW5nIGFtcGxpdHVkZSBzZXNzaW9uLXJlcGxheVwiLFxuICAgICAgICBlXG4gICAgICApO1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cbiAgfSwgW29yZ1Nob3J0TmFtZSwgZmVhdHVyZUZsYWdTZXJ2aWNlXSk7XG5cbiAgY29uc3QgZ2V0RXZlbnRUcmFja2luZ0VucmljaG1lbnQgPSB1c2VDYWxsYmFjaygoKTogRW5yaWNobWVudFBsdWdpbiA9PiB7XG4gICAgaWYgKGludGVybmFsRXZlbnRUcmFja2luZ0VucmljaG1lbnQgIT09IHVuZGVmaW5lZCkge1xuICAgICAgcmV0dXJuIGludGVybmFsRXZlbnRUcmFja2luZ0VucmljaG1lbnQ7XG4gICAgfVxuXG4gICAgY29uc3QgZW5yaWNobWVudDogRW5yaWNobWVudFBsdWdpbiA9IHtcbiAgICAgIG5hbWU6IFwicGFnZS12aWV3LXRyYWNraW5nLWVucmljaG1lbnRcIixcbiAgICAgIHR5cGU6IFwiZW5yaWNobWVudFwiLFxuICAgICAgc2V0dXA6IGFzeW5jICgpID0+IHVuZGVmaW5lZCxcbiAgICAgIGV4ZWN1dGU6IGFzeW5jIChldmVudDogRXZlbnQpOiBQcm9taXNlPEV2ZW50PiA9PiB7XG4gICAgICAgIGxldCBlbnJpY2hlZEV2ZW50OiBFdmVudDtcblxuICAgICAgICBjb25zdCBldmVudHNUb0VucmljaCA9IFtcbiAgICAgICAgICBcIltBbXBsaXR1ZGVdIFBhZ2UgVmlld2VkXCIsXG4gICAgICAgICAgYFtTcGlmZnldICR7U3BpZmZ5TWV0cmljc0V2ZW50TmFtZS5CdW5kbGVMb2FkZWR9YCxcbiAgICAgICAgXTtcblxuICAgICAgICBpZiAoZXZlbnRzVG9FbnJpY2guaW5jbHVkZXMoZXZlbnQuZXZlbnRfdHlwZSkpIHtcbiAgICAgICAgICBjb25zdCBnbG9iYWxQcm9wZXJ0aWVzOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0ge307XG5cbiAgICAgICAgICBpZiAocHVibGljS2V5KSB7XG4gICAgICAgICAgICBnbG9iYWxQcm9wZXJ0aWVzW1wiZ2xvYmFsUHJvcGVydGllcy5wdWJsaWNLZXlcIl0gPSBwdWJsaWNLZXk7XG4gICAgICAgICAgfVxuICAgICAgICAgIGlmIChmZWF0dXJlT3ZlcnJpZGVzKSB7XG4gICAgICAgICAgICBPYmplY3QuZW50cmllcyhmZWF0dXJlT3ZlcnJpZGVzKS5mb3JFYWNoKChba2V5LCB2YWx1ZV0pID0+IHtcbiAgICAgICAgICAgICAgZ2xvYmFsUHJvcGVydGllc1tgZ2xvYmFsUHJvcGVydGllcy5mZWF0dXJlT3ZlcnJpZGVzLiR7a2V5fWBdID1cbiAgICAgICAgICAgICAgICBTdHJpbmcodmFsdWUpO1xuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfVxuICAgICAgICAgIGlmICh2YXJpYW50VXJsT3ZlcnJpZGUpIHtcbiAgICAgICAgICAgIGdsb2JhbFByb3BlcnRpZXNbXCJnbG9iYWxQcm9wZXJ0aWVzLnZhcmlhbnRVcmxPdmVycmlkZVwiXSA9XG4gICAgICAgICAgICAgIHZhcmlhbnRVcmxPdmVycmlkZTtcbiAgICAgICAgICB9XG4gICAgICAgICAgaWYgKHZhcmlhbnRJbmZvT3ZlcnJpZGUpIHtcbiAgICAgICAgICAgIGdsb2JhbFByb3BlcnRpZXNbXCJnbG9iYWxQcm9wZXJ0aWVzLnZhcmlhbnRJbmZvT3ZlcnJpZGVcIl0gPVxuICAgICAgICAgICAgICBKU09OLnN0cmluZ2lmeSh2YXJpYW50SW5mb092ZXJyaWRlKTtcbiAgICAgICAgICB9XG4gICAgICAgICAgaWYgKHNob3cgIT0gbnVsbCkge1xuICAgICAgICAgICAgZ2xvYmFsUHJvcGVydGllc1tcImdsb2JhbFByb3BlcnRpZXMuc2hvd1wiXSA9IFN0cmluZyhzaG93KTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBjb25zdCBlbmFibGVkRmVhdHVyZXMgPSBmZWF0dXJlRmxhZ1NlcnZpY2UhLmdldEZlYXR1cmVGbGFncygpO1xuICAgICAgICAgIGNvbnN0IGVuYWJsZWRGZWF0dXJlc1Byb3BlcnRpZXMgPSBPYmplY3QuZW50cmllcyhcbiAgICAgICAgICAgIGVuYWJsZWRGZWF0dXJlc1xuICAgICAgICAgICkucmVkdWNlPFJlY29yZDxzdHJpbmcsIHN0cmluZz4+KFxuICAgICAgICAgICAgKGFjYywgW2tleSwgdmFsdWVdKSA9PiAoe1xuICAgICAgICAgICAgICAuLi5hY2MsXG4gICAgICAgICAgICAgIFtgZW5hYmxlZEZlYXR1cmVzLiR7a2V5fWBdOiBgJHt2YWx1ZX1gLFxuICAgICAgICAgICAgfSksXG4gICAgICAgICAgICB7fVxuICAgICAgICAgICk7XG5cbiAgICAgICAgICBjb25zdCB0aW1pbmdQcm9wZXJ0aWVzID0ge1xuICAgICAgICAgICAgXCJ0aW1pbmcuZW5yaWNoZWRfYXRfbXNcIjogd2luZG93LnBlcmZvcm1hbmNlPy5ub3coKSxcbiAgICAgICAgICB9O1xuXG4gICAgICAgICAgZW5yaWNoZWRFdmVudCA9IHtcbiAgICAgICAgICAgIC4uLmV2ZW50LFxuICAgICAgICAgICAgZXZlbnRfcHJvcGVydGllczoge1xuICAgICAgICAgICAgICAuLi5ldmVudC5ldmVudF9wcm9wZXJ0aWVzLFxuICAgICAgICAgICAgICAuLi5nZXREZWZhdWx0VHJhY2tpbmdQcm9wcygpLFxuICAgICAgICAgICAgICAuLi5nbG9iYWxQcm9wZXJ0aWVzLFxuICAgICAgICAgICAgICAuLi5lbmFibGVkRmVhdHVyZXNQcm9wZXJ0aWVzLFxuICAgICAgICAgICAgICAuLi50aW1pbmdQcm9wZXJ0aWVzLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICB9O1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGVucmljaGVkRXZlbnQgPSBldmVudDtcbiAgICAgICAgfVxuXG4gICAgICAgIEV2ZW50c0Rpc3BhdGNoZXIuZGlzcGF0Y2goU3BpZmZ5RXZlbnQuQU1QTElUVURFX0VWRU5ULCBlbnJpY2hlZEV2ZW50KTtcblxuICAgICAgICByZXR1cm4gZW5yaWNoZWRFdmVudDtcbiAgICAgIH0sXG4gICAgfTtcbiAgICBzZXRJbnRlcm5hbEV2ZW50VHJhY2tpbmdFbnJpY2htZW50KGVucmljaG1lbnQpO1xuICAgIHJldHVybiBlbnJpY2htZW50O1xuICB9LCBbXG4gICAgaW50ZXJuYWxFdmVudFRyYWNraW5nRW5yaWNobWVudCxcbiAgICBnZXREZWZhdWx0VHJhY2tpbmdQcm9wcyxcbiAgICBmZWF0dXJlRmxhZ1NlcnZpY2UsXG4gICAgcHVibGljS2V5LFxuICAgIGZlYXR1cmVPdmVycmlkZXMsXG4gICAgdmFyaWFudFVybE92ZXJyaWRlLFxuICAgIHZhcmlhbnRJbmZvT3ZlcnJpZGUsXG4gICAgc2hvdyxcbiAgXSk7XG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoaXNSZWFkeSAmJiAhYW1wbGl0dWRlQ2xpZW50KSB7XG4gICAgICBjb25zdCBjdXJyZW50QW1wbGl0dWRlSW5zdGFuY2U6IEJyb3dzZXJDbGllbnQgPSBjcmVhdGVJbnN0YW5jZSgpO1xuICAgICAgY29uc3QgaXNTZXNzaW9uc0VuYWJsZWQgPSBhbXBsaXR1ZGVTZXNzaW9uUmVwbGF5SW5pdCgpO1xuICAgICAgY3VycmVudEFtcGxpdHVkZUluc3RhbmNlLmFkZChnZXRFdmVudFRyYWNraW5nRW5yaWNobWVudCgpKTtcbiAgICAgIGN1cnJlbnRBbXBsaXR1ZGVJbnN0YW5jZS5pbml0KGFtcGxpdHVkZUFwaUtleSEsIHVzZXJJZCEsIHtcbiAgICAgICAgc2VydmVyWm9uZTogZGF0YVJlc2lkZW5jeSBhcyBTZXJ2ZXJab25lVHlwZSxcbiAgICAgICAgdHJhY2tpbmdPcHRpb25zOiB7XG4gICAgICAgICAgaXBBZGRyZXNzOiB0cnVlLFxuICAgICAgICB9LFxuICAgICAgICBhdXRvY2FwdHVyZToge1xuICAgICAgICAgIGF0dHJpYnV0aW9uOiB0cnVlLFxuICAgICAgICAgIHBhZ2VWaWV3czoge1xuICAgICAgICAgICAgdHJhY2tIaXN0b3J5Q2hhbmdlczogXCJwYXRoT25seVwiLFxuICAgICAgICAgIH0sXG4gICAgICAgICAgc2Vzc2lvbnM6IGlzU2Vzc2lvbnNFbmFibGVkLFxuICAgICAgICAgIGZvcm1JbnRlcmFjdGlvbnM6IGZhbHNlLFxuICAgICAgICAgIGZpbGVEb3dubG9hZHM6IGZhbHNlLFxuICAgICAgICB9LFxuICAgICAgfSk7XG4gICAgICBzZXRBbXBsaXR1ZGVDbGllbnQoY3VycmVudEFtcGxpdHVkZUluc3RhbmNlKTtcbiAgICB9XG4gIH0sIFtcbiAgICBpc1JlYWR5LFxuICAgIGFtcGxpdHVkZUNsaWVudCxcbiAgICBhbXBsaXR1ZGVBcGlLZXksXG4gICAgdXNlcklkLFxuICAgIGRhdGFSZXNpZGVuY3ksXG4gICAgYW1wbGl0dWRlU2Vzc2lvblJlcGxheUluaXQsXG4gICAgZ2V0RXZlbnRUcmFja2luZ0VucmljaG1lbnQsXG4gIF0pO1xuXG4gIGNvbnN0IHRyYWNrRXZlbnQgPSB1c2VDYWxsYmFjayhcbiAgICBhc3luYyAoe1xuICAgICAgZXZlbnROYW1lLFxuICAgICAgZXZlbnRQcm9wcyxcbiAgICAgIGV2ZW50R3JvdXBzLFxuICAgICAgYWxzb1NlbmRUb0dvb2dsZUFuYWx5dGljcyA9IGZhbHNlLFxuICAgIH06IFRyYWNrRXZlbnRQYXJhbXMpOiBQcm9taXNlPHZvaWQ+ID0+IHtcbiAgICAgIExvZ2dlci5sb2dEZWJ1ZyhcIlN1Ym1pdHRpbmcgZXZlbnRcIiwgZXZlbnROYW1lKTtcbiAgICAgIHRyeSB7XG4gICAgICAgIGNvbnN0IGRlY29yYXRlZEV2ZW50TmFtZSA9IGBbU3BpZmZ5XSAke2V2ZW50TmFtZX1gO1xuXG4gICAgICAgIGlmICghYW1wbGl0dWRlQ2xpZW50KSB7XG4gICAgICAgICAgTG9nZ2VyLmxvZ1dhcm4oXCJhbXBsaXR1ZGUgY2xpZW50IHVuZGVmaW5lZFwiLCB1bmRlZmluZWQsIHtcbiAgICAgICAgICAgIGV2ZW50X25hbWU6IGRlY29yYXRlZEV2ZW50TmFtZSxcbiAgICAgICAgICB9KTtcbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBldmVudERhdGEgPSBKU09OLnN0cmluZ2lmeSh7XG4gICAgICAgICAgZXZlbnROYW1lLFxuICAgICAgICAgIGV2ZW50UHJvcHMsXG4gICAgICAgICAgY3JlYXRlZF9hdDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgICAgICB9KTtcbiAgICAgICAgY29uc3QgZW5jb2RlciA9IG5ldyBUZXh0RW5jb2RlcigpO1xuICAgICAgICBjb25zdCBkYXRhID0gZW5jb2Rlci5lbmNvZGUoZXZlbnREYXRhKTtcbiAgICAgICAgLy8gY2FsY3VsYXRlIGEgaGFzaCBvZiB0aGUgZXZlbnQgcHJvcGVydGllcyB0byB1c2UgYXMgdGhlIGluc2VydF9pZCBzbyB0aGF0IGR1cGxpY2F0ZSBldmVudHNcbiAgICAgICAgLy8gYXJlIGF1dG9tYXRpY2FsbHkgZHJvcHBlZCBieSBBbXBsaXR1ZGVcbiAgICAgICAgY29uc3QgaGFzaEJ1ZmZlciA9IGF3YWl0IGNyeXB0by5zdWJ0bGUuZGlnZXN0KFwiU0hBLTI1NlwiLCBkYXRhKTtcbiAgICAgICAgY29uc3QgaGFzaEFycmF5ID0gQXJyYXkuZnJvbShuZXcgVWludDhBcnJheShoYXNoQnVmZmVyKSk7XG4gICAgICAgIGNvbnN0IGN1cnJlbnRJbnNlcnRJZCA9IGhhc2hBcnJheVxuICAgICAgICAgIC5tYXAoKGIpID0+IGIudG9TdHJpbmcoMTYpLnBhZFN0YXJ0KDIsIFwiMFwiKSlcbiAgICAgICAgICAuam9pbihcIlwiKTtcblxuICAgICAgICBMb2dnZXIubG9nRGVidWcoYGFtcGxpdHVkZSB0cmFja2luZyAke2RlY29yYXRlZEV2ZW50TmFtZX1gLCBudWxsLCB7XG4gICAgICAgICAgZXZlbnRfbmFtZTogZGVjb3JhdGVkRXZlbnROYW1lLFxuICAgICAgICAgIHByb3BzOiBldmVudFByb3BzLFxuICAgICAgICB9KTtcblxuICAgICAgICBhbXBsaXR1ZGVDbGllbnQudHJhY2soXG4gICAgICAgICAgZGVjb3JhdGVkRXZlbnROYW1lLFxuICAgICAgICAgIHtcbiAgICAgICAgICAgIC4uLmdldERlZmF1bHRUcmFja2luZ1Byb3BzKCksXG4gICAgICAgICAgICAuLi5ldmVudFByb3BzLFxuICAgICAgICAgICAgLi4uKGV2ZW50UHJvcHNcbiAgICAgICAgICAgICAgPyBldmVudFByb3BzVG9QcmVmaXhlZEV2ZW50UHJvcHMoZXZlbnROYW1lLCBldmVudFByb3BzKVxuICAgICAgICAgICAgICA6IHt9KSxcbiAgICAgICAgICB9LFxuICAgICAgICAgIHtcbiAgICAgICAgICAgIC4uLmV2ZW50R3JvdXBzLFxuICAgICAgICAgICAgaW5zZXJ0X2lkOiBjdXJyZW50SW5zZXJ0SWQsXG4gICAgICAgICAgfVxuICAgICAgICApO1xuXG4gICAgICAgIGlmIChhbHNvU2VuZFRvR29vZ2xlQW5hbHl0aWNzICYmIG9yZ0dhQ29uZmlnKSB7XG4gICAgICAgICAgLy8gVE9ETzogQWRkIGluIHdpbmRvd0RhdGFMYXllclNlcnZpY2Ugb3IgY29udGV4dCBhbHRlcm5hdGl2ZSBhbmQgaG9vayBpdCB1cCBoZXJlXG4gICAgICAgICAgTG9nZ2VyLmxvZ0RlYnVnKFwiW3NwaWZmeS1haV0gR0EgdHJhY2tpbmdcIiwgZGVjb3JhdGVkRXZlbnROYW1lKTtcbiAgICAgICAgICBpZiAod2luZG93LmRhdGFMYXllcikge1xuICAgICAgICAgICAgKHdpbmRvdy5kYXRhTGF5ZXIgYXMgYW55W10pLnB1c2goe1xuICAgICAgICAgICAgICBldmVudDogZGVjb3JhdGVkRXZlbnROYW1lLFxuICAgICAgICAgICAgICBldmVudFByb3BzOiBldmVudFByb3BzLFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9IGNhdGNoIChlcnIpIHtcbiAgICAgICAgTG9nZ2VyLmxvZ0Vycm9yKFwiW3NwaWZmeS1haV0gRXJyb3IgdHJhY2tpbmcgZXZlbnRcIiwgZXJyLCB7XG4gICAgICAgICAgZXZlbnROYW1lLFxuICAgICAgICAgIGV2ZW50UHJvcHMsXG4gICAgICAgIH0pO1xuICAgICAgfVxuICAgIH0sXG4gICAgW1xuICAgICAgYW1wbGl0dWRlQ2xpZW50LFxuICAgICAgZ2V0RGVmYXVsdFRyYWNraW5nUHJvcHMsXG4gICAgICBldmVudFByb3BzVG9QcmVmaXhlZEV2ZW50UHJvcHMsXG4gICAgICBvcmdHYUNvbmZpZyxcbiAgICBdXG4gICk7XG5cbiAgY29uc3QgdmFsdWUgPSB1c2VNZW1vKFxuICAgICgpID0+ICh7XG4gICAgICB0cmFja0V2ZW50LFxuICAgICAgaXNSZWFkeSxcbiAgICAgIHNldFN1cHBsZW1lbnRhbERlZmF1bHRQcm9wczogKHByb3BzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPikgPT5cbiAgICAgICAgc2V0U3VwcGxlbWVudGFsRGVmYXVsdFByb3BzKHByb3BzKSxcbiAgICB9KSxcbiAgICBbdHJhY2tFdmVudCwgaXNSZWFkeSwgc2V0U3VwcGxlbWVudGFsRGVmYXVsdFByb3BzXVxuICApO1xuXG4gIHJldHVybiAoXG4gICAgPEFtcGxpdHVkZUNvbnRleHQuUHJvdmlkZXIgdmFsdWU9e3ZhbHVlfT5cbiAgICAgIHtjaGlsZHJlbn1cbiAgICA8L0FtcGxpdHVkZUNvbnRleHQuUHJvdmlkZXI+XG4gICk7XG59O1xuXG5leHBvcnQgY29uc3QgdXNlQW1wbGl0dWRlID0gKCkgPT4ge1xuICBjb25zdCBjb250ZXh0ID0gdXNlQ29udGV4dChBbXBsaXR1ZGVDb250ZXh0KTtcbiAgaWYgKCFjb250ZXh0KSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKFwidXNlQW1wbGl0dWRlIG11c3QgYmUgdXNlZCB3aXRoaW4gQW1wbGl0dWRlUHJvdmlkZXJcIik7XG4gIH1cbiAgcmV0dXJuIGNvbnRleHQ7XG59O1xuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQXVDQSxNQUFNLDRDQUE4RCxLQUFLO0FBRXpFLE1BQWFBLHFCQUE4RCxFQUN6RSxlQUNJO0NBQ0osTUFBTSxpQ0FBc0JDLHVCQUFXO0NBQ3ZDLE1BQU0sMENBQStCQyx5Q0FBb0I7Q0FDekQsTUFBTSx3Q0FBNkJDLHVDQUFrQjtDQUNyRCxNQUFNLHNDQUEyQkMsa0RBQXNDO0NBQ3ZFLE1BQU0sOEJBQW1CQyw2QkFBUTtDQUNqQyxNQUFNLHdDQUE2QkMsdUNBQWtCO0FBQzNCLHlCQUFhQywyQ0FBc0I7Q0FDN0QsTUFBTSxFQUFFLFlBQVlDLDZDQUFpQjtDQUNyQyxNQUFNLEVBQ0osV0FDQSxrQkFDQSxvQkFDQSxxQkFDQSxNQUNBLGNBQ0EsaUJBQ0VDLDZDQUFpQjtDQUVyQixNQUFNLEVBQUUsdUJBQXVCQyx5REFBdUI7Q0FFdEQsTUFBTSxDQUFDLGlCQUFpQixzQkFBc0JDLGNBQU0sU0FFbEQsT0FBVTtDQUNaLE1BQU0sQ0FBQyxpQ0FBaUMsc0NBQ3RDQSxjQUFNLFNBQXVDLE9BQVU7Q0FDekQsTUFBTSxDQUFDLDBCQUEwQiwrQkFDL0JBLGNBQU0sU0FBa0MsRUFBRSxDQUFDO0NBRTdDLE1BQU0sVUFBVSxRQUNkLFVBQVUsc0JBQXNCLG1CQUFtQixPQUNwRDtDQUVELE1BQU0sdURBQXFFO0VBQ3pFLE1BQU0sYUFBYSxlQUNm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
@@ -1,232 +0,0 @@
1
- import { EventsDispatcher } from "./events-WOOrnUAx.js";
2
- import { FeatureGates, LocalStorageKeys, OrgShortName, SpiffyEvent, SpiffyMetricsEventName } from "./dist-C38adNK1.js";
3
- import { logger_default } from "./logger-BMVdhQOV.js";
4
- import { amplitudeApiKeyAtom, contextSourceAtom, dataResidencyAtom, envAtom, identifyingPrefixAtom } from "./enviveConfig-Cwlp7-Qz.js";
5
- import { orgAnalyticsGoogleAnalyticsConfigAtom } from "./org-CScQqL4k.js";
6
- import { userIdAtom } from "./app-BT3g-mDp.js";
7
- import { useLocalStorage } from "./localStorageContext-BFwvuEcf.js";
8
- import { useEnviveConfig } from "./enviveConfigContext-De7lbzWa.js";
9
- import { useFeatureFlagService } from "./featureFlagServiceContext-Cvp7NlpC.js";
10
- import React, { createContext, useCallback, useContext, useEffect, useMemo } from "react";
11
- import { useAtomValue } from "jotai";
12
- import { jsx } from "react/jsx-runtime";
13
- import { createInstance } from "@amplitude/analytics-browser";
14
-
15
- //#region src/contexts/amplitudeContext/amplitudeContext.tsx
16
- const AmplitudeContext = createContext(null);
17
- const AmplitudeProvider = ({ children }) => {
18
- const userId = useAtomValue(userIdAtom);
19
- const amplitudeApiKey = useAtomValue(amplitudeApiKeyAtom);
20
- const dataResidency = useAtomValue(dataResidencyAtom);
21
- const orgGaConfig = useAtomValue(orgAnalyticsGoogleAnalyticsConfigAtom);
22
- const env = useAtomValue(envAtom);
23
- const contextSource = useAtomValue(contextSourceAtom);
24
- useAtomValue(identifyingPrefixAtom);
25
- const { getItem } = useLocalStorage();
26
- const { publicKey, featureOverrides, variantUrlOverride, variantInfoOverride, show, orgShortName, featureGates } = useEnviveConfig();
27
- const { featureFlagService } = useFeatureFlagService();
28
- const [amplitudeClient, setAmplitudeClient] = React.useState(void 0);
29
- const [internalEventTrackingEnrichment, setInternalEventTrackingEnrichment] = React.useState(void 0);
30
- const [supplementalDefaultProps, setSupplementalDefaultProps] = React.useState({});
31
- const isReady = Boolean(userId && featureFlagService && amplitudeApiKey && userId);
32
- const getDefaultTrackingProps = useCallback(() => {
33
- const gatesProps = featureGates ? featureGates.reduce((acc, curr) => {
34
- if (curr.name && curr.value != null) return {
35
- ...acc,
36
- [`feature_gate.${curr.name}`]: curr.value
37
- };
38
- return acc;
39
- }, {}) : {};
40
- const experimentProps = {};
41
- return {
42
- ...gatesProps,
43
- ...experimentProps,
44
- ...supplementalDefaultProps,
45
- app_id: "commerce-chat-react-component",
46
- chat_id: getItem(LocalStorageKeys.ChatId),
47
- env: env || "unknown",
48
- app_source: contextSource,
49
- "org.short_name": orgShortName,
50
- "user.id": userId,
51
- "cdp.user_id": null,
52
- "cdp.provider": null,
53
- "event.source": "web-browser",
54
- "event.type": "user-activity",
55
- "event.id": null,
56
- "event.channel": "web",
57
- "event.timestamp": null
58
- };
59
- }, [
60
- featureGates,
61
- supplementalDefaultProps,
62
- env,
63
- contextSource,
64
- orgShortName,
65
- userId
66
- ]);
67
- const eventPropsToPrefixedEventProps = useCallback((eventName, eventProps) => {
68
- const prefix = eventName.toLowerCase().replace(/\s+/g, "_");
69
- return Object.entries(eventProps).reduce((acc, [key, value$1]) => {
70
- acc[`${prefix}.${key}`] = value$1;
71
- return acc;
72
- }, {});
73
- }, []);
74
- const amplitudeSessionReplayInit = useCallback(() => {
75
- const isEnabled = Boolean(orgShortName === OrgShortName.UniqueVintage && featureFlagService?.isClientSessionEnabled() && featureFlagService?.isFeatureGateEnabled(FeatureGates.IsNewFeatureEnabled));
76
- const sampleRate = 1;
77
- try {
78
- logger_default.logDebug(`[spiffy-ai] amplitude session-replay initializing isEnabled=${isEnabled} sampleRate=${sampleRate}`);
79
- if (!isEnabled) return isEnabled;
80
- return isEnabled;
81
- } catch (e) {
82
- logger_default.logError("[spiffy-ai] Error initializing amplitude session-replay", e);
83
- return false;
84
- }
85
- }, [orgShortName, featureFlagService]);
86
- const getEventTrackingEnrichment = useCallback(() => {
87
- if (internalEventTrackingEnrichment !== void 0) return internalEventTrackingEnrichment;
88
- const enrichment = {
89
- name: "page-view-tracking-enrichment",
90
- type: "enrichment",
91
- setup: async () => void 0,
92
- execute: async (event) => {
93
- let enrichedEvent;
94
- if (["[Amplitude] Page Viewed", `[Spiffy] ${SpiffyMetricsEventName.BundleLoaded}`].includes(event.event_type)) {
95
- const globalProperties = {};
96
- if (publicKey) globalProperties["globalProperties.publicKey"] = publicKey;
97
- if (featureOverrides) Object.entries(featureOverrides).forEach(([key, value$1]) => {
98
- globalProperties[`globalProperties.featureOverrides.${key}`] = String(value$1);
99
- });
100
- if (variantUrlOverride) globalProperties["globalProperties.variantUrlOverride"] = variantUrlOverride;
101
- if (variantInfoOverride) globalProperties["globalProperties.variantInfoOverride"] = JSON.stringify(variantInfoOverride);
102
- if (show != null) globalProperties["globalProperties.show"] = String(show);
103
- const enabledFeatures = featureFlagService.getFeatureFlags();
104
- const enabledFeaturesProperties = Object.entries(enabledFeatures).reduce((acc, [key, value$1]) => ({
105
- ...acc,
106
- [`enabledFeatures.${key}`]: `${value$1}`
107
- }), {});
108
- const timingProperties = { "timing.enriched_at_ms": window.performance?.now() };
109
- enrichedEvent = {
110
- ...event,
111
- event_properties: {
112
- ...event.event_properties,
113
- ...getDefaultTrackingProps(),
114
- ...globalProperties,
115
- ...enabledFeaturesProperties,
116
- ...timingProperties
117
- }
118
- };
119
- } else enrichedEvent = event;
120
- EventsDispatcher.dispatch(SpiffyEvent.AMPLITUDE_EVENT, enrichedEvent);
121
- return enrichedEvent;
122
- }
123
- };
124
- setInternalEventTrackingEnrichment(enrichment);
125
- return enrichment;
126
- }, [
127
- internalEventTrackingEnrichment,
128
- getDefaultTrackingProps,
129
- featureFlagService,
130
- publicKey,
131
- featureOverrides,
132
- variantUrlOverride,
133
- variantInfoOverride,
134
- show
135
- ]);
136
- useEffect(() => {
137
- if (isReady && !amplitudeClient) {
138
- const currentAmplitudeInstance = createInstance();
139
- const isSessionsEnabled = amplitudeSessionReplayInit();
140
- currentAmplitudeInstance.add(getEventTrackingEnrichment());
141
- currentAmplitudeInstance.init(amplitudeApiKey, userId, {
142
- serverZone: dataResidency,
143
- trackingOptions: { ipAddress: true },
144
- autocapture: {
145
- attribution: true,
146
- pageViews: { trackHistoryChanges: "pathOnly" },
147
- sessions: isSessionsEnabled,
148
- formInteractions: false,
149
- fileDownloads: false
150
- }
151
- });
152
- setAmplitudeClient(currentAmplitudeInstance);
153
- }
154
- }, [
155
- isReady,
156
- amplitudeClient,
157
- amplitudeApiKey,
158
- userId,
159
- dataResidency,
160
- amplitudeSessionReplayInit,
161
- getEventTrackingEnrichment
162
- ]);
163
- const trackEvent = useCallback(async ({ eventName, eventProps, eventGroups, alsoSendToGoogleAnalytics = false }) => {
164
- logger_default.logDebug("Submitting event", eventName);
165
- try {
166
- const decoratedEventName = `[Spiffy] ${eventName}`;
167
- if (!amplitudeClient) {
168
- logger_default.logWarn("amplitude client undefined", void 0, { event_name: decoratedEventName });
169
- return;
170
- }
171
- const eventData = JSON.stringify({
172
- eventName,
173
- eventProps,
174
- created_at: (/* @__PURE__ */ new Date()).toISOString()
175
- });
176
- const data = new TextEncoder().encode(eventData);
177
- const hashBuffer = await crypto.subtle.digest("SHA-256", data);
178
- const currentInsertId = Array.from(new Uint8Array(hashBuffer)).map((b) => b.toString(16).padStart(2, "0")).join("");
179
- logger_default.logDebug(`amplitude tracking ${decoratedEventName}`, null, {
180
- event_name: decoratedEventName,
181
- props: eventProps
182
- });
183
- amplitudeClient.track(decoratedEventName, {
184
- ...getDefaultTrackingProps(),
185
- ...eventProps,
186
- ...eventProps ? eventPropsToPrefixedEventProps(eventName, eventProps) : {}
187
- }, {
188
- ...eventGroups,
189
- insert_id: currentInsertId
190
- });
191
- if (alsoSendToGoogleAnalytics && orgGaConfig) {
192
- logger_default.logDebug("[spiffy-ai] GA tracking", decoratedEventName);
193
- if (window.dataLayer) window.dataLayer.push({
194
- event: decoratedEventName,
195
- eventProps
196
- });
197
- }
198
- } catch (err) {
199
- logger_default.logError("[spiffy-ai] Error tracking event", err, {
200
- eventName,
201
- eventProps
202
- });
203
- }
204
- }, [
205
- amplitudeClient,
206
- getDefaultTrackingProps,
207
- eventPropsToPrefixedEventProps,
208
- orgGaConfig
209
- ]);
210
- const value = useMemo(() => ({
211
- trackEvent,
212
- isReady,
213
- setSupplementalDefaultProps: (props) => setSupplementalDefaultProps(props)
214
- }), [
215
- trackEvent,
216
- isReady,
217
- setSupplementalDefaultProps
218
- ]);
219
- return /* @__PURE__ */ jsx(AmplitudeContext.Provider, {
220
- value,
221
- children
222
- });
223
- };
224
- const useAmplitude = () => {
225
- const context = useContext(AmplitudeContext);
226
- if (!context) throw new Error("useAmplitude must be used within AmplitudeProvider");
227
- return context;
228
- };
229
-
230
- //#endregion
231
- export { AmplitudeProvider, useAmplitude };
232
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYW1wbGl0dWRlQ29udGV4dC1ESEN0ZmNlNS5qcyIsIm5hbWVzIjpbIkFtcGxpdHVkZVByb3ZpZGVyOiBSZWFjdC5GQzx7IGNoaWxkcmVuOiBSZWFjdC5SZWFjdE5vZGUgfT4iLCJ2YWx1ZSIsImVucmljaG1lbnQ6IEVucmljaG1lbnRQbHVnaW4iLCJlbnJpY2hlZEV2ZW50OiBFdmVudCIsImdsb2JhbFByb3BlcnRpZXM6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4iLCJjdXJyZW50QW1wbGl0dWRlSW5zdGFuY2U6IEJyb3dzZXJDbGllbnQiXSwic291cmNlcyI6WyIuLi9zcmMvY29udGV4dHMvYW1wbGl0dWRlQ29udGV4dC9hbXBsaXR1ZGVDb250ZXh0LnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QsIHtcbiAgY3JlYXRlQ29udGV4dCxcbiAgdXNlQ29udGV4dCxcbiAgdXNlQ2FsbGJhY2ssXG4gIHVzZU1lbW8sXG4gIHVzZUVmZmVjdCxcbn0gZnJvbSBcInJlYWN0XCI7XG5pbXBvcnQgeyB1c2VBdG9tVmFsdWUgfSBmcm9tIFwiam90YWlcIjtcbmltcG9ydCB7IGNyZWF0ZUluc3RhbmNlIH0gZnJvbSBcIkBhbXBsaXR1ZGUvYW5hbHl0aWNzLWJyb3dzZXJcIjtcbmltcG9ydCB7XG4gIEZlYXR1cmVHYXRlcyxcbiAgT3JnU2hvcnROYW1lLFxuICBMb2NhbFN0b3JhZ2VLZXlzLFxuICBTcGlmZnlFdmVudCxcbiAgQW1wbGl0dWRlQ29udGV4dFR5cGUsXG4gIFNwaWZmeU1ldHJpY3NFdmVudE5hbWUsXG4gIFRyYWNrRXZlbnRQYXJhbXMsXG59IGZyb20gXCJAZW52aXZlLWFpL3R5cGVzXCI7XG5pbXBvcnQge1xuICBhbXBsaXR1ZGVBcGlLZXlBdG9tLFxuICBjb250ZXh0U291cmNlQXRvbSxcbiAgZGF0YVJlc2lkZW5jeUF0b20sXG4gIGlkZW50aWZ5aW5nUHJlZml4QXRvbSxcbn0gZnJvbSBcInNyYy9hdG9tcy9lbnZpdmUvZW52aXZlQ29uZmlnXCI7XG5pbXBvcnQgeyB1c2VMb2NhbFN0b3JhZ2UgfSBmcm9tIFwic3JjL2NvbnRleHRzL2xvY2FsU3RvcmFnZUNvbnRleHRcIjtcbmltcG9ydCB7IG9yZ0FuYWx5dGljc0dvb2dsZUFuYWx5dGljc0NvbmZpZ0F0b20gfSBmcm9tIFwic3JjL2F0b21zL29yZy9vcmdBbmFseXRpY3NDb25maWdcIjtcbmltcG9ydCB7IHVzZXJJZEF0b20gfSBmcm9tIFwic3JjL2F0b21zL2FwcFwiO1xuaW1wb3J0IHsgZW52QXRvbSB9IGZyb20gXCJzcmMvYXRvbXMvZW52aXZlL2Vudml2ZUNvbmZpZ1wiO1xuaW1wb3J0IExvZ2dlciBmcm9tIFwic3JjL2FwcGxpY2F0aW9uL2xvZ2dpbmcvbG9nZ2VyXCI7XG5pbXBvcnQgdHlwZSB7XG4gIEJyb3dzZXJDbGllbnQsXG4gIEVucmljaG1lbnRQbHVnaW4sXG4gIEV2ZW50LFxuICBTZXJ2ZXJab25lVHlwZSxcbn0gZnJvbSBcIkBhbXBsaXR1ZGUvYW5hbHl0aWNzLXR5cGVzXCI7XG5pbXBvcnQgeyB1c2VFbnZpdmVDb25maWcgfSBmcm9tIFwic3JjL2NvbnRleHRzL2Vudml2ZUNvbmZpZ0NvbnRleHQvZW52aXZlQ29uZmlnQ29udGV4dFwiO1xuaW1wb3J0IHsgdXNlRmVhdHVyZUZsYWdTZXJ2aWNlIH0gZnJvbSBcInNyYy9jb250ZXh0cy9mZWF0dXJlRmxhZ1NlcnZpY2VDb250ZXh0L2ZlYXR1cmVGbGFnU2VydmljZUNvbnRleHRcIjtcbmltcG9ydCB7IEV2ZW50c0Rpc3BhdGNoZXIgfSBmcm9tIFwic3JjL2V2ZW50c1wiO1xuXG5jb25zdCBBbXBsaXR1ZGVDb250ZXh0ID0gY3JlYXRlQ29udGV4dDxBbXBsaXR1ZGVDb250ZXh0VHlwZSB8IG51bGw+KG51bGwpO1xuXG5leHBvcnQgY29uc3QgQW1wbGl0dWRlUHJvdmlkZXI6IFJlYWN0LkZDPHsgY2hpbGRyZW46IFJlYWN0LlJlYWN0Tm9kZSB9PiA9ICh7XG4gIGNoaWxkcmVuLFxufSkgPT4ge1xuICBjb25zdCB1c2VySWQgPSB1c2VBdG9tVmFsdWUodXNlcklkQXRvbSk7XG4gIGNvbnN0IGFtcGxpdHVkZUFwaUtleSA9IHVzZUF0b21WYWx1ZShhbXBsaXR1ZGVBcGlLZXlBdG9tKTtcbiAgY29uc3QgZGF0YVJlc2lkZW5jeSA9IHVzZUF0b21WYWx1ZShkYXRhUmVzaWRlbmN5QXRvbSk7XG4gIGNvbnN0IG9yZ0dhQ29uZmlnID0gdXNlQXRvbVZhbHVlKG9yZ0FuYWx5dGljc0dvb2dsZUFuYWx5dGljc0NvbmZpZ0F0b20pO1xuICBjb25zdCBlbnYgPSB1c2VBdG9tVmFsdWUoZW52QXRvbSk7XG4gIGNvbnN0IGNvbnRleHRTb3VyY2UgPSB1c2VBdG9tVmFsdWUoY29udGV4dFNvdXJjZUF0b20pO1xuICBjb25zdCBpZGVudGlmeWluZ1ByZWZpeCA9IHVzZUF0b21WYWx1ZShpZGVudGlmeWluZ1ByZWZpeEF0b20pO1xuICBjb25zdCB7IGdldEl0ZW0gfSA9IHVzZUxvY2FsU3RvcmFnZSgpO1xuICBjb25zdCB7XG4gICAgcHVibGljS2V5LFxuICAgIGZlYXR1cmVPdmVycmlkZXMsXG4gICAgdmFyaWFudFVybE92ZXJyaWRlLFxuICAgIHZhcmlhbnRJbmZvT3ZlcnJpZGUsXG4gICAgc2hvdyxcbiAgICBvcmdTaG9ydE5hbWUsXG4gICAgZmVhdHVyZUdhdGVzLFxuICB9ID0gdXNlRW52aXZlQ29uZmlnKCk7XG5cbiAgY29uc3QgeyBmZWF0dXJlRmxhZ1NlcnZpY2UgfSA9IHVzZUZlYXR1cmVGbGFnU2VydmljZSgpO1xuXG4gIGNvbnN0IFthbXBsaXR1ZGVDbGllbnQsIHNldEFtcGxpdHVkZUNsaWVudF0gPSBSZWFjdC51c2VTdGF0ZTxcbiAgICBCcm93c2VyQ2xpZW50IHwgdW5kZWZpbmVkXG4gID4odW5kZWZpbmVkKTtcbiAgY29uc3QgW2ludGVybmFsRXZlbnRUcmFja2luZ0VucmljaG1lbnQsIHNldEludGVybmFsRXZlbnRUcmFja2luZ0VucmljaG1lbnRdID1cbiAgICBSZWFjdC51c2VTdGF0ZTxFbnJpY2htZW50UGx1Z2luIHwgdW5kZWZpbmVkPih1bmRlZmluZWQpO1xuICBjb25zdCBbc3VwcGxlbWVudGFsRGVmYXVsdFByb3BzLCBzZXRTdXBwbGVtZW50YWxEZWZhdWx0UHJvcHNdID1cbiAgICBSZWFjdC51c2VTdGF0ZTxSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4oe30pO1xuXG4gIGNvbnN0IGlzUmVhZHkgPSBCb29sZWFuKFxuICAgIHVzZXJJZCAmJiBmZWF0dXJlRmxhZ1NlcnZpY2UgJiYgYW1wbGl0dWRlQXBpS2V5ICYmIHVzZXJJZFxuICApO1xuXG4gIGNvbnN0IGdldERlZmF1bHRUcmFja2luZ1Byb3BzID0gdXNlQ2FsbGJhY2soKCk6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgICBjb25zdCBnYXRlc1Byb3BzID0gZmVhdHVyZUdhdGVzXG4gICAgICA/IGZlYXR1cmVHYXRlcy5yZWR1Y2U8UmVjb3JkPHN0cmluZywgYm9vbGVhbj4+KChhY2MsIGN1cnIpID0+IHtcbiAgICAgICAgICBpZiAoY3Vyci5uYW1lICYmIGN1cnIudmFsdWUgIT0gbnVsbCkge1xuICAgICAgICAgICAgcmV0dXJuIHsgLi4uYWNjLCBbYGZlYXR1cmVfZ2F0ZS4ke2N1cnIubmFtZX1gXTogY3Vyci52YWx1ZSB9O1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm4gYWNjO1xuICAgICAgICB9LCB7fSlcbiAgICAgIDoge307XG4gICAgY29uc3QgZXhwZXJpbWVudFByb3BzID0ge307IC8vIE5vIGRpcmVjdCBlcXVpdmFsZW50IGZvciBleHBlcmltZW50cyBpbiBFbnZpdmVDb25maWcgeWV0XG5cbiAgICBjb25zdCBvcmdMZXZlbEFtcGxpdHVkZVRyYWNraW5nUHJvcHMgPSB7XG4gICAgICAuLi5nYXRlc1Byb3BzLFxuICAgICAgLi4uZXhwZXJpbWVudFByb3BzLFxuICAgIH07XG5cbiAgICByZXR1cm4ge1xuICAgICAgLi4ub3JnTGV2ZWxBbXBsaXR1ZGVUcmFja2luZ1Byb3BzLFxuICAgICAgLi4uc3VwcGxlbWVudGFsRGVmYXVsdFByb3BzLFxuICAgICAgLy8gVE9ETzogb3JnX2lkIGlzIG5vdCBkaXJlY3RseSBhdmFpbGFibGUgaW4gRW52aXZlQ29uZmlnLiBOZWVkIHRvIGZpbmQgYSBuZXcgc291cmNlIG9yIGRlcml2ZSBpdC5cbiAgICAgIC8vIG9yZ19pZDogb3JnQ29uZmlnPy5vcmc/Lm9yZz8uaWQsXG4gICAgICBhcHBfaWQ6IFwiY29tbWVyY2UtY2hhdC1yZWFjdC1jb21wb25lbnRcIixcbiAgICAgIGNoYXRfaWQ6IGdldEl0ZW0oTG9jYWxTdG9yYWdlS2V5cy5DaGF0SWQpLFxuICAgICAgZW52OiBlbnYgfHwgXCJ1bmtub3duXCIsXG4gICAgICBhcHBfc291cmNlOiBjb250ZXh0U291cmNlLFxuICAgICAgXCJvcmcuc2hvcnRfbmFtZVwiOiBvcmdTaG9ydE5hbWUsXG4gICAgICBcInVzZXIuaWRcIjogdXNlcklkLFxuICAgICAgXCJjZHAudXNlcl9pZFwiOiBudWxsLFxuICAgICAgXCJjZHAucHJvdmlkZXJcIjogbnVsbCxcbiAgICAgIFwiZXZlbnQuc291cmNlXCI6IFwid2ViLWJyb3dzZXJcIixcbiAgICAgIFwiZXZlbnQudHlwZVwiOiBcInVzZXItYWN0aXZpdHlcIixcbiAgICAgIFwiZXZlbnQuaWRcIjogbnVsbCxcbiAgICAgIFwiZXZlbnQuY2hhbm5lbFwiOiBcIndlYlwiLFxuICAgICAgXCJldmVudC50aW1lc3RhbXBcIjogbnVsbCxcbiAgICB9O1xuICB9LCBbXG4gICAgZmVhdHVyZUdhdGVzLFxuICAgIHN1cHBsZW1lbnRhbERlZmF1bHRQcm9wcyxcbiAgICBlbnYsXG4gICAgY29udGV4dFNvdXJjZSxcbiAgICBvcmdTaG9ydE5hbWUsXG4gICAgdXNlcklkLFxuICBdKTtcblxuICBjb25zdCBldmVudFByb3BzVG9QcmVmaXhlZEV2ZW50UHJvcHMgPSB1c2VDYWxsYmFjayhcbiAgICAoXG4gICAgICBldmVudE5hbWU6IFNwaWZmeU1ldHJpY3NFdmVudE5hbWUsXG4gICAgICBldmVudFByb3BzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPlxuICAgICk6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgICAgIGNvbnN0IHByZWZpeCA9IGV2ZW50TmFtZS50b0xvd2VyQ2FzZSgpLnJlcGxhY2UoL1xccysvZywgXCJfXCIpO1xuICAgICAgcmV0dXJuIE9iamVjdC5lbnRyaWVzKGV2ZW50UHJvcHMpLnJlZHVjZSgoYWNjLCBba2V5LCB2YWx1ZV0pID0+IHtcbiAgICAgICAgYWNjW2Ake3ByZWZpeH0uJHtrZXl9YF0gPSB2YWx1ZTtcbiAgICAgICAgcmV0dXJuIGFjYztcbiAgICAgIH0sIHt9IGFzIFJlY29yZDxzdHJpbmcsIHVua25vd24+KTtcbiAgICB9LFxuICAgIFtdXG4gICk7XG5cbiAgY29uc3QgYW1wbGl0dWRlU2Vzc2lvblJlcGxheUluaXQgPSB1c2VDYWxsYmFjaygoKTogYm9vbGVhbiA9PiB7XG4gICAgY29uc3QgaXNFbmFibGVkID0gQm9vbGVhbihcbiAgICAgIG9yZ1Nob3J0TmFtZSA9PT0gT3JnU2hvcnROYW1lLlVuaXF1ZVZpbnRhZ2UgJiZcbiAgICAgICAgZmVhdHVyZUZsYWdTZXJ2aWNlPy5pc0NsaWVudFNlc3Npb25FbmFibGVkKCkgJiZcbiAgICAgICAgZmVhdHVyZUZsYWdTZXJ2aWNlPy5pc0ZlYXR1cmVHYXRlRW5hYmxlZChcbiAgICAgICAgICBGZWF0dXJlR2F0ZXMuSXNOZXdGZWF0dXJlRW5hYmxlZFxuICAgICAgICApXG4gICAgKTtcbiAgICBjb25zdCBzYW1wbGVSYXRlID0gMTtcblxuICAgIHRyeSB7XG4gICAgICBMb2dnZXIubG9nRGVidWcoXG4gICAgICAgIGBbc3BpZmZ5LWFpXSBhbXBsaXR1ZGUgc2Vzc2lvbi1yZXBsYXkgaW5pdGlhbGl6aW5nIGlzRW5hYmxlZD0ke2lzRW5hYmxlZH0gc2FtcGxlUmF0ZT0ke3NhbXBsZVJhdGV9YFxuICAgICAgKTtcblxuICAgICAgaWYgKCFpc0VuYWJsZWQpIHtcbiAgICAgICAgcmV0dXJuIGlzRW5hYmxlZDtcbiAgICAgIH1cblxuICAgICAgLy8gYW1wbGl0dWRlSW5zdGFuY2UuYWRkKHNlc3Npb25SZXBsYXlQbHVnaW4oeyBzYW1wbGVSYXRlIH0pKTtcblxuICAgICAgcmV0dXJuIGlzRW5hYmxlZDtcbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICBMb2dnZXIubG9nRXJyb3IoXG4gICAgICAgIFwiW3NwaWZmeS1haV0gRXJyb3IgaW5pdGlhbGl6aW5nIGFtcGxpdHVkZSBzZXNzaW9uLXJlcGxheVwiLFxuICAgICAgICBlXG4gICAgICApO1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cbiAgfSwgW29yZ1Nob3J0TmFtZSwgZmVhdHVyZUZsYWdTZXJ2aWNlXSk7XG5cbiAgY29uc3QgZ2V0RXZlbnRUcmFja2luZ0VucmljaG1lbnQgPSB1c2VDYWxsYmFjaygoKTogRW5yaWNobWVudFBsdWdpbiA9PiB7XG4gICAgaWYgKGludGVybmFsRXZlbnRUcmFja2luZ0VucmljaG1lbnQgIT09IHVuZGVmaW5lZCkge1xuICAgICAgcmV0dXJuIGludGVybmFsRXZlbnRUcmFja2luZ0VucmljaG1lbnQ7XG4gICAgfVxuXG4gICAgY29uc3QgZW5yaWNobWVudDogRW5yaWNobWVudFBsdWdpbiA9IHtcbiAgICAgIG5hbWU6IFwicGFnZS12aWV3LXRyYWNraW5nLWVucmljaG1lbnRcIixcbiAgICAgIHR5cGU6IFwiZW5yaWNobWVudFwiLFxuICAgICAgc2V0dXA6IGFzeW5jICgpID0+IHVuZGVmaW5lZCxcbiAgICAgIGV4ZWN1dGU6IGFzeW5jIChldmVudDogRXZlbnQpOiBQcm9taXNlPEV2ZW50PiA9PiB7XG4gICAgICAgIGxldCBlbnJpY2hlZEV2ZW50OiBFdmVudDtcblxuICAgICAgICBjb25zdCBldmVudHNUb0VucmljaCA9IFtcbiAgICAgICAgICBcIltBbXBsaXR1ZGVdIFBhZ2UgVmlld2VkXCIsXG4gICAgICAgICAgYFtTcGlmZnldICR7U3BpZmZ5TWV0cmljc0V2ZW50TmFtZS5CdW5kbGVMb2FkZWR9YCxcbiAgICAgICAgXTtcblxuICAgICAgICBpZiAoZXZlbnRzVG9FbnJpY2guaW5jbHVkZXMoZXZlbnQuZXZlbnRfdHlwZSkpIHtcbiAgICAgICAgICBjb25zdCBnbG9iYWxQcm9wZXJ0aWVzOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0ge307XG5cbiAgICAgICAgICBpZiAocHVibGljS2V5KSB7XG4gICAgICAgICAgICBnbG9iYWxQcm9wZXJ0aWVzW1wiZ2xvYmFsUHJvcGVydGllcy5wdWJsaWNLZXlcIl0gPSBwdWJsaWNLZXk7XG4gICAgICAgICAgfVxuICAgICAgICAgIGlmIChmZWF0dXJlT3ZlcnJpZGVzKSB7XG4gICAgICAgICAgICBPYmplY3QuZW50cmllcyhmZWF0dXJlT3ZlcnJpZGVzKS5mb3JFYWNoKChba2V5LCB2YWx1ZV0pID0+IHtcbiAgICAgICAgICAgICAgZ2xvYmFsUHJvcGVydGllc1tgZ2xvYmFsUHJvcGVydGllcy5mZWF0dXJlT3ZlcnJpZGVzLiR7a2V5fWBdID1cbiAgICAgICAgICAgICAgICBTdHJpbmcodmFsdWUpO1xuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfVxuICAgICAgICAgIGlmICh2YXJpYW50VXJsT3ZlcnJpZGUpIHtcbiAgICAgICAgICAgIGdsb2JhbFByb3BlcnRpZXNbXCJnbG9iYWxQcm9wZXJ0aWVzLnZhcmlhbnRVcmxPdmVycmlkZVwiXSA9XG4gICAgICAgICAgICAgIHZhcmlhbnRVcmxPdmVycmlkZTtcbiAgICAgICAgICB9XG4gICAgICAgICAgaWYgKHZhcmlhbnRJbmZvT3ZlcnJpZGUpIHtcbiAgICAgICAgICAgIGdsb2JhbFByb3BlcnRpZXNbXCJnbG9iYWxQcm9wZXJ0aWVzLnZhcmlhbnRJbmZvT3ZlcnJpZGVcIl0gPVxuICAgICAgICAgICAgICBKU09OLnN0cmluZ2lmeSh2YXJpYW50SW5mb092ZXJyaWRlKTtcbiAgICAgICAgICB9XG4gICAgICAgICAgaWYgKHNob3cgIT0gbnVsbCkge1xuICAgICAgICAgICAgZ2xvYmFsUHJvcGVydGllc1tcImdsb2JhbFByb3BlcnRpZXMuc2hvd1wiXSA9IFN0cmluZyhzaG93KTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBjb25zdCBlbmFibGVkRmVhdHVyZXMgPSBmZWF0dXJlRmxhZ1NlcnZpY2UhLmdldEZlYXR1cmVGbGFncygpO1xuICAgICAgICAgIGNvbnN0IGVuYWJsZWRGZWF0dXJlc1Byb3BlcnRpZXMgPSBPYmplY3QuZW50cmllcyhcbiAgICAgICAgICAgIGVuYWJsZWRGZWF0dXJlc1xuICAgICAgICAgICkucmVkdWNlPFJlY29yZDxzdHJpbmcsIHN0cmluZz4+KFxuICAgICAgICAgICAgKGFjYywgW2tleSwgdmFsdWVdKSA9PiAoe1xuICAgICAgICAgICAgICAuLi5hY2MsXG4gICAgICAgICAgICAgIFtgZW5hYmxlZEZlYXR1cmVzLiR7a2V5fWBdOiBgJHt2YWx1ZX1gLFxuICAgICAgICAgICAgfSksXG4gICAgICAgICAgICB7fVxuICAgICAgICAgICk7XG5cbiAgICAgICAgICBjb25zdCB0aW1pbmdQcm9wZXJ0aWVzID0ge1xuICAgICAgICAgICAgXCJ0aW1pbmcuZW5yaWNoZWRfYXRfbXNcIjogd2luZG93LnBlcmZvcm1hbmNlPy5ub3coKSxcbiAgICAgICAgICB9O1xuXG4gICAgICAgICAgZW5yaWNoZWRFdmVudCA9IHtcbiAgICAgICAgICAgIC4uLmV2ZW50LFxuICAgICAgICAgICAgZXZlbnRfcHJvcGVydGllczoge1xuICAgICAgICAgICAgICAuLi5ldmVudC5ldmVudF9wcm9wZXJ0aWVzLFxuICAgICAgICAgICAgICAuLi5nZXREZWZhdWx0VHJhY2tpbmdQcm9wcygpLFxuICAgICAgICAgICAgICAuLi5nbG9iYWxQcm9wZXJ0aWVzLFxuICAgICAgICAgICAgICAuLi5lbmFibGVkRmVhdHVyZXNQcm9wZXJ0aWVzLFxuICAgICAgICAgICAgICAuLi50aW1pbmdQcm9wZXJ0aWVzLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICB9O1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGVucmljaGVkRXZlbnQgPSBldmVudDtcbiAgICAgICAgfVxuXG4gICAgICAgIEV2ZW50c0Rpc3BhdGNoZXIuZGlzcGF0Y2goU3BpZmZ5RXZlbnQuQU1QTElUVURFX0VWRU5ULCBlbnJpY2hlZEV2ZW50KTtcblxuICAgICAgICByZXR1cm4gZW5yaWNoZWRFdmVudDtcbiAgICAgIH0sXG4gICAgfTtcbiAgICBzZXRJbnRlcm5hbEV2ZW50VHJhY2tpbmdFbnJpY2htZW50KGVucmljaG1lbnQpO1xuICAgIHJldHVybiBlbnJpY2htZW50O1xuICB9LCBbXG4gICAgaW50ZXJuYWxFdmVudFRyYWNraW5nRW5yaWNobWVudCxcbiAgICBnZXREZWZhdWx0VHJhY2tpbmdQcm9wcyxcbiAgICBmZWF0dXJlRmxhZ1NlcnZpY2UsXG4gICAgcHVibGljS2V5LFxuICAgIGZlYXR1cmVPdmVycmlkZXMsXG4gICAgdmFyaWFudFVybE92ZXJyaWRlLFxuICAgIHZhcmlhbnRJbmZvT3ZlcnJpZGUsXG4gICAgc2hvdyxcbiAgXSk7XG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoaXNSZWFkeSAmJiAhYW1wbGl0dWRlQ2xpZW50KSB7XG4gICAgICBjb25zdCBjdXJyZW50QW1wbGl0dWRlSW5zdGFuY2U6IEJyb3dzZXJDbGllbnQgPSBjcmVhdGVJbnN0YW5jZSgpO1xuICAgICAgY29uc3QgaXNTZXNzaW9uc0VuYWJsZWQgPSBhbXBsaXR1ZGVTZXNzaW9uUmVwbGF5SW5pdCgpO1xuICAgICAgY3VycmVudEFtcGxpdHVkZUluc3RhbmNlLmFkZChnZXRFdmVudFRyYWNraW5nRW5yaWNobWVudCgpKTtcbiAgICAgIGN1cnJlbnRBbXBsaXR1ZGVJbnN0YW5jZS5pbml0KGFtcGxpdHVkZUFwaUtleSEsIHVzZXJJZCEsIHtcbiAgICAgICAgc2VydmVyWm9uZTogZGF0YVJlc2lkZW5jeSBhcyBTZXJ2ZXJab25lVHlwZSxcbiAgICAgICAgdHJhY2tpbmdPcHRpb25zOiB7XG4gICAgICAgICAgaXBBZGRyZXNzOiB0cnVlLFxuICAgICAgICB9LFxuICAgICAgICBhdXRvY2FwdHVyZToge1xuICAgICAgICAgIGF0dHJpYnV0aW9uOiB0cnVlLFxuICAgICAgICAgIHBhZ2VWaWV3czoge1xuICAgICAgICAgICAgdHJhY2tIaXN0b3J5Q2hhbmdlczogXCJwYXRoT25seVwiLFxuICAgICAgICAgIH0sXG4gICAgICAgICAgc2Vzc2lvbnM6IGlzU2Vzc2lvbnNFbmFibGVkLFxuICAgICAgICAgIGZvcm1JbnRlcmFjdGlvbnM6IGZhbHNlLFxuICAgICAgICAgIGZpbGVEb3dubG9hZHM6IGZhbHNlLFxuICAgICAgICB9LFxuICAgICAgfSk7XG4gICAgICBzZXRBbXBsaXR1ZGVDbGllbnQoY3VycmVudEFtcGxpdHVkZUluc3RhbmNlKTtcbiAgICB9XG4gIH0sIFtcbiAgICBpc1JlYWR5LFxuICAgIGFtcGxpdHVkZUNsaWVudCxcbiAgICBhbXBsaXR1ZGVBcGlLZXksXG4gICAgdXNlcklkLFxuICAgIGRhdGFSZXNpZGVuY3ksXG4gICAgYW1wbGl0dWRlU2Vzc2lvblJlcGxheUluaXQsXG4gICAgZ2V0RXZlbnRUcmFja2luZ0VucmljaG1lbnQsXG4gIF0pO1xuXG4gIGNvbnN0IHRyYWNrRXZlbnQgPSB1c2VDYWxsYmFjayhcbiAgICBhc3luYyAoe1xuICAgICAgZXZlbnROYW1lLFxuICAgICAgZXZlbnRQcm9wcyxcbiAgICAgIGV2ZW50R3JvdXBzLFxuICAgICAgYWxzb1NlbmRUb0dvb2dsZUFuYWx5dGljcyA9IGZhbHNlLFxuICAgIH06IFRyYWNrRXZlbnRQYXJhbXMpOiBQcm9taXNlPHZvaWQ+ID0+IHtcbiAgICAgIExvZ2dlci5sb2dEZWJ1ZyhcIlN1Ym1pdHRpbmcgZXZlbnRcIiwgZXZlbnROYW1lKTtcbiAgICAgIHRyeSB7XG4gICAgICAgIGNvbnN0IGRlY29yYXRlZEV2ZW50TmFtZSA9IGBbU3BpZmZ5XSAke2V2ZW50TmFtZX1gO1xuXG4gICAgICAgIGlmICghYW1wbGl0dWRlQ2xpZW50KSB7XG4gICAgICAgICAgTG9nZ2VyLmxvZ1dhcm4oXCJhbXBsaXR1ZGUgY2xpZW50IHVuZGVmaW5lZFwiLCB1bmRlZmluZWQsIHtcbiAgICAgICAgICAgIGV2ZW50X25hbWU6IGRlY29yYXRlZEV2ZW50TmFtZSxcbiAgICAgICAgICB9KTtcbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBldmVudERhdGEgPSBKU09OLnN0cmluZ2lmeSh7XG4gICAgICAgICAgZXZlbnROYW1lLFxuICAgICAgICAgIGV2ZW50UHJvcHMsXG4gICAgICAgICAgY3JlYXRlZF9hdDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgICAgICB9KTtcbiAgICAgICAgY29uc3QgZW5jb2RlciA9IG5ldyBUZXh0RW5jb2RlcigpO1xuICAgICAgICBjb25zdCBkYXRhID0gZW5jb2Rlci5lbmNvZGUoZXZlbnREYXRhKTtcbiAgICAgICAgLy8gY2FsY3VsYXRlIGEgaGFzaCBvZiB0aGUgZXZlbnQgcHJvcGVydGllcyB0byB1c2UgYXMgdGhlIGluc2VydF9pZCBzbyB0aGF0IGR1cGxpY2F0ZSBldmVudHNcbiAgICAgICAgLy8gYXJlIGF1dG9tYXRpY2FsbHkgZHJvcHBlZCBieSBBbXBsaXR1ZGVcbiAgICAgICAgY29uc3QgaGFzaEJ1ZmZlciA9IGF3YWl0IGNyeXB0by5zdWJ0bGUuZGlnZXN0KFwiU0hBLTI1NlwiLCBkYXRhKTtcbiAgICAgICAgY29uc3QgaGFzaEFycmF5ID0gQXJyYXkuZnJvbShuZXcgVWludDhBcnJheShoYXNoQnVmZmVyKSk7XG4gICAgICAgIGNvbnN0IGN1cnJlbnRJbnNlcnRJZCA9IGhhc2hBcnJheVxuICAgICAgICAgIC5tYXAoKGIpID0+IGIudG9TdHJpbmcoMTYpLnBhZFN0YXJ0KDIsIFwiMFwiKSlcbiAgICAgICAgICAuam9pbihcIlwiKTtcblxuICAgICAgICBMb2dnZXIubG9nRGVidWcoYGFtcGxpdHVkZSB0cmFja2luZyAke2RlY29yYXRlZEV2ZW50TmFtZX1gLCBudWxsLCB7XG4gICAgICAgICAgZXZlbnRfbmFtZTogZGVjb3JhdGVkRXZlbnROYW1lLFxuICAgICAgICAgIHByb3BzOiBldmVudFByb3BzLFxuICAgICAgICB9KTtcblxuICAgICAgICBhbXBsaXR1ZGVDbGllbnQudHJhY2soXG4gICAgICAgICAgZGVjb3JhdGVkRXZlbnROYW1lLFxuICAgICAgICAgIHtcbiAgICAgICAgICAgIC4uLmdldERlZmF1bHRUcmFja2luZ1Byb3BzKCksXG4gICAgICAgICAgICAuLi5ldmVudFByb3BzLFxuICAgICAgICAgICAgLi4uKGV2ZW50UHJvcHNcbiAgICAgICAgICAgICAgPyBldmVudFByb3BzVG9QcmVmaXhlZEV2ZW50UHJvcHMoZXZlbnROYW1lLCBldmVudFByb3BzKVxuICAgICAgICAgICAgICA6IHt9KSxcbiAgICAgICAgICB9LFxuICAgICAgICAgIHtcbiAgICAgICAgICAgIC4uLmV2ZW50R3JvdXBzLFxuICAgICAgICAgICAgaW5zZXJ0X2lkOiBjdXJyZW50SW5zZXJ0SWQsXG4gICAgICAgICAgfVxuICAgICAgICApO1xuXG4gICAgICAgIGlmIChhbHNvU2VuZFRvR29vZ2xlQW5hbHl0aWNzICYmIG9yZ0dhQ29uZmlnKSB7XG4gICAgICAgICAgLy8gVE9ETzogQWRkIGluIHdpbmRvd0RhdGFMYXllclNlcnZpY2Ugb3IgY29udGV4dCBhbHRlcm5hdGl2ZSBhbmQgaG9vayBpdCB1cCBoZXJlXG4gICAgICAgICAgTG9nZ2VyLmxvZ0RlYnVnKFwiW3NwaWZmeS1haV0gR0EgdHJhY2tpbmdcIiwgZGVjb3JhdGVkRXZlbnROYW1lKTtcbiAgICAgICAgICBpZiAod2luZG93LmRhdGFMYXllcikge1xuICAgICAgICAgICAgKHdpbmRvdy5kYXRhTGF5ZXIgYXMgYW55W10pLnB1c2goe1xuICAgICAgICAgICAgICBldmVudDogZGVjb3JhdGVkRXZlbnROYW1lLFxuICAgICAgICAgICAgICBldmVudFByb3BzOiBldmVudFByb3BzLFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9IGNhdGNoIChlcnIpIHtcbiAgICAgICAgTG9nZ2VyLmxvZ0Vycm9yKFwiW3NwaWZmeS1haV0gRXJyb3IgdHJhY2tpbmcgZXZlbnRcIiwgZXJyLCB7XG4gICAgICAgICAgZXZlbnROYW1lLFxuICAgICAgICAgIGV2ZW50UHJvcHMsXG4gICAgICAgIH0pO1xuICAgICAgfVxuICAgIH0sXG4gICAgW1xuICAgICAgYW1wbGl0dWRlQ2xpZW50LFxuICAgICAgZ2V0RGVmYXVsdFRyYWNraW5nUHJvcHMsXG4gICAgICBldmVudFByb3BzVG9QcmVmaXhlZEV2ZW50UHJvcHMsXG4gICAgICBvcmdHYUNvbmZpZyxcbiAgICBdXG4gICk7XG5cbiAgY29uc3QgdmFsdWUgPSB1c2VNZW1vKFxuICAgICgpID0+ICh7XG4gICAgICB0cmFja0V2ZW50LFxuICAgICAgaXNSZWFkeSxcbiAgICAgIHNldFN1cHBsZW1lbnRhbERlZmF1bHRQcm9wczogKHByb3BzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPikgPT5cbiAgICAgICAgc2V0U3VwcGxlbWVudGFsRGVmYXVsdFByb3BzKHByb3BzKSxcbiAgICB9KSxcbiAgICBbdHJhY2tFdmVudCwgaXNSZWFkeSwgc2V0U3VwcGxlbWVudGFsRGVmYXVsdFByb3BzXVxuICApO1xuXG4gIHJldHVybiAoXG4gICAgPEFtcGxpdHVkZUNvbnRleHQuUHJvdmlkZXIgdmFsdWU9e3ZhbHVlfT5cbiAgICAgIHtjaGlsZHJlbn1cbiAgICA8L0FtcGxpdHVkZUNvbnRleHQuUHJvdmlkZXI+XG4gICk7XG59O1xuXG5leHBvcnQgY29uc3QgdXNlQW1wbGl0dWRlID0gKCkgPT4ge1xuICBjb25zdCBjb250ZXh0ID0gdXNlQ29udGV4dChBbXBsaXR1ZGVDb250ZXh0KTtcbiAgaWYgKCFjb250ZXh0KSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKFwidXNlQW1wbGl0dWRlIG11c3QgYmUgdXNlZCB3aXRoaW4gQW1wbGl0dWRlUHJvdmlkZXJcIik7XG4gIH1cbiAgcmV0dXJuIGNvbnRleHQ7XG59O1xuIl0sIm1hcHBpbmdzIjoiOzs7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