@envive-ai/react-hooks 0.3.19 → 0.3.21

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 (70) hide show
  1. package/dist/application/models/guards/api/isApiPDPEventAttributes.cjs +2 -2
  2. package/dist/application/models/guards/api/isApiPDPEventAttributes.js +2 -2
  3. package/dist/atoms/app/index.d.ts +1 -1
  4. package/dist/atoms/chat/chatState.d.cts +18 -18
  5. package/dist/atoms/chat/chatState.d.ts +1 -1
  6. package/dist/atoms/chat/form.d.cts +2 -2
  7. package/dist/atoms/chat/form.d.ts +2 -2
  8. package/dist/atoms/chat/index.d.cts +3 -3
  9. package/dist/atoms/chat/lastMessage.d.cts +2 -2
  10. package/dist/atoms/chat/lastMessage.d.ts +2 -2
  11. package/dist/atoms/chat/messageQueue.d.cts +6 -6
  12. package/dist/atoms/chat/messageQueue.d.ts +6 -6
  13. package/dist/atoms/chat/performanceMetrics.d.cts +6 -6
  14. package/dist/atoms/chat/performanceMetrics.d.ts +6 -6
  15. package/dist/atoms/chat/renderedWidgetRefs.d.cts +2 -2
  16. package/dist/atoms/chat/renderedWidgetRefs.d.ts +2 -2
  17. package/dist/atoms/chat/replies.d.cts +3 -3
  18. package/dist/atoms/chat/replies.d.ts +3 -3
  19. package/dist/atoms/chat/suggestions.d.cts +2 -2
  20. package/dist/atoms/chat/suggestions.d.ts +2 -2
  21. package/dist/atoms/envive/enviveConfig.d.cts +13 -13
  22. package/dist/atoms/envive/enviveConfig.d.ts +13 -13
  23. package/dist/atoms/globalSearch/globalSearch.d.cts +5 -5
  24. package/dist/atoms/globalSearch/globalSearch.d.ts +5 -5
  25. package/dist/atoms/org/customerService.d.cts +6 -6
  26. package/dist/atoms/org/customerService.d.ts +6 -6
  27. package/dist/atoms/org/graphqlConfig.d.cts +4 -4
  28. package/dist/atoms/org/graphqlConfig.d.ts +4 -4
  29. package/dist/atoms/org/newOrgConfigAtom.d.cts +2 -2
  30. package/dist/atoms/org/newOrgConfigAtom.d.ts +2 -2
  31. package/dist/atoms/org/orgAnalyticsConfig.d.cts +5 -5
  32. package/dist/atoms/org/orgAnalyticsConfig.d.ts +5 -5
  33. package/dist/atoms/search/chatSearch.d.cts +17 -17
  34. package/dist/atoms/search/chatSearch.d.ts +17 -17
  35. package/dist/atoms/search/searchAPI.d.cts +13 -13
  36. package/dist/atoms/search/searchAPI.d.ts +13 -13
  37. package/dist/atoms/search/types.d.ts +1 -1
  38. package/dist/atoms/widget/chatPreviewLoading.d.cts +2 -2
  39. package/dist/atoms/widget/chatPreviewLoading.d.ts +2 -2
  40. package/dist/contexts/enviveContext/enviveContext.cjs +41 -7
  41. package/dist/contexts/enviveContext/enviveContext.d.cts +8 -2
  42. package/dist/contexts/enviveContext/enviveContext.d.ts +8 -2
  43. package/dist/contexts/enviveContext/enviveContext.js +42 -8
  44. package/dist/contexts/enviveContext/index.d.cts +2 -2
  45. package/dist/contexts/enviveContext/index.d.ts +2 -2
  46. package/dist/contexts/enviveContext/types.d.ts +1 -1
  47. package/dist/contexts/pageContext/pageContext.cjs +78 -3
  48. package/dist/contexts/pageContext/pageContext.d.cts +3 -1
  49. package/dist/contexts/pageContext/pageContext.d.ts +3 -1
  50. package/dist/contexts/pageContext/pageContext.js +81 -6
  51. package/dist/contexts/types.d.cts +1 -1
  52. package/dist/contexts/types.d.ts +1 -1
  53. package/dist/contexts/typesV3.d.ts +1 -1
  54. package/dist/hooks/PageViewedEvent/index.cjs +4 -0
  55. package/dist/hooks/PageViewedEvent/index.d.cts +2 -0
  56. package/dist/hooks/PageViewedEvent/index.d.ts +2 -0
  57. package/dist/hooks/PageViewedEvent/index.js +3 -0
  58. package/dist/hooks/PageViewedEvent/usePageViewedEvent.cjs +84 -0
  59. package/dist/hooks/PageViewedEvent/usePageViewedEvent.d.cts +18 -0
  60. package/dist/hooks/PageViewedEvent/usePageViewedEvent.d.ts +18 -0
  61. package/dist/hooks/PageViewedEvent/usePageViewedEvent.js +82 -0
  62. package/dist/hooks/utils.d.cts +1 -1
  63. package/package.json +5 -1
  64. package/src/application/models/guards/api/isApiPDPEventAttributes.ts +1 -1
  65. package/src/contexts/enviveContext/enviveContext.tsx +71 -7
  66. package/src/contexts/pageContext/__tests__/pageContext.test.tsx +6 -0
  67. package/src/contexts/pageContext/pageContext.tsx +79 -2
  68. package/src/hooks/PageViewedEvent/__tests__/usePageViewedEvent.test.ts +297 -0
  69. package/src/hooks/PageViewedEvent/index.ts +1 -0
  70. package/src/hooks/PageViewedEvent/usePageViewedEvent.ts +103 -0
@@ -52,8 +52,21 @@ const SalesAgentWrapper = ({ children, enabledAgents, previewMode, showDebugBar,
52
52
  })
53
53
  });
54
54
  };
55
- const EnviveProvider = ({ children, previewMode = false, overrideConfig, enabledAgents, enviveConfigService: inputEnviveConfigService, featureFlagService: inputFeatureFlagService, amplitudeService: inputAmplitudeService, userIdentityService: inputUserIdentityService, mockV3ConfigToDeprecatedConfig = false, requestV3Config = false, mockSalesAgentData, hardcopyOverride, ...config }) => {
55
+ const EnviveProvider = ({ children, previewMode = false, overrideConfig, enabledAgents, enviveConfigService: inputEnviveConfigService, featureFlagService: inputFeatureFlagService, amplitudeService: inputAmplitudeService, userIdentityService: inputUserIdentityService, mockV3ConfigToDeprecatedConfig = false, requestV3Config = false, mockSalesAgentData, hardcopyOverride, extensionService, ...config }) => {
56
56
  const [userId, setUserId] = (0, react.useState)("");
57
+ const didReportSuccessRef = (0, react.useRef)(false);
58
+ const invokeExtensionCallback = (0, react.useCallback)((callback) => {
59
+ if (!callback) return;
60
+ Promise.resolve(callback()).catch((error) => {
61
+ logger.logDebug("Extension status callback failed", error);
62
+ });
63
+ }, []);
64
+ const invokeExtensionErrorCallback = (0, react.useCallback)((error) => {
65
+ if (!extensionService?.onInjectionError) return;
66
+ Promise.resolve(extensionService.onInjectionError(error)).catch((callbackError) => {
67
+ logger.logDebug("Extension error callback failed", callbackError);
68
+ });
69
+ }, [extensionService]);
57
70
  const userIdService = (0, react.useMemo)(() => inputUserIdentityService ?? new require_userIdentityService.UserIdentityService({ apiKey: config?.amplitudeApiKey || "" }), [config?.amplitudeApiKey, inputUserIdentityService]);
58
71
  (0, react.useEffect)(() => {
59
72
  const fetchUserId = async () => {
@@ -69,22 +82,42 @@ const EnviveProvider = ({ children, previewMode = false, overrideConfig, enabled
69
82
  const source = "app";
70
83
  const [enviveServiceConfig, setEnviveServiceConfig] = (0, react.useState)(null);
71
84
  (0, react.useEffect)(() => {
72
- (inputEnviveConfigService ?? new require_enviveConfigService.EnviveConfigService({
85
+ invokeExtensionCallback(extensionService?.onInjectionLoading);
86
+ const enviveConfigService = inputEnviveConfigService ?? new require_enviveConfigService.EnviveConfigService({
73
87
  baseUrl: config?.baseUrl || "",
74
88
  apiKey: config?.orgLevelApiKey || "",
75
89
  userId,
76
90
  namespace,
77
91
  source
78
- })).getEnviveConfig().then((newConfig) => {
79
- setEnviveServiceConfig(newConfig);
80
- return newConfig;
81
92
  });
93
+ const fetchEnviveConfig = async () => {
94
+ try {
95
+ setEnviveServiceConfig(await enviveConfigService.getEnviveConfig());
96
+ } catch (error) {
97
+ const resolvedError = error instanceof Error ? error : /* @__PURE__ */ new Error("Failed to load envive config");
98
+ logger.logError("Failed to initialize EnviveProvider", resolvedError);
99
+ invokeExtensionErrorCallback(resolvedError);
100
+ }
101
+ };
102
+ fetchEnviveConfig();
82
103
  }, [
83
104
  config,
84
105
  userId,
85
106
  namespace,
86
107
  source,
87
- inputEnviveConfigService
108
+ inputEnviveConfigService,
109
+ extensionService,
110
+ invokeExtensionCallback,
111
+ invokeExtensionErrorCallback
112
+ ]);
113
+ (0, react.useEffect)(() => {
114
+ if (!enviveServiceConfig || didReportSuccessRef.current) return;
115
+ didReportSuccessRef.current = true;
116
+ invokeExtensionCallback(extensionService?.onInjectionSuccess);
117
+ }, [
118
+ enviveServiceConfig,
119
+ extensionService,
120
+ invokeExtensionCallback
88
121
  ]);
89
122
  const enviveConfig = (0, react.useMemo)(() => ({
90
123
  ...config,
@@ -111,6 +144,7 @@ const EnviveProvider = ({ children, previewMode = false, overrideConfig, enabled
111
144
  externalAmplitudeService: inputAmplitudeService,
112
145
  children: /* @__PURE__ */ (0, react_jsx_runtime.jsx)(require_pageContext.PageProvider, {
113
146
  previewMode,
147
+ onUrlResolverNotReady: extensionService?.onInjectionError,
114
148
  children: /* @__PURE__ */ (0, react_jsx_runtime.jsx)(require_widgetConfigContext.WidgetConfigProvider, { children: /* @__PURE__ */ (0, react_jsx_runtime.jsx)(require_hardcopyContext.HardcopyProvider, {
115
149
  hardcopyOverride,
116
150
  children: /* @__PURE__ */ (0, react_jsx_runtime.jsx)(require_enviveCssContext.EnviveCssProvider, { children: /* @__PURE__ */ (0, react_jsx_runtime.jsxs)(SearchAgentWrapper, {
@@ -135,4 +169,4 @@ const EnviveProvider = ({ children, previewMode = false, overrideConfig, enabled
135
169
 
136
170
  //#endregion
137
171
  exports.EnviveProvider = EnviveProvider;
138
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"enviveContext.cjs","names":["Logger","SearchAgentWrapper: React.FC<AgentWrapperProps>","EnviveAgent","SearchProvider","SalesAgentWrapper: React.FC<SalesAgentWrapperProps>","SystemSettingsContextProvider","SalesAgentProvider","EnviveProvider: React.FC<EnviveProviderProps>","UserIdentityService","EnviveConfigService","FeatureFlagService","EnviveConfigProvider","GraphQLProvider","NewOrgConfigProvider","LocalStorageProvider","FeatureFlagServiceProvider","UserIdentityProvider","AmplitudeProvider","PageProvider","WidgetConfigProvider","HardcopyProvider","EnviveCssProvider","WindowChatToggleBinder"],"sources":["../../../src/contexts/enviveContext/enviveContext.tsx"],"sourcesContent":["import { ReactNode, useEffect, useMemo, useState } from 'react';\n\nimport { OrgConfigFeatureGate } from 'src/application/models/api/orgConfigResults';\nimport { EnviveConfigService, EnviveServiceConfig } from 'src/services/enviveConfigService';\nimport { AmplitudeService } from 'src/services/amplitudeService/amplitudeService';\nimport Logger from 'src/application/logging/logger';\nimport { UserIdentityService } from 'src/services/userIdentityService';\nimport { AmplitudeProvider } from '../amplitudeContext';\nimport { EnviveConfigProvider } from '../enviveConfigContext';\nimport { EnviveCssProvider } from '../enviveCssContext';\nimport { FeatureFlagService, FeatureFlagServiceProvider } from '../featureFlagServiceContext';\nimport { GraphQLProvider, GraphQlConfigValues } from '../graphqlContext';\nimport { LocalStorageProvider } from '../localStorageContext';\nimport { NewOrgConfigProvider } from '../newOrgConfigContext';\nimport { UserIdentityProvider } from '../userIdentityContext';\nimport { EnviveAgent } from './types';\nimport { SearchProvider } from '../searchContext';\nimport { SalesAgentProvider } from '../salesAgentContext/salesAgentContext';\nimport { SystemSettingsContextProvider } from '../systemSettingsContext';\nimport { HardcopyProvider, HardcopyResponse } from '../hardcopyContext';\nimport { WidgetTypeV3 } from '../typesV3';\nimport { WidgetConfigProvider } from '../widgetConfigContext';\nimport { PageProvider } from '../pageContext';\nimport { WindowChatToggleBinder } from './WindowChatToggleBinder';\n\nconst logger = new Logger('enviveContext');\n\ninterface AgentWrapperProps {\n  children: ReactNode;\n  enabledAgents: EnviveAgent[];\n}\n\ninterface EnviveProviderProps extends AgentWrapperProps {\n  amplitudeApiKey?: string;\n  dataResidency?: string;\n  env?: string;\n  baseUrl?: string;\n  reactAppName?: string;\n  cdnUrl?: string;\n  contextSource?: string;\n  orgLevelApiKey?: string;\n  orgShortName?: string;\n  identifyingPrefix: string;\n  featureOverrides?: Record<string, boolean>;\n  variantUrlOverride?: string;\n  variantInfoOverride?: any;\n  show?: boolean;\n  enviveOn?: boolean;\n  publicKey?: string;\n  featureGates?: OrgConfigFeatureGate[]; // New prop for feature gates\n  featureFlagService?: FeatureFlagService;\n  enviveConfigService?: EnviveConfigService;\n  amplitudeService?: AmplitudeService;\n  mockV3ConfigToDeprecatedConfig?: boolean;\n  requestV3Config?: boolean;\n  userIdentityService?: UserIdentityService;\n  previewMode?: boolean;\n  overrideConfig?: GraphQlConfigValues;\n  mockSalesAgentData?: any;\n  hardcopyOverride?: Partial<Record<WidgetTypeV3, HardcopyResponse>>;\n}\n\nconst SearchAgentWrapper: React.FC<AgentWrapperProps> = ({ children, enabledAgents }) => {\n  if (!enabledAgents.includes(EnviveAgent.SearchAgent)) {\n    return children;\n  }\n  return <SearchProvider>{children}</SearchProvider>;\n};\n\ninterface SalesAgentWrapperProps extends AgentWrapperProps {\n  previewMode?: boolean;\n  showDebugBar: boolean;\n  mockSalesAgentData?: any;\n}\n\nconst SalesAgentWrapper: React.FC<SalesAgentWrapperProps> = ({\n  children,\n  enabledAgents,\n  previewMode,\n  showDebugBar,\n  mockSalesAgentData,\n}) => {\n  if (!enabledAgents.includes(EnviveAgent.SalesAgent)) {\n    return children;\n  }\n  return (\n    <SystemSettingsContextProvider showDebugBar={showDebugBar}>\n      <SalesAgentProvider\n        previewMode={previewMode}\n        mockData={mockSalesAgentData}\n      >\n        {children}\n      </SalesAgentProvider>\n    </SystemSettingsContextProvider>\n  );\n};\n\nexport const EnviveProvider: React.FC<EnviveProviderProps> = ({\n  children,\n  previewMode = false,\n  overrideConfig,\n  enabledAgents,\n  enviveConfigService: inputEnviveConfigService,\n  featureFlagService: inputFeatureFlagService,\n  amplitudeService: inputAmplitudeService,\n  userIdentityService: inputUserIdentityService,\n  mockV3ConfigToDeprecatedConfig = false,\n  requestV3Config = false,\n  mockSalesAgentData,\n  hardcopyOverride,\n  ...config\n}) => {\n  const [userId, setUserId] = useState<string>('');\n\n  const userIdService = useMemo(\n    () =>\n      inputUserIdentityService ??\n      new UserIdentityService({ apiKey: config?.amplitudeApiKey || '' }),\n    [config?.amplitudeApiKey, inputUserIdentityService],\n  );\n\n  useEffect(() => {\n    const fetchUserId = async () => {\n      const id = await userIdService.getUserIdOrDefault();\n      setUserId(id);\n    };\n\n    fetchUserId();\n  }, [config.amplitudeApiKey, inputUserIdentityService, userIdService]);\n\n  const namespace = 'spiffy-react-components'; // TODO: Should this be different from the current namespace?\n  const source = 'app'; // TODO: Make this dependent upon the \"spiffy_on\" query param\n  const [enviveServiceConfig, setEnviveServiceConfig] = useState<EnviveServiceConfig | null>(null);\n\n  useEffect(() => {\n    // Really not happy with this approach, but I'm seeing 429 errors in the tests\n    // because of the rate limiting on the API.\n    const enviveConfigService =\n      inputEnviveConfigService ??\n      new EnviveConfigService({\n        baseUrl: config?.baseUrl || '',\n        apiKey: config?.orgLevelApiKey || '',\n        userId,\n        namespace: namespace || '',\n        source: source || '',\n      });\n    enviveConfigService.getEnviveConfig().then(newConfig => {\n      setEnviveServiceConfig(newConfig);\n      return newConfig;\n    });\n  }, [config, userId, namespace, source, inputEnviveConfigService]);\n\n  const enviveConfig = useMemo(\n    () => ({\n      ...config,\n      featureGates: enviveServiceConfig?.gates || [],\n      orgShortName: enviveServiceConfig?.org?.org?.short_name || '',\n      orgId: enviveServiceConfig?.org?.org?.id || '',\n    }),\n    [config, enviveServiceConfig],\n  );\n  logger.logDebug('enviveContext: enviveServiceConfig', enviveServiceConfig, config);\n  logger.logDebug('enviveContext: enviveConfig', enviveConfig);\n\n  const featureFlagService = useMemo(\n    () => inputFeatureFlagService ?? new FeatureFlagService(enviveConfig.featureGates),\n    [enviveConfig.featureGates, inputFeatureFlagService],\n  );\n\n  if (enviveServiceConfig === null || enviveConfig === null) {\n    return null;\n  }\n\n  return (\n    <EnviveConfigProvider {...enviveConfig}>\n      <GraphQLProvider\n        mockV3ConfigToDeprecatedConfig={mockV3ConfigToDeprecatedConfig}\n        requestV3Config={requestV3Config}\n      >\n        <NewOrgConfigProvider overrideConfig={overrideConfig}>\n          <LocalStorageProvider>\n            <FeatureFlagServiceProvider featureFlagService={featureFlagService}>\n              <UserIdentityProvider userIdService={userIdService}>\n                <AmplitudeProvider externalAmplitudeService={inputAmplitudeService}>\n                  <PageProvider previewMode={previewMode}>\n                    <WidgetConfigProvider>\n                      <HardcopyProvider hardcopyOverride={hardcopyOverride}>\n                        <EnviveCssProvider>\n                          <SearchAgentWrapper enabledAgents={enabledAgents}>\n                            <WindowChatToggleBinder />\n                            <SalesAgentWrapper\n                              enabledAgents={enabledAgents}\n                              showDebugBar={config.enviveOn || false}\n                              previewMode={previewMode}\n                              mockSalesAgentData={mockSalesAgentData}\n                            >\n                              {children}\n                            </SalesAgentWrapper>\n                          </SearchAgentWrapper>\n                        </EnviveCssProvider>\n                      </HardcopyProvider>\n                    </WidgetConfigProvider>\n                  </PageProvider>\n                </AmplitudeProvider>\n              </UserIdentityProvider>\n            </FeatureFlagServiceProvider>\n          </LocalStorageProvider>\n        </NewOrgConfigProvider>\n      </GraphQLProvider>\n    </EnviveConfigProvider>\n  );\n};\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAyBA,MAAM,SAAS,IAAIA,uBAAO,gBAAgB;AAqC1C,MAAMC,sBAAmD,EAAE,UAAU,oBAAoB;AACvF,KAAI,CAAC,cAAc,SAASC,iDAAY,YAAY,CAClD,QAAO;AAET,QAAO,2CAACC,wCAAgB,WAA0B;;AASpD,MAAMC,qBAAuD,EAC3D,UACA,eACA,aACA,cACA,yBACI;AACJ,KAAI,CAAC,cAAc,SAASF,iDAAY,WAAW,CACjD,QAAO;AAET,QACE,2CAACG;EAA4C;YAC3C,2CAACC;GACc;GACb,UAAU;GAET;IACkB;GACS;;AAIpC,MAAaC,kBAAiD,EAC5D,UACA,cAAc,OACd,gBACA,eACA,qBAAqB,0BACrB,oBAAoB,yBACpB,kBAAkB,uBAClB,qBAAqB,0BACrB,iCAAiC,OACjC,kBAAkB,OAClB,oBACA,kBACA,GAAG,aACC;CACJ,MAAM,CAAC,QAAQ,iCAA8B,GAAG;CAEhD,MAAM,yCAEF,4BACA,IAAIC,gDAAoB,EAAE,QAAQ,QAAQ,mBAAmB,IAAI,CAAC,EACpE,CAAC,QAAQ,iBAAiB,yBAAyB,CACpD;AAED,4BAAgB;EACd,MAAM,cAAc,YAAY;AAE9B,aADW,MAAM,cAAc,oBAAoB,CACtC;;AAGf,eAAa;IACZ;EAAC,OAAO;EAAiB;EAA0B;EAAc,CAAC;CAErE,MAAM,YAAY;CAClB,MAAM,SAAS;CACf,MAAM,CAAC,qBAAqB,8CAA+D,KAAK;AAEhG,4BAAgB;AAYd,GARE,4BACA,IAAIC,gDAAoB;GACtB,SAAS,QAAQ,WAAW;GAC5B,QAAQ,QAAQ,kBAAkB;GAClC;GACW;GACH;GACT,CAAC,EACgB,iBAAiB,CAAC,MAAK,cAAa;AACtD,0BAAuB,UAAU;AACjC,UAAO;IACP;IACD;EAAC;EAAQ;EAAQ;EAAW;EAAQ;EAAyB,CAAC;CAEjE,MAAM,yCACG;EACL,GAAG;EACH,cAAc,qBAAqB,SAAS,EAAE;EAC9C,cAAc,qBAAqB,KAAK,KAAK,cAAc;EAC3D,OAAO,qBAAqB,KAAK,KAAK,MAAM;EAC7C,GACD,CAAC,QAAQ,oBAAoB,CAC9B;AACD,QAAO,SAAS,sCAAsC,qBAAqB,OAAO;AAClF,QAAO,SAAS,+BAA+B,aAAa;CAE5D,MAAM,8CACE,2BAA2B,IAAIC,qDAAmB,aAAa,aAAa,EAClF,CAAC,aAAa,cAAc,wBAAwB,CACrD;AAED,KAAI,wBAAwB,QAAQ,iBAAiB,KACnD,QAAO;AAGT,QACE,2CAACC;EAAqB,GAAI;YACxB,2CAACC;GACiC;GACf;aAEjB,2CAACC;IAAqC;cACpC,2CAACC,8DACC,2CAACC;KAA+C;eAC9C,2CAACC;MAAoC;gBACnC,2CAACC;OAAkB,0BAA0B;iBAC3C,2CAACC;QAA0B;kBACzB,2CAACC,8DACC,2CAACC;SAAmC;mBAClC,2CAACC,wDACC,4CAAC;UAAkC;qBACjC,2CAACC,0DAAyB,EAC1B,2CAAC;WACgB;WACf,cAAc,OAAO,YAAY;WACpB;WACO;WAEnB;YACiB;WACD,GACH;UACH,GACE;SACV;QACG;OACC;MACI,GACR;KACF;IACP;GACG"}
172
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"enviveContext.cjs","names":["Logger","SearchAgentWrapper: React.FC<AgentWrapperProps>","EnviveAgent","SearchProvider","SalesAgentWrapper: React.FC<SalesAgentWrapperProps>","SystemSettingsContextProvider","SalesAgentProvider","EnviveProvider: React.FC<EnviveProviderProps>","UserIdentityService","EnviveConfigService","FeatureFlagService","EnviveConfigProvider","GraphQLProvider","NewOrgConfigProvider","LocalStorageProvider","FeatureFlagServiceProvider","UserIdentityProvider","AmplitudeProvider","PageProvider","WidgetConfigProvider","HardcopyProvider","EnviveCssProvider","WindowChatToggleBinder"],"sources":["../../../src/contexts/enviveContext/enviveContext.tsx"],"sourcesContent":["import { ReactNode, useCallback, useEffect, useMemo, useRef, useState } from 'react';\n\nimport { OrgConfigFeatureGate } from 'src/application/models/api/orgConfigResults';\nimport { EnviveConfigService, EnviveServiceConfig } from 'src/services/enviveConfigService';\nimport { AmplitudeService } from 'src/services/amplitudeService/amplitudeService';\nimport Logger from 'src/application/logging/logger';\nimport { UserIdentityService } from 'src/services/userIdentityService';\nimport { AmplitudeProvider } from '../amplitudeContext';\nimport { EnviveConfigProvider } from '../enviveConfigContext';\nimport { EnviveCssProvider } from '../enviveCssContext';\nimport { FeatureFlagService, FeatureFlagServiceProvider } from '../featureFlagServiceContext';\nimport { GraphQLProvider, GraphQlConfigValues } from '../graphqlContext';\nimport { LocalStorageProvider } from '../localStorageContext';\nimport { NewOrgConfigProvider } from '../newOrgConfigContext';\nimport { UserIdentityProvider } from '../userIdentityContext';\nimport { EnviveAgent } from './types';\nimport { SearchProvider } from '../searchContext';\nimport { SalesAgentProvider } from '../salesAgentContext/salesAgentContext';\nimport { SystemSettingsContextProvider } from '../systemSettingsContext';\nimport { HardcopyProvider, HardcopyResponse } from '../hardcopyContext';\nimport { WidgetTypeV3 } from '../typesV3';\nimport { WidgetConfigProvider } from '../widgetConfigContext';\nimport { PageProvider } from '../pageContext';\nimport { WindowChatToggleBinder } from './WindowChatToggleBinder';\n\nconst logger = new Logger('enviveContext');\n\ninterface AgentWrapperProps {\n  children: ReactNode;\n  enabledAgents: EnviveAgent[];\n}\n\nexport interface ExtensionStatusCallbacks {\n  onInjectionLoading?: () => Promise<unknown> | void;\n  onInjectionSuccess?: () => Promise<unknown> | void;\n  onInjectionError?: (error: Error) => Promise<unknown> | void;\n}\n\ninterface EnviveProviderProps extends AgentWrapperProps {\n  amplitudeApiKey?: string;\n  dataResidency?: string;\n  env?: string;\n  baseUrl?: string;\n  reactAppName?: string;\n  cdnUrl?: string;\n  contextSource?: string;\n  orgLevelApiKey?: string;\n  orgShortName?: string;\n  identifyingPrefix: string;\n  featureOverrides?: Record<string, boolean>;\n  variantUrlOverride?: string;\n  variantInfoOverride?: any;\n  show?: boolean;\n  enviveOn?: boolean;\n  publicKey?: string;\n  featureGates?: OrgConfigFeatureGate[]; // New prop for feature gates\n  featureFlagService?: FeatureFlagService;\n  enviveConfigService?: EnviveConfigService;\n  amplitudeService?: AmplitudeService;\n  mockV3ConfigToDeprecatedConfig?: boolean;\n  requestV3Config?: boolean;\n  userIdentityService?: UserIdentityService;\n  previewMode?: boolean;\n  overrideConfig?: GraphQlConfigValues;\n  mockSalesAgentData?: any;\n  hardcopyOverride?: Partial<Record<WidgetTypeV3, HardcopyResponse>>;\n  extensionService?: ExtensionStatusCallbacks;\n}\n\nconst SearchAgentWrapper: React.FC<AgentWrapperProps> = ({ children, enabledAgents }) => {\n  if (!enabledAgents.includes(EnviveAgent.SearchAgent)) {\n    return children;\n  }\n  return <SearchProvider>{children}</SearchProvider>;\n};\n\ninterface SalesAgentWrapperProps extends AgentWrapperProps {\n  previewMode?: boolean;\n  showDebugBar: boolean;\n  mockSalesAgentData?: any;\n}\n\nconst SalesAgentWrapper: React.FC<SalesAgentWrapperProps> = ({\n  children,\n  enabledAgents,\n  previewMode,\n  showDebugBar,\n  mockSalesAgentData,\n}) => {\n  if (!enabledAgents.includes(EnviveAgent.SalesAgent)) {\n    return children;\n  }\n  return (\n    <SystemSettingsContextProvider showDebugBar={showDebugBar}>\n      <SalesAgentProvider\n        previewMode={previewMode}\n        mockData={mockSalesAgentData}\n      >\n        {children}\n      </SalesAgentProvider>\n    </SystemSettingsContextProvider>\n  );\n};\n\nexport const EnviveProvider: React.FC<EnviveProviderProps> = ({\n  children,\n  previewMode = false,\n  overrideConfig,\n  enabledAgents,\n  enviveConfigService: inputEnviveConfigService,\n  featureFlagService: inputFeatureFlagService,\n  amplitudeService: inputAmplitudeService,\n  userIdentityService: inputUserIdentityService,\n  mockV3ConfigToDeprecatedConfig = false,\n  requestV3Config = false,\n  mockSalesAgentData,\n  hardcopyOverride,\n  extensionService,\n  ...config\n}) => {\n  const [userId, setUserId] = useState<string>('');\n  const didReportSuccessRef = useRef(false);\n\n  const invokeExtensionCallback = useCallback((callback?: () => Promise<unknown> | void) => {\n    if (!callback) {\n      return;\n    }\n\n    Promise.resolve(callback()).catch(error => {\n      logger.logDebug('Extension status callback failed', error);\n    });\n  }, []);\n\n  const invokeExtensionErrorCallback = useCallback(\n    (error: Error) => {\n      if (!extensionService?.onInjectionError) {\n        return;\n      }\n\n      Promise.resolve(extensionService.onInjectionError(error)).catch(callbackError => {\n        logger.logDebug('Extension error callback failed', callbackError);\n      });\n    },\n    [extensionService],\n  );\n\n  const userIdService = useMemo(\n    () =>\n      inputUserIdentityService ??\n      new UserIdentityService({ apiKey: config?.amplitudeApiKey || '' }),\n    [config?.amplitudeApiKey, inputUserIdentityService],\n  );\n\n  useEffect(() => {\n    const fetchUserId = async () => {\n      const id = await userIdService.getUserIdOrDefault();\n      setUserId(id);\n    };\n\n    fetchUserId();\n  }, [config.amplitudeApiKey, inputUserIdentityService, userIdService]);\n\n  const namespace = 'spiffy-react-components'; // TODO: Should this be different from the current namespace?\n  const source = 'app'; // TODO: Make this dependent upon the \"spiffy_on\" query param\n  const [enviveServiceConfig, setEnviveServiceConfig] = useState<EnviveServiceConfig | null>(null);\n\n  useEffect(() => {\n    invokeExtensionCallback(extensionService?.onInjectionLoading);\n\n    // Really not happy with this approach, but I'm seeing 429 errors in the tests\n    // because of the rate limiting on the API.\n    const enviveConfigService =\n      inputEnviveConfigService ??\n      new EnviveConfigService({\n        baseUrl: config?.baseUrl || '',\n        apiKey: config?.orgLevelApiKey || '',\n        userId,\n        namespace: namespace || '',\n        source: source || '',\n      });\n    const fetchEnviveConfig = async () => {\n      try {\n        const newConfig = await enviveConfigService.getEnviveConfig();\n        setEnviveServiceConfig(newConfig);\n      } catch (error) {\n        const resolvedError =\n          error instanceof Error ? error : new Error('Failed to load envive config');\n        logger.logError('Failed to initialize EnviveProvider', resolvedError);\n        invokeExtensionErrorCallback(resolvedError);\n      }\n    };\n\n    fetchEnviveConfig();\n  }, [\n    config,\n    userId,\n    namespace,\n    source,\n    inputEnviveConfigService,\n    extensionService,\n    invokeExtensionCallback,\n    invokeExtensionErrorCallback,\n  ]);\n\n  useEffect(() => {\n    if (!enviveServiceConfig || didReportSuccessRef.current) {\n      return;\n    }\n\n    didReportSuccessRef.current = true;\n    invokeExtensionCallback(extensionService?.onInjectionSuccess);\n  }, [enviveServiceConfig, extensionService, invokeExtensionCallback]);\n\n  const enviveConfig = useMemo(\n    () => ({\n      ...config,\n      featureGates: enviveServiceConfig?.gates || [],\n      orgShortName: enviveServiceConfig?.org?.org?.short_name || '',\n      orgId: enviveServiceConfig?.org?.org?.id || '',\n    }),\n    [config, enviveServiceConfig],\n  );\n  logger.logDebug('enviveContext: enviveServiceConfig', enviveServiceConfig, config);\n  logger.logDebug('enviveContext: enviveConfig', enviveConfig);\n\n  const featureFlagService = useMemo(\n    () => inputFeatureFlagService ?? new FeatureFlagService(enviveConfig.featureGates),\n    [enviveConfig.featureGates, inputFeatureFlagService],\n  );\n\n  if (enviveServiceConfig === null || enviveConfig === null) {\n    return null;\n  }\n\n  return (\n    <EnviveConfigProvider {...enviveConfig}>\n      <GraphQLProvider\n        mockV3ConfigToDeprecatedConfig={mockV3ConfigToDeprecatedConfig}\n        requestV3Config={requestV3Config}\n      >\n        <NewOrgConfigProvider overrideConfig={overrideConfig}>\n          <LocalStorageProvider>\n            <FeatureFlagServiceProvider featureFlagService={featureFlagService}>\n              <UserIdentityProvider userIdService={userIdService}>\n                <AmplitudeProvider externalAmplitudeService={inputAmplitudeService}>\n                  <PageProvider\n                    previewMode={previewMode}\n                    onUrlResolverNotReady={extensionService?.onInjectionError}\n                  >\n                    <WidgetConfigProvider>\n                      <HardcopyProvider hardcopyOverride={hardcopyOverride}>\n                        <EnviveCssProvider>\n                          <SearchAgentWrapper enabledAgents={enabledAgents}>\n                            <WindowChatToggleBinder />\n                            <SalesAgentWrapper\n                              enabledAgents={enabledAgents}\n                              showDebugBar={config.enviveOn || false}\n                              previewMode={previewMode}\n                              mockSalesAgentData={mockSalesAgentData}\n                            >\n                              {children}\n                            </SalesAgentWrapper>\n                          </SearchAgentWrapper>\n                        </EnviveCssProvider>\n                      </HardcopyProvider>\n                    </WidgetConfigProvider>\n                  </PageProvider>\n                </AmplitudeProvider>\n              </UserIdentityProvider>\n            </FeatureFlagServiceProvider>\n          </LocalStorageProvider>\n        </NewOrgConfigProvider>\n      </GraphQLProvider>\n    </EnviveConfigProvider>\n  );\n};\n"],"mappings":"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"}
@@ -14,6 +14,11 @@ interface AgentWrapperProps {
14
14
  children: ReactNode;
15
15
  enabledAgents: EnviveAgent[];
16
16
  }
17
+ interface ExtensionStatusCallbacks {
18
+ onInjectionLoading?: () => Promise<unknown> | void;
19
+ onInjectionSuccess?: () => Promise<unknown> | void;
20
+ onInjectionError?: (error: Error) => Promise<unknown> | void;
21
+ }
17
22
  interface EnviveProviderProps extends AgentWrapperProps {
18
23
  amplitudeApiKey?: string;
19
24
  dataResidency?: string;
@@ -42,8 +47,9 @@ interface EnviveProviderProps extends AgentWrapperProps {
42
47
  overrideConfig?: GraphQlConfigValues;
43
48
  mockSalesAgentData?: any;
44
49
  hardcopyOverride?: Partial<Record<WidgetTypeV3, HardcopyResponse>>;
50
+ extensionService?: ExtensionStatusCallbacks;
45
51
  }
46
52
  declare const EnviveProvider: React.FC<EnviveProviderProps>;
47
53
  //#endregion
48
- export { EnviveProvider };
49
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW52aXZlQ29udGV4dC5kLmN0cyIsIm5hbWVzIjpbXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvY29udGV4dHMvZW52aXZlQ29udGV4dC9lbnZpdmVDb250ZXh0LmQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUmVhY3ROb2RlIH0gZnJvbSAncmVhY3QnO1xuaW1wb3J0IHsgT3JnQ29uZmlnRmVhdHVyZUdhdGUgfSBmcm9tICdzcmMvYXBwbGljYXRpb24vbW9kZWxzL2FwaS9vcmdDb25maWdSZXN1bHRzJztcbmltcG9ydCB7IEVudml2ZUNvbmZpZ1NlcnZpY2UgfSBmcm9tICdzcmMvc2VydmljZXMvZW52aXZlQ29uZmlnU2VydmljZSc7XG5pbXBvcnQgeyBBbXBsaXR1ZGVTZXJ2aWNlIH0gZnJvbSAnc3JjL3NlcnZpY2VzL2FtcGxpdHVkZVNlcnZpY2UvYW1wbGl0dWRlU2VydmljZSc7XG5pbXBvcnQgeyBVc2VySWRlbnRpdHlTZXJ2aWNlIH0gZnJvbSAnc3JjL3NlcnZpY2VzL3VzZXJJZGVudGl0eVNlcnZpY2UnO1xuaW1wb3J0IHsgRmVhdHVyZUZsYWdTZXJ2aWNlIH0gZnJvbSAnLi4vZmVhdHVyZUZsYWdTZXJ2aWNlQ29udGV4dCc7XG5pbXBvcnQgeyBHcmFwaFFsQ29uZmlnVmFsdWVzIH0gZnJvbSAnLi4vZ3JhcGhxbENvbnRleHQnO1xuaW1wb3J0IHsgRW52aXZlQWdlbnQgfSBmcm9tICcuL3R5cGVzJztcbmltcG9ydCB7IEhhcmRjb3B5UmVzcG9uc2UgfSBmcm9tICcuLi9oYXJkY29weUNvbnRleHQnO1xuaW1wb3J0IHsgV2lkZ2V0VHlwZVYzIH0gZnJvbSAnLi4vdHlwZXNWMyc7XG5pbnRlcmZhY2UgQWdlbnRXcmFwcGVyUHJvcHMge1xuICAgIGNoaWxkcmVuOiBSZWFjdE5vZGU7XG4gICAgZW5hYmxlZEFnZW50czogRW52aXZlQWdlbnRbXTtcbn1cbmludGVyZmFjZSBFbnZpdmVQcm92aWRlclByb3BzIGV4dGVuZHMgQWdlbnRXcmFwcGVyUHJvcHMge1xuICAgIGFtcGxpdHVkZUFwaUtleT86IHN0cmluZztcbiAgICBkYXRhUmVzaWRlbmN5Pzogc3RyaW5nO1xuICAgIGVudj86IHN0cmluZztcbiAgICBiYXNlVXJsPzogc3RyaW5nO1xuICAgIHJlYWN0QXBwTmFtZT86IHN0cmluZztcbiAgICBjZG5Vcmw/OiBzdHJpbmc7XG4gICAgY29udGV4dFNvdXJjZT86IHN0cmluZztcbiAgICBvcmdMZXZlbEFwaUtleT86IHN0cmluZztcbiAgICBvcmdTaG9ydE5hbWU/OiBzdHJpbmc7XG4gICAgaWRlbnRpZnlpbmdQcmVmaXg6IHN0cmluZztcbiAgICBmZWF0dXJlT3ZlcnJpZGVzPzogUmVjb3JkPHN0cmluZywgYm9vbGVhbj47XG4gICAgdmFyaWFudFVybE92ZXJyaWRlPzogc3RyaW5nO1xuICAgIHZhcmlhbnRJbmZvT3ZlcnJpZGU/OiBhbnk7XG4gICAgc2hvdz86IGJvb2xlYW47XG4gICAgZW52aXZlT24/OiBib29sZWFuO1xuICAgIHB1YmxpY0tleT86IHN0cmluZztcbiAgICBmZWF0dXJlR2F0ZXM/OiBPcmdDb25maWdGZWF0dXJlR2F0ZVtdO1xuICAgIGZlYXR1cmVGbGFnU2VydmljZT86IEZlYXR1cmVGbGFnU2VydmljZTtcbiAgICBlbnZpdmVDb25maWdTZXJ2aWNlPzogRW52aXZlQ29uZmlnU2VydmljZTtcbiAgICBhbXBsaXR1ZGVTZXJ2aWNlPzogQW1wbGl0dWRlU2VydmljZTtcbiAgICBtb2NrVjNDb25maWdUb0RlcHJlY2F0ZWRDb25maWc/OiBib29sZWFuO1xuICAgIHJlcXVlc3RWM0NvbmZpZz86IGJvb2xlYW47XG4gICAgdXNlcklkZW50aXR5U2VydmljZT86IFVzZXJJZGVudGl0eVNlcnZpY2U7XG4gICAgcHJldmlld01vZGU/OiBib29sZWFuO1xuICAgIG92ZXJyaWRlQ29uZmlnPzogR3JhcGhRbENvbmZpZ1ZhbHVlcztcbiAgICBtb2NrU2FsZXNBZ2VudERhdGE/OiBhbnk7XG4gICAgaGFyZGNvcHlPdmVycmlkZT86IFBhcnRpYWw8UmVjb3JkPFdpZGdldFR5cGVWMywgSGFyZGNvcHlSZXNwb25zZT4+O1xufVxuZXhwb3J0IGRlY2xhcmUgY29uc3QgRW52aXZlUHJvdmlkZXI6IFJlYWN0LkZDPEVudml2ZVByb3ZpZGVyUHJvcHM+O1xuZXhwb3J0IHt9O1xuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7QUFVQSxJQUFJLG9CQUFvQjtDQUFDO09BQUc7T0FBQTtDQUFBO0FBQzVCLElBQUksc0JBQW1CO0NBQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7Q0FBQTtBQUN2QixJQUFXLGlCQUFpQjtDQUFDO09BQUc7T0FBQSxNQUFBO0NBQUEifQ==
54
+ export { EnviveProvider, ExtensionStatusCallbacks };
55
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW52aXZlQ29udGV4dC5kLmN0cyIsIm5hbWVzIjpbXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvY29udGV4dHMvZW52aXZlQ29udGV4dC9lbnZpdmVDb250ZXh0LmQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUmVhY3ROb2RlIH0gZnJvbSAncmVhY3QnO1xuaW1wb3J0IHsgT3JnQ29uZmlnRmVhdHVyZUdhdGUgfSBmcm9tICdzcmMvYXBwbGljYXRpb24vbW9kZWxzL2FwaS9vcmdDb25maWdSZXN1bHRzJztcbmltcG9ydCB7IEVudml2ZUNvbmZpZ1NlcnZpY2UgfSBmcm9tICdzcmMvc2VydmljZXMvZW52aXZlQ29uZmlnU2VydmljZSc7XG5pbXBvcnQgeyBBbXBsaXR1ZGVTZXJ2aWNlIH0gZnJvbSAnc3JjL3NlcnZpY2VzL2FtcGxpdHVkZVNlcnZpY2UvYW1wbGl0dWRlU2VydmljZSc7XG5pbXBvcnQgeyBVc2VySWRlbnRpdHlTZXJ2aWNlIH0gZnJvbSAnc3JjL3NlcnZpY2VzL3VzZXJJZGVudGl0eVNlcnZpY2UnO1xuaW1wb3J0IHsgRmVhdHVyZUZsYWdTZXJ2aWNlIH0gZnJvbSAnLi4vZmVhdHVyZUZsYWdTZXJ2aWNlQ29udGV4dCc7XG5pbXBvcnQgeyBHcmFwaFFsQ29uZmlnVmFsdWVzIH0gZnJvbSAnLi4vZ3JhcGhxbENvbnRleHQnO1xuaW1wb3J0IHsgRW52aXZlQWdlbnQgfSBmcm9tICcuL3R5cGVzJztcbmltcG9ydCB7IEhhcmRjb3B5UmVzcG9uc2UgfSBmcm9tICcuLi9oYXJkY29weUNvbnRleHQnO1xuaW1wb3J0IHsgV2lkZ2V0VHlwZVYzIH0gZnJvbSAnLi4vdHlwZXNWMyc7XG5pbnRlcmZhY2UgQWdlbnRXcmFwcGVyUHJvcHMge1xuICAgIGNoaWxkcmVuOiBSZWFjdE5vZGU7XG4gICAgZW5hYmxlZEFnZW50czogRW52aXZlQWdlbnRbXTtcbn1cbmV4cG9ydCBpbnRlcmZhY2UgRXh0ZW5zaW9uU3RhdHVzQ2FsbGJhY2tzIHtcbiAgICBvbkluamVjdGlvbkxvYWRpbmc/OiAoKSA9PiBQcm9taXNlPHVua25vd24+IHwgdm9pZDtcbiAgICBvbkluamVjdGlvblN1Y2Nlc3M/OiAoKSA9PiBQcm9taXNlPHVua25vd24+IHwgdm9pZDtcbiAgICBvbkluamVjdGlvbkVycm9yPzogKGVycm9yOiBFcnJvcikgPT4gUHJvbWlzZTx1bmtub3duPiB8IHZvaWQ7XG59XG5pbnRlcmZhY2UgRW52aXZlUHJvdmlkZXJQcm9wcyBleHRlbmRzIEFnZW50V3JhcHBlclByb3BzIHtcbiAgICBhbXBsaXR1ZGVBcGlLZXk/OiBzdHJpbmc7XG4gICAgZGF0YVJlc2lkZW5jeT86IHN0cmluZztcbiAgICBlbnY/OiBzdHJpbmc7XG4gICAgYmFzZVVybD86IHN0cmluZztcbiAgICByZWFjdEFwcE5hbWU/OiBzdHJpbmc7XG4gICAgY2RuVXJsPzogc3RyaW5nO1xuICAgIGNvbnRleHRTb3VyY2U/OiBzdHJpbmc7XG4gICAgb3JnTGV2ZWxBcGlLZXk/OiBzdHJpbmc7XG4gICAgb3JnU2hvcnROYW1lPzogc3RyaW5nO1xuICAgIGlkZW50aWZ5aW5nUHJlZml4OiBzdHJpbmc7XG4gICAgZmVhdHVyZU92ZXJyaWRlcz86IFJlY29yZDxzdHJpbmcsIGJvb2xlYW4+O1xuICAgIHZhcmlhbnRVcmxPdmVycmlkZT86IHN0cmluZztcbiAgICB2YXJpYW50SW5mb092ZXJyaWRlPzogYW55O1xuICAgIHNob3c/OiBib29sZWFuO1xuICAgIGVudml2ZU9uPzogYm9vbGVhbjtcbiAgICBwdWJsaWNLZXk/OiBzdHJpbmc7XG4gICAgZmVhdHVyZUdhdGVzPzogT3JnQ29uZmlnRmVhdHVyZUdhdGVbXTtcbiAgICBmZWF0dXJlRmxhZ1NlcnZpY2U/OiBGZWF0dXJlRmxhZ1NlcnZpY2U7XG4gICAgZW52aXZlQ29uZmlnU2VydmljZT86IEVudml2ZUNvbmZpZ1NlcnZpY2U7XG4gICAgYW1wbGl0dWRlU2VydmljZT86IEFtcGxpdHVkZVNlcnZpY2U7XG4gICAgbW9ja1YzQ29uZmlnVG9EZXByZWNhdGVkQ29uZmlnPzogYm9vbGVhbjtcbiAgICByZXF1ZXN0VjNDb25maWc/OiBib29sZWFuO1xuICAgIHVzZXJJZGVudGl0eVNlcnZpY2U/OiBVc2VySWRlbnRpdHlTZXJ2aWNlO1xuICAgIHByZXZpZXdNb2RlPzogYm9vbGVhbjtcbiAgICBvdmVycmlkZUNvbmZpZz86IEdyYXBoUWxDb25maWdWYWx1ZXM7XG4gICAgbW9ja1NhbGVzQWdlbnREYXRhPzogYW55O1xuICAgIGhhcmRjb3B5T3ZlcnJpZGU/OiBQYXJ0aWFsPFJlY29yZDxXaWRnZXRUeXBlVjMsIEhhcmRjb3B5UmVzcG9uc2U+PjtcbiAgICBleHRlbnNpb25TZXJ2aWNlPzogRXh0ZW5zaW9uU3RhdHVzQ2FsbGJhY2tzO1xufVxuZXhwb3J0IGRlY2xhcmUgY29uc3QgRW52aXZlUHJvdmlkZXI6IFJlYWN0LkZDPEVudml2ZVByb3ZpZGVyUHJvcHM+O1xuZXhwb3J0IHt9O1xuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7QUFVQSxJQUFJLG9CQUFvQjtDQUFDO09BQUc7T0FBQTtDQUFBO0FBQzVCLElBQVcsMkJBQVk7Q0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO0NBQUE7QUFDdkIsSUFBSSxzQkFBc0I7Q0FBQztPQUFLO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO0NBQUE7QUFDaEMsSUFBQSxpQkFBQTtDQUFBO09BQUE7T0FBQSxNQUFBO0NBQUEifQ==
@@ -14,6 +14,11 @@ interface AgentWrapperProps {
14
14
  children: ReactNode;
15
15
  enabledAgents: EnviveAgent[];
16
16
  }
17
+ interface ExtensionStatusCallbacks {
18
+ onInjectionLoading?: () => Promise<unknown> | void;
19
+ onInjectionSuccess?: () => Promise<unknown> | void;
20
+ onInjectionError?: (error: Error) => Promise<unknown> | void;
21
+ }
17
22
  interface EnviveProviderProps extends AgentWrapperProps {
18
23
  amplitudeApiKey?: string;
19
24
  dataResidency?: string;
@@ -42,8 +47,9 @@ interface EnviveProviderProps extends AgentWrapperProps {
42
47
  overrideConfig?: GraphQlConfigValues;
43
48
  mockSalesAgentData?: any;
44
49
  hardcopyOverride?: Partial<Record<WidgetTypeV3, HardcopyResponse>>;
50
+ extensionService?: ExtensionStatusCallbacks;
45
51
  }
46
52
  declare const EnviveProvider: React.FC<EnviveProviderProps>;
47
53
  //#endregion
48
- export { EnviveProvider };
49
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW52aXZlQ29udGV4dC5kLnRzIiwibmFtZXMiOltdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb250ZXh0cy9lbnZpdmVDb250ZXh0L2Vudml2ZUNvbnRleHQuZC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZWFjdE5vZGUgfSBmcm9tICdyZWFjdCc7XG5pbXBvcnQgeyBPcmdDb25maWdGZWF0dXJlR2F0ZSB9IGZyb20gJ3NyYy9hcHBsaWNhdGlvbi9tb2RlbHMvYXBpL29yZ0NvbmZpZ1Jlc3VsdHMnO1xuaW1wb3J0IHsgRW52aXZlQ29uZmlnU2VydmljZSB9IGZyb20gJ3NyYy9zZXJ2aWNlcy9lbnZpdmVDb25maWdTZXJ2aWNlJztcbmltcG9ydCB7IEFtcGxpdHVkZVNlcnZpY2UgfSBmcm9tICdzcmMvc2VydmljZXMvYW1wbGl0dWRlU2VydmljZS9hbXBsaXR1ZGVTZXJ2aWNlJztcbmltcG9ydCB7IFVzZXJJZGVudGl0eVNlcnZpY2UgfSBmcm9tICdzcmMvc2VydmljZXMvdXNlcklkZW50aXR5U2VydmljZSc7XG5pbXBvcnQgeyBGZWF0dXJlRmxhZ1NlcnZpY2UgfSBmcm9tICcuLi9mZWF0dXJlRmxhZ1NlcnZpY2VDb250ZXh0JztcbmltcG9ydCB7IEdyYXBoUWxDb25maWdWYWx1ZXMgfSBmcm9tICcuLi9ncmFwaHFsQ29udGV4dCc7XG5pbXBvcnQgeyBFbnZpdmVBZ2VudCB9IGZyb20gJy4vdHlwZXMnO1xuaW1wb3J0IHsgSGFyZGNvcHlSZXNwb25zZSB9IGZyb20gJy4uL2hhcmRjb3B5Q29udGV4dCc7XG5pbXBvcnQgeyBXaWRnZXRUeXBlVjMgfSBmcm9tICcuLi90eXBlc1YzJztcbmludGVyZmFjZSBBZ2VudFdyYXBwZXJQcm9wcyB7XG4gICAgY2hpbGRyZW46IFJlYWN0Tm9kZTtcbiAgICBlbmFibGVkQWdlbnRzOiBFbnZpdmVBZ2VudFtdO1xufVxuaW50ZXJmYWNlIEVudml2ZVByb3ZpZGVyUHJvcHMgZXh0ZW5kcyBBZ2VudFdyYXBwZXJQcm9wcyB7XG4gICAgYW1wbGl0dWRlQXBpS2V5Pzogc3RyaW5nO1xuICAgIGRhdGFSZXNpZGVuY3k/OiBzdHJpbmc7XG4gICAgZW52Pzogc3RyaW5nO1xuICAgIGJhc2VVcmw/OiBzdHJpbmc7XG4gICAgcmVhY3RBcHBOYW1lPzogc3RyaW5nO1xuICAgIGNkblVybD86IHN0cmluZztcbiAgICBjb250ZXh0U291cmNlPzogc3RyaW5nO1xuICAgIG9yZ0xldmVsQXBpS2V5Pzogc3RyaW5nO1xuICAgIG9yZ1Nob3J0TmFtZT86IHN0cmluZztcbiAgICBpZGVudGlmeWluZ1ByZWZpeDogc3RyaW5nO1xuICAgIGZlYXR1cmVPdmVycmlkZXM/OiBSZWNvcmQ8c3RyaW5nLCBib29sZWFuPjtcbiAgICB2YXJpYW50VXJsT3ZlcnJpZGU/OiBzdHJpbmc7XG4gICAgdmFyaWFudEluZm9PdmVycmlkZT86IGFueTtcbiAgICBzaG93PzogYm9vbGVhbjtcbiAgICBlbnZpdmVPbj86IGJvb2xlYW47XG4gICAgcHVibGljS2V5Pzogc3RyaW5nO1xuICAgIGZlYXR1cmVHYXRlcz86IE9yZ0NvbmZpZ0ZlYXR1cmVHYXRlW107XG4gICAgZmVhdHVyZUZsYWdTZXJ2aWNlPzogRmVhdHVyZUZsYWdTZXJ2aWNlO1xuICAgIGVudml2ZUNvbmZpZ1NlcnZpY2U/OiBFbnZpdmVDb25maWdTZXJ2aWNlO1xuICAgIGFtcGxpdHVkZVNlcnZpY2U/OiBBbXBsaXR1ZGVTZXJ2aWNlO1xuICAgIG1vY2tWM0NvbmZpZ1RvRGVwcmVjYXRlZENvbmZpZz86IGJvb2xlYW47XG4gICAgcmVxdWVzdFYzQ29uZmlnPzogYm9vbGVhbjtcbiAgICB1c2VySWRlbnRpdHlTZXJ2aWNlPzogVXNlcklkZW50aXR5U2VydmljZTtcbiAgICBwcmV2aWV3TW9kZT86IGJvb2xlYW47XG4gICAgb3ZlcnJpZGVDb25maWc/OiBHcmFwaFFsQ29uZmlnVmFsdWVzO1xuICAgIG1vY2tTYWxlc0FnZW50RGF0YT86IGFueTtcbiAgICBoYXJkY29weU92ZXJyaWRlPzogUGFydGlhbDxSZWNvcmQ8V2lkZ2V0VHlwZVYzLCBIYXJkY29weVJlc3BvbnNlPj47XG59XG5leHBvcnQgZGVjbGFyZSBjb25zdCBFbnZpdmVQcm92aWRlcjogUmVhY3QuRkM8RW52aXZlUHJvdmlkZXJQcm9wcz47XG5leHBvcnQge307XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7OztBQVVBLElBQUksb0JBQW9CO0NBQUM7T0FBRztPQUFBO0NBQUE7QUFDNUIsSUFBSSxzQkFBbUI7Q0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtDQUFBO0FBQ3ZCLElBQVcsaUJBQWlCO0NBQUM7T0FBRztPQUFBLE1BQUE7Q0FBQSJ9
54
+ export { EnviveProvider, ExtensionStatusCallbacks };
55
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW52aXZlQ29udGV4dC5kLnRzIiwibmFtZXMiOltdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb250ZXh0cy9lbnZpdmVDb250ZXh0L2Vudml2ZUNvbnRleHQuZC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZWFjdE5vZGUgfSBmcm9tICdyZWFjdCc7XG5pbXBvcnQgeyBPcmdDb25maWdGZWF0dXJlR2F0ZSB9IGZyb20gJ3NyYy9hcHBsaWNhdGlvbi9tb2RlbHMvYXBpL29yZ0NvbmZpZ1Jlc3VsdHMnO1xuaW1wb3J0IHsgRW52aXZlQ29uZmlnU2VydmljZSB9IGZyb20gJ3NyYy9zZXJ2aWNlcy9lbnZpdmVDb25maWdTZXJ2aWNlJztcbmltcG9ydCB7IEFtcGxpdHVkZVNlcnZpY2UgfSBmcm9tICdzcmMvc2VydmljZXMvYW1wbGl0dWRlU2VydmljZS9hbXBsaXR1ZGVTZXJ2aWNlJztcbmltcG9ydCB7IFVzZXJJZGVudGl0eVNlcnZpY2UgfSBmcm9tICdzcmMvc2VydmljZXMvdXNlcklkZW50aXR5U2VydmljZSc7XG5pbXBvcnQgeyBGZWF0dXJlRmxhZ1NlcnZpY2UgfSBmcm9tICcuLi9mZWF0dXJlRmxhZ1NlcnZpY2VDb250ZXh0JztcbmltcG9ydCB7IEdyYXBoUWxDb25maWdWYWx1ZXMgfSBmcm9tICcuLi9ncmFwaHFsQ29udGV4dCc7XG5pbXBvcnQgeyBFbnZpdmVBZ2VudCB9IGZyb20gJy4vdHlwZXMnO1xuaW1wb3J0IHsgSGFyZGNvcHlSZXNwb25zZSB9IGZyb20gJy4uL2hhcmRjb3B5Q29udGV4dCc7XG5pbXBvcnQgeyBXaWRnZXRUeXBlVjMgfSBmcm9tICcuLi90eXBlc1YzJztcbmludGVyZmFjZSBBZ2VudFdyYXBwZXJQcm9wcyB7XG4gICAgY2hpbGRyZW46IFJlYWN0Tm9kZTtcbiAgICBlbmFibGVkQWdlbnRzOiBFbnZpdmVBZ2VudFtdO1xufVxuZXhwb3J0IGludGVyZmFjZSBFeHRlbnNpb25TdGF0dXNDYWxsYmFja3Mge1xuICAgIG9uSW5qZWN0aW9uTG9hZGluZz86ICgpID0+IFByb21pc2U8dW5rbm93bj4gfCB2b2lkO1xuICAgIG9uSW5qZWN0aW9uU3VjY2Vzcz86ICgpID0+IFByb21pc2U8dW5rbm93bj4gfCB2b2lkO1xuICAgIG9uSW5qZWN0aW9uRXJyb3I/OiAoZXJyb3I6IEVycm9yKSA9PiBQcm9taXNlPHVua25vd24+IHwgdm9pZDtcbn1cbmludGVyZmFjZSBFbnZpdmVQcm92aWRlclByb3BzIGV4dGVuZHMgQWdlbnRXcmFwcGVyUHJvcHMge1xuICAgIGFtcGxpdHVkZUFwaUtleT86IHN0cmluZztcbiAgICBkYXRhUmVzaWRlbmN5Pzogc3RyaW5nO1xuICAgIGVudj86IHN0cmluZztcbiAgICBiYXNlVXJsPzogc3RyaW5nO1xuICAgIHJlYWN0QXBwTmFtZT86IHN0cmluZztcbiAgICBjZG5Vcmw/OiBzdHJpbmc7XG4gICAgY29udGV4dFNvdXJjZT86IHN0cmluZztcbiAgICBvcmdMZXZlbEFwaUtleT86IHN0cmluZztcbiAgICBvcmdTaG9ydE5hbWU/OiBzdHJpbmc7XG4gICAgaWRlbnRpZnlpbmdQcmVmaXg6IHN0cmluZztcbiAgICBmZWF0dXJlT3ZlcnJpZGVzPzogUmVjb3JkPHN0cmluZywgYm9vbGVhbj47XG4gICAgdmFyaWFudFVybE92ZXJyaWRlPzogc3RyaW5nO1xuICAgIHZhcmlhbnRJbmZvT3ZlcnJpZGU/OiBhbnk7XG4gICAgc2hvdz86IGJvb2xlYW47XG4gICAgZW52aXZlT24/OiBib29sZWFuO1xuICAgIHB1YmxpY0tleT86IHN0cmluZztcbiAgICBmZWF0dXJlR2F0ZXM/OiBPcmdDb25maWdGZWF0dXJlR2F0ZVtdO1xuICAgIGZlYXR1cmVGbGFnU2VydmljZT86IEZlYXR1cmVGbGFnU2VydmljZTtcbiAgICBlbnZpdmVDb25maWdTZXJ2aWNlPzogRW52aXZlQ29uZmlnU2VydmljZTtcbiAgICBhbXBsaXR1ZGVTZXJ2aWNlPzogQW1wbGl0dWRlU2VydmljZTtcbiAgICBtb2NrVjNDb25maWdUb0RlcHJlY2F0ZWRDb25maWc/OiBib29sZWFuO1xuICAgIHJlcXVlc3RWM0NvbmZpZz86IGJvb2xlYW47XG4gICAgdXNlcklkZW50aXR5U2VydmljZT86IFVzZXJJZGVudGl0eVNlcnZpY2U7XG4gICAgcHJldmlld01vZGU/OiBib29sZWFuO1xuICAgIG92ZXJyaWRlQ29uZmlnPzogR3JhcGhRbENvbmZpZ1ZhbHVlcztcbiAgICBtb2NrU2FsZXNBZ2VudERhdGE/OiBhbnk7XG4gICAgaGFyZGNvcHlPdmVycmlkZT86IFBhcnRpYWw8UmVjb3JkPFdpZGdldFR5cGVWMywgSGFyZGNvcHlSZXNwb25zZT4+O1xuICAgIGV4dGVuc2lvblNlcnZpY2U/OiBFeHRlbnNpb25TdGF0dXNDYWxsYmFja3M7XG59XG5leHBvcnQgZGVjbGFyZSBjb25zdCBFbnZpdmVQcm92aWRlcjogUmVhY3QuRkM8RW52aXZlUHJvdmlkZXJQcm9wcz47XG5leHBvcnQge307XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7OztBQVVBLElBQUksb0JBQW9CO0NBQUM7T0FBRztPQUFBO0NBQUE7QUFDNUIsSUFBVywyQkFBWTtDQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7Q0FBQTtBQUN2QixJQUFJLHNCQUFzQjtDQUFDO09BQUs7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7T0FBQTtPQUFBO09BQUE7Q0FBQTtBQUNoQyxJQUFBLGlCQUFBO0NBQUE7T0FBQTtPQUFBLE1BQUE7Q0FBQSJ9
@@ -31,7 +31,7 @@ import "../widgetConfigContext/index.js";
31
31
  import { PageProvider } from "../pageContext/pageContext.js";
32
32
  import "../pageContext/index.js";
33
33
  import { WindowChatToggleBinder } from "./WindowChatToggleBinder.js";
34
- import { useEffect, useMemo, useState } from "react";
34
+ import { useCallback, useEffect, useMemo, useRef, useState } from "react";
35
35
  import { jsx, jsxs } from "react/jsx-runtime";
36
36
 
37
37
  //#region src/contexts/enviveContext/enviveContext.tsx
@@ -51,8 +51,21 @@ const SalesAgentWrapper = ({ children, enabledAgents, previewMode, showDebugBar,
51
51
  })
52
52
  });
53
53
  };
54
- const EnviveProvider = ({ children, previewMode = false, overrideConfig, enabledAgents, enviveConfigService: inputEnviveConfigService, featureFlagService: inputFeatureFlagService, amplitudeService: inputAmplitudeService, userIdentityService: inputUserIdentityService, mockV3ConfigToDeprecatedConfig = false, requestV3Config = false, mockSalesAgentData, hardcopyOverride, ...config }) => {
54
+ const EnviveProvider = ({ children, previewMode = false, overrideConfig, enabledAgents, enviveConfigService: inputEnviveConfigService, featureFlagService: inputFeatureFlagService, amplitudeService: inputAmplitudeService, userIdentityService: inputUserIdentityService, mockV3ConfigToDeprecatedConfig = false, requestV3Config = false, mockSalesAgentData, hardcopyOverride, extensionService, ...config }) => {
55
55
  const [userId, setUserId] = useState("");
56
+ const didReportSuccessRef = useRef(false);
57
+ const invokeExtensionCallback = useCallback((callback) => {
58
+ if (!callback) return;
59
+ Promise.resolve(callback()).catch((error) => {
60
+ logger.logDebug("Extension status callback failed", error);
61
+ });
62
+ }, []);
63
+ const invokeExtensionErrorCallback = useCallback((error) => {
64
+ if (!extensionService?.onInjectionError) return;
65
+ Promise.resolve(extensionService.onInjectionError(error)).catch((callbackError) => {
66
+ logger.logDebug("Extension error callback failed", callbackError);
67
+ });
68
+ }, [extensionService]);
56
69
  const userIdService = useMemo(() => inputUserIdentityService ?? new UserIdentityService({ apiKey: config?.amplitudeApiKey || "" }), [config?.amplitudeApiKey, inputUserIdentityService]);
57
70
  useEffect(() => {
58
71
  const fetchUserId = async () => {
@@ -68,22 +81,42 @@ const EnviveProvider = ({ children, previewMode = false, overrideConfig, enabled
68
81
  const source = "app";
69
82
  const [enviveServiceConfig, setEnviveServiceConfig] = useState(null);
70
83
  useEffect(() => {
71
- (inputEnviveConfigService ?? new EnviveConfigService({
84
+ invokeExtensionCallback(extensionService?.onInjectionLoading);
85
+ const enviveConfigService = inputEnviveConfigService ?? new EnviveConfigService({
72
86
  baseUrl: config?.baseUrl || "",
73
87
  apiKey: config?.orgLevelApiKey || "",
74
88
  userId,
75
89
  namespace,
76
90
  source
77
- })).getEnviveConfig().then((newConfig) => {
78
- setEnviveServiceConfig(newConfig);
79
- return newConfig;
80
91
  });
92
+ const fetchEnviveConfig = async () => {
93
+ try {
94
+ setEnviveServiceConfig(await enviveConfigService.getEnviveConfig());
95
+ } catch (error) {
96
+ const resolvedError = error instanceof Error ? error : /* @__PURE__ */ new Error("Failed to load envive config");
97
+ logger.logError("Failed to initialize EnviveProvider", resolvedError);
98
+ invokeExtensionErrorCallback(resolvedError);
99
+ }
100
+ };
101
+ fetchEnviveConfig();
81
102
  }, [
82
103
  config,
83
104
  userId,
84
105
  namespace,
85
106
  source,
86
- inputEnviveConfigService
107
+ inputEnviveConfigService,
108
+ extensionService,
109
+ invokeExtensionCallback,
110
+ invokeExtensionErrorCallback
111
+ ]);
112
+ useEffect(() => {
113
+ if (!enviveServiceConfig || didReportSuccessRef.current) return;
114
+ didReportSuccessRef.current = true;
115
+ invokeExtensionCallback(extensionService?.onInjectionSuccess);
116
+ }, [
117
+ enviveServiceConfig,
118
+ extensionService,
119
+ invokeExtensionCallback
87
120
  ]);
88
121
  const enviveConfig = useMemo(() => ({
89
122
  ...config,
@@ -110,6 +143,7 @@ const EnviveProvider = ({ children, previewMode = false, overrideConfig, enabled
110
143
  externalAmplitudeService: inputAmplitudeService,
111
144
  children: /* @__PURE__ */ jsx(PageProvider, {
112
145
  previewMode,
146
+ onUrlResolverNotReady: extensionService?.onInjectionError,
113
147
  children: /* @__PURE__ */ jsx(WidgetConfigProvider, { children: /* @__PURE__ */ jsx(HardcopyProvider, {
114
148
  hardcopyOverride,
115
149
  children: /* @__PURE__ */ jsx(EnviveCssProvider, { children: /* @__PURE__ */ jsxs(SearchAgentWrapper, {
@@ -134,4 +168,4 @@ const EnviveProvider = ({ children, previewMode = false, overrideConfig, enabled
134
168
 
135
169
  //#endregion
136
170
  export { EnviveProvider };
137
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"enviveContext.js","names":["Logger","SearchAgentWrapper: React.FC<AgentWrapperProps>","SalesAgentWrapper: React.FC<SalesAgentWrapperProps>","EnviveProvider: React.FC<EnviveProviderProps>"],"sources":["../../../src/contexts/enviveContext/enviveContext.tsx"],"sourcesContent":["import { ReactNode, useEffect, useMemo, useState } from 'react';\n\nimport { OrgConfigFeatureGate } from 'src/application/models/api/orgConfigResults';\nimport { EnviveConfigService, EnviveServiceConfig } from 'src/services/enviveConfigService';\nimport { AmplitudeService } from 'src/services/amplitudeService/amplitudeService';\nimport Logger from 'src/application/logging/logger';\nimport { UserIdentityService } from 'src/services/userIdentityService';\nimport { AmplitudeProvider } from '../amplitudeContext';\nimport { EnviveConfigProvider } from '../enviveConfigContext';\nimport { EnviveCssProvider } from '../enviveCssContext';\nimport { FeatureFlagService, FeatureFlagServiceProvider } from '../featureFlagServiceContext';\nimport { GraphQLProvider, GraphQlConfigValues } from '../graphqlContext';\nimport { LocalStorageProvider } from '../localStorageContext';\nimport { NewOrgConfigProvider } from '../newOrgConfigContext';\nimport { UserIdentityProvider } from '../userIdentityContext';\nimport { EnviveAgent } from './types';\nimport { SearchProvider } from '../searchContext';\nimport { SalesAgentProvider } from '../salesAgentContext/salesAgentContext';\nimport { SystemSettingsContextProvider } from '../systemSettingsContext';\nimport { HardcopyProvider, HardcopyResponse } from '../hardcopyContext';\nimport { WidgetTypeV3 } from '../typesV3';\nimport { WidgetConfigProvider } from '../widgetConfigContext';\nimport { PageProvider } from '../pageContext';\nimport { WindowChatToggleBinder } from './WindowChatToggleBinder';\n\nconst logger = new Logger('enviveContext');\n\ninterface AgentWrapperProps {\n  children: ReactNode;\n  enabledAgents: EnviveAgent[];\n}\n\ninterface EnviveProviderProps extends AgentWrapperProps {\n  amplitudeApiKey?: string;\n  dataResidency?: string;\n  env?: string;\n  baseUrl?: string;\n  reactAppName?: string;\n  cdnUrl?: string;\n  contextSource?: string;\n  orgLevelApiKey?: string;\n  orgShortName?: string;\n  identifyingPrefix: string;\n  featureOverrides?: Record<string, boolean>;\n  variantUrlOverride?: string;\n  variantInfoOverride?: any;\n  show?: boolean;\n  enviveOn?: boolean;\n  publicKey?: string;\n  featureGates?: OrgConfigFeatureGate[]; // New prop for feature gates\n  featureFlagService?: FeatureFlagService;\n  enviveConfigService?: EnviveConfigService;\n  amplitudeService?: AmplitudeService;\n  mockV3ConfigToDeprecatedConfig?: boolean;\n  requestV3Config?: boolean;\n  userIdentityService?: UserIdentityService;\n  previewMode?: boolean;\n  overrideConfig?: GraphQlConfigValues;\n  mockSalesAgentData?: any;\n  hardcopyOverride?: Partial<Record<WidgetTypeV3, HardcopyResponse>>;\n}\n\nconst SearchAgentWrapper: React.FC<AgentWrapperProps> = ({ children, enabledAgents }) => {\n  if (!enabledAgents.includes(EnviveAgent.SearchAgent)) {\n    return children;\n  }\n  return <SearchProvider>{children}</SearchProvider>;\n};\n\ninterface SalesAgentWrapperProps extends AgentWrapperProps {\n  previewMode?: boolean;\n  showDebugBar: boolean;\n  mockSalesAgentData?: any;\n}\n\nconst SalesAgentWrapper: React.FC<SalesAgentWrapperProps> = ({\n  children,\n  enabledAgents,\n  previewMode,\n  showDebugBar,\n  mockSalesAgentData,\n}) => {\n  if (!enabledAgents.includes(EnviveAgent.SalesAgent)) {\n    return children;\n  }\n  return (\n    <SystemSettingsContextProvider showDebugBar={showDebugBar}>\n      <SalesAgentProvider\n        previewMode={previewMode}\n        mockData={mockSalesAgentData}\n      >\n        {children}\n      </SalesAgentProvider>\n    </SystemSettingsContextProvider>\n  );\n};\n\nexport const EnviveProvider: React.FC<EnviveProviderProps> = ({\n  children,\n  previewMode = false,\n  overrideConfig,\n  enabledAgents,\n  enviveConfigService: inputEnviveConfigService,\n  featureFlagService: inputFeatureFlagService,\n  amplitudeService: inputAmplitudeService,\n  userIdentityService: inputUserIdentityService,\n  mockV3ConfigToDeprecatedConfig = false,\n  requestV3Config = false,\n  mockSalesAgentData,\n  hardcopyOverride,\n  ...config\n}) => {\n  const [userId, setUserId] = useState<string>('');\n\n  const userIdService = useMemo(\n    () =>\n      inputUserIdentityService ??\n      new UserIdentityService({ apiKey: config?.amplitudeApiKey || '' }),\n    [config?.amplitudeApiKey, inputUserIdentityService],\n  );\n\n  useEffect(() => {\n    const fetchUserId = async () => {\n      const id = await userIdService.getUserIdOrDefault();\n      setUserId(id);\n    };\n\n    fetchUserId();\n  }, [config.amplitudeApiKey, inputUserIdentityService, userIdService]);\n\n  const namespace = 'spiffy-react-components'; // TODO: Should this be different from the current namespace?\n  const source = 'app'; // TODO: Make this dependent upon the \"spiffy_on\" query param\n  const [enviveServiceConfig, setEnviveServiceConfig] = useState<EnviveServiceConfig | null>(null);\n\n  useEffect(() => {\n    // Really not happy with this approach, but I'm seeing 429 errors in the tests\n    // because of the rate limiting on the API.\n    const enviveConfigService =\n      inputEnviveConfigService ??\n      new EnviveConfigService({\n        baseUrl: config?.baseUrl || '',\n        apiKey: config?.orgLevelApiKey || '',\n        userId,\n        namespace: namespace || '',\n        source: source || '',\n      });\n    enviveConfigService.getEnviveConfig().then(newConfig => {\n      setEnviveServiceConfig(newConfig);\n      return newConfig;\n    });\n  }, [config, userId, namespace, source, inputEnviveConfigService]);\n\n  const enviveConfig = useMemo(\n    () => ({\n      ...config,\n      featureGates: enviveServiceConfig?.gates || [],\n      orgShortName: enviveServiceConfig?.org?.org?.short_name || '',\n      orgId: enviveServiceConfig?.org?.org?.id || '',\n    }),\n    [config, enviveServiceConfig],\n  );\n  logger.logDebug('enviveContext: enviveServiceConfig', enviveServiceConfig, config);\n  logger.logDebug('enviveContext: enviveConfig', enviveConfig);\n\n  const featureFlagService = useMemo(\n    () => inputFeatureFlagService ?? new FeatureFlagService(enviveConfig.featureGates),\n    [enviveConfig.featureGates, inputFeatureFlagService],\n  );\n\n  if (enviveServiceConfig === null || enviveConfig === null) {\n    return null;\n  }\n\n  return (\n    <EnviveConfigProvider {...enviveConfig}>\n      <GraphQLProvider\n        mockV3ConfigToDeprecatedConfig={mockV3ConfigToDeprecatedConfig}\n        requestV3Config={requestV3Config}\n      >\n        <NewOrgConfigProvider overrideConfig={overrideConfig}>\n          <LocalStorageProvider>\n            <FeatureFlagServiceProvider featureFlagService={featureFlagService}>\n              <UserIdentityProvider userIdService={userIdService}>\n                <AmplitudeProvider externalAmplitudeService={inputAmplitudeService}>\n                  <PageProvider previewMode={previewMode}>\n                    <WidgetConfigProvider>\n                      <HardcopyProvider hardcopyOverride={hardcopyOverride}>\n                        <EnviveCssProvider>\n                          <SearchAgentWrapper enabledAgents={enabledAgents}>\n                            <WindowChatToggleBinder />\n                            <SalesAgentWrapper\n                              enabledAgents={enabledAgents}\n                              showDebugBar={config.enviveOn || false}\n                              previewMode={previewMode}\n                              mockSalesAgentData={mockSalesAgentData}\n                            >\n                              {children}\n                            </SalesAgentWrapper>\n                          </SearchAgentWrapper>\n                        </EnviveCssProvider>\n                      </HardcopyProvider>\n                    </WidgetConfigProvider>\n                  </PageProvider>\n                </AmplitudeProvider>\n              </UserIdentityProvider>\n            </FeatureFlagServiceProvider>\n          </LocalStorageProvider>\n        </NewOrgConfigProvider>\n      </GraphQLProvider>\n    </EnviveConfigProvider>\n  );\n};\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAyBA,MAAM,SAAS,IAAIA,eAAO,gBAAgB;AAqC1C,MAAMC,sBAAmD,EAAE,UAAU,oBAAoB;AACvF,KAAI,CAAC,cAAc,SAAS,YAAY,YAAY,CAClD,QAAO;AAET,QAAO,oBAAC,kBAAgB,WAA0B;;AASpD,MAAMC,qBAAuD,EAC3D,UACA,eACA,aACA,cACA,yBACI;AACJ,KAAI,CAAC,cAAc,SAAS,YAAY,WAAW,CACjD,QAAO;AAET,QACE,oBAAC;EAA4C;YAC3C,oBAAC;GACc;GACb,UAAU;GAET;IACkB;GACS;;AAIpC,MAAaC,kBAAiD,EAC5D,UACA,cAAc,OACd,gBACA,eACA,qBAAqB,0BACrB,oBAAoB,yBACpB,kBAAkB,uBAClB,qBAAqB,0BACrB,iCAAiC,OACjC,kBAAkB,OAClB,oBACA,kBACA,GAAG,aACC;CACJ,MAAM,CAAC,QAAQ,aAAa,SAAiB,GAAG;CAEhD,MAAM,gBAAgB,cAElB,4BACA,IAAI,oBAAoB,EAAE,QAAQ,QAAQ,mBAAmB,IAAI,CAAC,EACpE,CAAC,QAAQ,iBAAiB,yBAAyB,CACpD;AAED,iBAAgB;EACd,MAAM,cAAc,YAAY;AAE9B,aADW,MAAM,cAAc,oBAAoB,CACtC;;AAGf,eAAa;IACZ;EAAC,OAAO;EAAiB;EAA0B;EAAc,CAAC;CAErE,MAAM,YAAY;CAClB,MAAM,SAAS;CACf,MAAM,CAAC,qBAAqB,0BAA0B,SAAqC,KAAK;AAEhG,iBAAgB;AAYd,GARE,4BACA,IAAI,oBAAoB;GACtB,SAAS,QAAQ,WAAW;GAC5B,QAAQ,QAAQ,kBAAkB;GAClC;GACW;GACH;GACT,CAAC,EACgB,iBAAiB,CAAC,MAAK,cAAa;AACtD,0BAAuB,UAAU;AACjC,UAAO;IACP;IACD;EAAC;EAAQ;EAAQ;EAAW;EAAQ;EAAyB,CAAC;CAEjE,MAAM,eAAe,eACZ;EACL,GAAG;EACH,cAAc,qBAAqB,SAAS,EAAE;EAC9C,cAAc,qBAAqB,KAAK,KAAK,cAAc;EAC3D,OAAO,qBAAqB,KAAK,KAAK,MAAM;EAC7C,GACD,CAAC,QAAQ,oBAAoB,CAC9B;AACD,QAAO,SAAS,sCAAsC,qBAAqB,OAAO;AAClF,QAAO,SAAS,+BAA+B,aAAa;CAE5D,MAAM,qBAAqB,cACnB,2BAA2B,IAAI,mBAAmB,aAAa,aAAa,EAClF,CAAC,aAAa,cAAc,wBAAwB,CACrD;AAED,KAAI,wBAAwB,QAAQ,iBAAiB,KACnD,QAAO;AAGT,QACE,oBAAC;EAAqB,GAAI;YACxB,oBAAC;GACiC;GACf;aAEjB,oBAAC;IAAqC;cACpC,oBAAC,kCACC,oBAAC;KAA+C;eAC9C,oBAAC;MAAoC;gBACnC,oBAAC;OAAkB,0BAA0B;iBAC3C,oBAAC;QAA0B;kBACzB,oBAAC,kCACC,oBAAC;SAAmC;mBAClC,oBAAC,+BACC,qBAAC;UAAkC;qBACjC,oBAAC,2BAAyB,EAC1B,oBAAC;WACgB;WACf,cAAc,OAAO,YAAY;WACpB;WACO;WAEnB;YACiB;WACD,GACH;UACH,GACE;SACV;QACG;OACC;MACI,GACR;KACF;IACP;GACG"}
171
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"enviveContext.js","names":["Logger","SearchAgentWrapper: React.FC<AgentWrapperProps>","SalesAgentWrapper: React.FC<SalesAgentWrapperProps>","EnviveProvider: React.FC<EnviveProviderProps>"],"sources":["../../../src/contexts/enviveContext/enviveContext.tsx"],"sourcesContent":["import { ReactNode, useCallback, useEffect, useMemo, useRef, useState } from 'react';\n\nimport { OrgConfigFeatureGate } from 'src/application/models/api/orgConfigResults';\nimport { EnviveConfigService, EnviveServiceConfig } from 'src/services/enviveConfigService';\nimport { AmplitudeService } from 'src/services/amplitudeService/amplitudeService';\nimport Logger from 'src/application/logging/logger';\nimport { UserIdentityService } from 'src/services/userIdentityService';\nimport { AmplitudeProvider } from '../amplitudeContext';\nimport { EnviveConfigProvider } from '../enviveConfigContext';\nimport { EnviveCssProvider } from '../enviveCssContext';\nimport { FeatureFlagService, FeatureFlagServiceProvider } from '../featureFlagServiceContext';\nimport { GraphQLProvider, GraphQlConfigValues } from '../graphqlContext';\nimport { LocalStorageProvider } from '../localStorageContext';\nimport { NewOrgConfigProvider } from '../newOrgConfigContext';\nimport { UserIdentityProvider } from '../userIdentityContext';\nimport { EnviveAgent } from './types';\nimport { SearchProvider } from '../searchContext';\nimport { SalesAgentProvider } from '../salesAgentContext/salesAgentContext';\nimport { SystemSettingsContextProvider } from '../systemSettingsContext';\nimport { HardcopyProvider, HardcopyResponse } from '../hardcopyContext';\nimport { WidgetTypeV3 } from '../typesV3';\nimport { WidgetConfigProvider } from '../widgetConfigContext';\nimport { PageProvider } from '../pageContext';\nimport { WindowChatToggleBinder } from './WindowChatToggleBinder';\n\nconst logger = new Logger('enviveContext');\n\ninterface AgentWrapperProps {\n  children: ReactNode;\n  enabledAgents: EnviveAgent[];\n}\n\nexport interface ExtensionStatusCallbacks {\n  onInjectionLoading?: () => Promise<unknown> | void;\n  onInjectionSuccess?: () => Promise<unknown> | void;\n  onInjectionError?: (error: Error) => Promise<unknown> | void;\n}\n\ninterface EnviveProviderProps extends AgentWrapperProps {\n  amplitudeApiKey?: string;\n  dataResidency?: string;\n  env?: string;\n  baseUrl?: string;\n  reactAppName?: string;\n  cdnUrl?: string;\n  contextSource?: string;\n  orgLevelApiKey?: string;\n  orgShortName?: string;\n  identifyingPrefix: string;\n  featureOverrides?: Record<string, boolean>;\n  variantUrlOverride?: string;\n  variantInfoOverride?: any;\n  show?: boolean;\n  enviveOn?: boolean;\n  publicKey?: string;\n  featureGates?: OrgConfigFeatureGate[]; // New prop for feature gates\n  featureFlagService?: FeatureFlagService;\n  enviveConfigService?: EnviveConfigService;\n  amplitudeService?: AmplitudeService;\n  mockV3ConfigToDeprecatedConfig?: boolean;\n  requestV3Config?: boolean;\n  userIdentityService?: UserIdentityService;\n  previewMode?: boolean;\n  overrideConfig?: GraphQlConfigValues;\n  mockSalesAgentData?: any;\n  hardcopyOverride?: Partial<Record<WidgetTypeV3, HardcopyResponse>>;\n  extensionService?: ExtensionStatusCallbacks;\n}\n\nconst SearchAgentWrapper: React.FC<AgentWrapperProps> = ({ children, enabledAgents }) => {\n  if (!enabledAgents.includes(EnviveAgent.SearchAgent)) {\n    return children;\n  }\n  return <SearchProvider>{children}</SearchProvider>;\n};\n\ninterface SalesAgentWrapperProps extends AgentWrapperProps {\n  previewMode?: boolean;\n  showDebugBar: boolean;\n  mockSalesAgentData?: any;\n}\n\nconst SalesAgentWrapper: React.FC<SalesAgentWrapperProps> = ({\n  children,\n  enabledAgents,\n  previewMode,\n  showDebugBar,\n  mockSalesAgentData,\n}) => {\n  if (!enabledAgents.includes(EnviveAgent.SalesAgent)) {\n    return children;\n  }\n  return (\n    <SystemSettingsContextProvider showDebugBar={showDebugBar}>\n      <SalesAgentProvider\n        previewMode={previewMode}\n        mockData={mockSalesAgentData}\n      >\n        {children}\n      </SalesAgentProvider>\n    </SystemSettingsContextProvider>\n  );\n};\n\nexport const EnviveProvider: React.FC<EnviveProviderProps> = ({\n  children,\n  previewMode = false,\n  overrideConfig,\n  enabledAgents,\n  enviveConfigService: inputEnviveConfigService,\n  featureFlagService: inputFeatureFlagService,\n  amplitudeService: inputAmplitudeService,\n  userIdentityService: inputUserIdentityService,\n  mockV3ConfigToDeprecatedConfig = false,\n  requestV3Config = false,\n  mockSalesAgentData,\n  hardcopyOverride,\n  extensionService,\n  ...config\n}) => {\n  const [userId, setUserId] = useState<string>('');\n  const didReportSuccessRef = useRef(false);\n\n  const invokeExtensionCallback = useCallback((callback?: () => Promise<unknown> | void) => {\n    if (!callback) {\n      return;\n    }\n\n    Promise.resolve(callback()).catch(error => {\n      logger.logDebug('Extension status callback failed', error);\n    });\n  }, []);\n\n  const invokeExtensionErrorCallback = useCallback(\n    (error: Error) => {\n      if (!extensionService?.onInjectionError) {\n        return;\n      }\n\n      Promise.resolve(extensionService.onInjectionError(error)).catch(callbackError => {\n        logger.logDebug('Extension error callback failed', callbackError);\n      });\n    },\n    [extensionService],\n  );\n\n  const userIdService = useMemo(\n    () =>\n      inputUserIdentityService ??\n      new UserIdentityService({ apiKey: config?.amplitudeApiKey || '' }),\n    [config?.amplitudeApiKey, inputUserIdentityService],\n  );\n\n  useEffect(() => {\n    const fetchUserId = async () => {\n      const id = await userIdService.getUserIdOrDefault();\n      setUserId(id);\n    };\n\n    fetchUserId();\n  }, [config.amplitudeApiKey, inputUserIdentityService, userIdService]);\n\n  const namespace = 'spiffy-react-components'; // TODO: Should this be different from the current namespace?\n  const source = 'app'; // TODO: Make this dependent upon the \"spiffy_on\" query param\n  const [enviveServiceConfig, setEnviveServiceConfig] = useState<EnviveServiceConfig | null>(null);\n\n  useEffect(() => {\n    invokeExtensionCallback(extensionService?.onInjectionLoading);\n\n    // Really not happy with this approach, but I'm seeing 429 errors in the tests\n    // because of the rate limiting on the API.\n    const enviveConfigService =\n      inputEnviveConfigService ??\n      new EnviveConfigService({\n        baseUrl: config?.baseUrl || '',\n        apiKey: config?.orgLevelApiKey || '',\n        userId,\n        namespace: namespace || '',\n        source: source || '',\n      });\n    const fetchEnviveConfig = async () => {\n      try {\n        const newConfig = await enviveConfigService.getEnviveConfig();\n        setEnviveServiceConfig(newConfig);\n      } catch (error) {\n        const resolvedError =\n          error instanceof Error ? error : new Error('Failed to load envive config');\n        logger.logError('Failed to initialize EnviveProvider', resolvedError);\n        invokeExtensionErrorCallback(resolvedError);\n      }\n    };\n\n    fetchEnviveConfig();\n  }, [\n    config,\n    userId,\n    namespace,\n    source,\n    inputEnviveConfigService,\n    extensionService,\n    invokeExtensionCallback,\n    invokeExtensionErrorCallback,\n  ]);\n\n  useEffect(() => {\n    if (!enviveServiceConfig || didReportSuccessRef.current) {\n      return;\n    }\n\n    didReportSuccessRef.current = true;\n    invokeExtensionCallback(extensionService?.onInjectionSuccess);\n  }, [enviveServiceConfig, extensionService, invokeExtensionCallback]);\n\n  const enviveConfig = useMemo(\n    () => ({\n      ...config,\n      featureGates: enviveServiceConfig?.gates || [],\n      orgShortName: enviveServiceConfig?.org?.org?.short_name || '',\n      orgId: enviveServiceConfig?.org?.org?.id || '',\n    }),\n    [config, enviveServiceConfig],\n  );\n  logger.logDebug('enviveContext: enviveServiceConfig', enviveServiceConfig, config);\n  logger.logDebug('enviveContext: enviveConfig', enviveConfig);\n\n  const featureFlagService = useMemo(\n    () => inputFeatureFlagService ?? new FeatureFlagService(enviveConfig.featureGates),\n    [enviveConfig.featureGates, inputFeatureFlagService],\n  );\n\n  if (enviveServiceConfig === null || enviveConfig === null) {\n    return null;\n  }\n\n  return (\n    <EnviveConfigProvider {...enviveConfig}>\n      <GraphQLProvider\n        mockV3ConfigToDeprecatedConfig={mockV3ConfigToDeprecatedConfig}\n        requestV3Config={requestV3Config}\n      >\n        <NewOrgConfigProvider overrideConfig={overrideConfig}>\n          <LocalStorageProvider>\n            <FeatureFlagServiceProvider featureFlagService={featureFlagService}>\n              <UserIdentityProvider userIdService={userIdService}>\n                <AmplitudeProvider externalAmplitudeService={inputAmplitudeService}>\n                  <PageProvider\n                    previewMode={previewMode}\n                    onUrlResolverNotReady={extensionService?.onInjectionError}\n                  >\n                    <WidgetConfigProvider>\n                      <HardcopyProvider hardcopyOverride={hardcopyOverride}>\n                        <EnviveCssProvider>\n                          <SearchAgentWrapper enabledAgents={enabledAgents}>\n                            <WindowChatToggleBinder />\n                            <SalesAgentWrapper\n                              enabledAgents={enabledAgents}\n                              showDebugBar={config.enviveOn || false}\n                              previewMode={previewMode}\n                              mockSalesAgentData={mockSalesAgentData}\n                            >\n                              {children}\n                            </SalesAgentWrapper>\n                          </SearchAgentWrapper>\n                        </EnviveCssProvider>\n                      </HardcopyProvider>\n                    </WidgetConfigProvider>\n                  </PageProvider>\n                </AmplitudeProvider>\n              </UserIdentityProvider>\n            </FeatureFlagServiceProvider>\n          </LocalStorageProvider>\n        </NewOrgConfigProvider>\n      </GraphQLProvider>\n    </EnviveConfigProvider>\n  );\n};\n"],"mappings":"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"}
@@ -1,3 +1,3 @@
1
1
  import { EnviveAgent } from "./types.cjs";
2
- import { EnviveProvider } from "./enviveContext.cjs";
3
- export { EnviveAgent, EnviveProvider };
2
+ import { EnviveProvider, ExtensionStatusCallbacks } from "./enviveContext.cjs";
3
+ export { EnviveAgent, EnviveProvider, ExtensionStatusCallbacks };
@@ -1,3 +1,3 @@
1
1
  import { EnviveAgent } from "./types.js";
2
- import { EnviveProvider } from "./enviveContext.js";
3
- export { EnviveAgent, EnviveProvider };
2
+ import { EnviveProvider, ExtensionStatusCallbacks } from "./enviveContext.js";
3
+ export { EnviveAgent, EnviveProvider, ExtensionStatusCallbacks };
@@ -5,4 +5,4 @@ declare enum EnviveAgent {
5
5
  }
6
6
  //#endregion
7
7
  export { EnviveAgent };
8
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuZC50cyIsIm5hbWVzIjpbXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvY29udGV4dHMvZW52aXZlQ29udGV4dC90eXBlcy5kLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBkZWNsYXJlIGVudW0gRW52aXZlQWdlbnQge1xuICAgIFNhbGVzQWdlbnQgPSBcInNhbGVzXCIsXG4gICAgU2VhcmNoQWdlbnQgPSBcInNlYXJjaFwiXG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxJQUFXLGNBQWMsQ0FBQyxJQUFJIn0=
8
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuZC50cyIsIm5hbWVzIjpbXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvY29udGV4dHMvZW52aXZlQ29udGV4dC90eXBlcy5kLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBkZWNsYXJlIGVudW0gRW52aXZlQWdlbnQge1xuICAgIFNhbGVzQWdlbnQgPSBcInNhbGVzXCIsXG4gICAgU2VhcmNoQWdlbnQgPSBcInNlYXJjaFwiXG59XG4iXSwibWFwcGluZ3MiOiI7QUFBQSxJQUFXLGNBQWMsQ0FBQyxHQUFHIn0=
@@ -2,8 +2,11 @@ const require_rolldown_runtime = require('../../_virtual/rolldown_runtime.cjs');
2
2
  const require_logger = require('../../application/logging/logger.cjs');
3
3
  const require_variantInfo = require('../../application/models/variantInfo/variantInfo.cjs');
4
4
  require('../../application/models/index.cjs');
5
+ const require_eventNames = require('../../services/amplitudeService/eventNames.cjs');
5
6
  const require_variant = require('../../atoms/app/variant.cjs');
6
7
  const require_application_commerce_api = require('../../application/commerce-api.cjs');
8
+ const require_amplitudeContext = require('../amplitudeContext/amplitudeContext.cjs');
9
+ require('../amplitudeContext/index.cjs');
7
10
  const require_mapping = require('./mapping.cjs');
8
11
  let __spiffy_ai_commerce_api_client = require("@spiffy-ai/commerce-api-client");
9
12
  let react = require("react");
@@ -13,13 +16,14 @@ let jotai = require("jotai");
13
16
  //#region src/contexts/pageContext/pageContext.tsx
14
17
  const logger = new require_logger.default("pageContext");
15
18
  const PageContext = (0, react.createContext)(void 0);
16
- const PageProvider = ({ children, previewMode = false }) => {
19
+ const PageProvider = ({ children, previewMode = false, onUrlResolverNotReady }) => {
17
20
  const [pageUrl, setPageUrl] = (0, react.useState)(void 0);
18
21
  const [urlResolverResponse, setUrlResolverResponse] = (0, jotai.useAtom)(require_variant.urlResolverAtom);
19
22
  const [isLoading, setIsLoading] = (0, react.useState)(false);
20
23
  const [variantInfo, setVariantInfo] = (0, jotai.useAtom)(require_variant.pageVariantInfoAtom);
21
24
  const [userEvent, setUserEvent] = (0, jotai.useAtom)(require_variant.pageUserEventAtom);
22
25
  const [isSupported, setIsSupported] = (0, react.useState)(false);
26
+ const lastNotReadyUrlRef = (0, react.useRef)(null);
23
27
  (0, react.useEffect)(() => {
24
28
  setPageUrl(window.location.href);
25
29
  }, []);
@@ -51,11 +55,19 @@ const PageProvider = ({ children, previewMode = false }) => {
51
55
  if (response.ready) {
52
56
  const newVariantInfo = require_mapping.mapUrlResolverResponseToVariantInfo(url, response);
53
57
  const newUserEvent = require_mapping.mapApiUserEventToUserEvent(response.user_event ?? void 0);
58
+ logger.logDebug("setting variant info and user event", {
59
+ newVariantInfo,
60
+ newUserEvent
61
+ });
54
62
  setVariantInfo(newVariantInfo);
55
63
  setUserEvent(newUserEvent);
56
64
  } else {
57
65
  setUserEvent(void 0);
58
66
  setVariantInfo(void 0);
67
+ logger.logDebug("set variant info and user event to undefined", {
68
+ variantInfo,
69
+ userEvent
70
+ });
59
71
  }
60
72
  }, [setVariantInfo, setUserEvent]);
61
73
  (0, react.useEffect)(() => {
@@ -75,6 +87,14 @@ const PageProvider = ({ children, previewMode = false }) => {
75
87
  setVariantFromUrlResolver(cleansedUrl, response);
76
88
  setIsLoading(false);
77
89
  setIsSupported(response.ready);
90
+ if (response.ready) lastNotReadyUrlRef.current = null;
91
+ else if (onUrlResolverNotReady && lastNotReadyUrlRef.current !== cleansedUrl) {
92
+ lastNotReadyUrlRef.current = cleansedUrl;
93
+ const err = /* @__PURE__ */ new Error(`URL resolver returned ready=false for ${cleansedUrl}`);
94
+ Promise.resolve(onUrlResolverNotReady(err)).catch((callbackError) => {
95
+ logger.logDebug("onUrlResolverNotReady callback failed", callbackError);
96
+ });
97
+ }
78
98
  } catch (e) {
79
99
  setIsLoading(false);
80
100
  logger.logError("Failed to resolve page URL", e, { pageUrl });
@@ -88,7 +108,62 @@ const PageProvider = ({ children, previewMode = false }) => {
88
108
  setVariantFromUrlResolver,
89
109
  userEvent,
90
110
  variantInfo,
91
- previewMode
111
+ previewMode,
112
+ onUrlResolverNotReady
113
+ ]);
114
+ const { trackEvent } = require_amplitudeContext.useAmplitude();
115
+ const resolvedVariantInfo = (0, jotai.useAtomValue)(require_variant.variantInfoAtom);
116
+ const hasParsedVariantInfo = (0, jotai.useAtomValue)(require_variant.hasParsedVariantInfoAtom);
117
+ const pageViewedFired = (0, react.useRef)(false);
118
+ (0, react.useEffect)(() => {
119
+ if (pageViewedFired.current || isLoading || !hasParsedVariantInfo || !variantInfo) return;
120
+ let pageType;
121
+ let pageId;
122
+ switch (variantInfo.variantType) {
123
+ case require_variantInfo.VariantTypeEnum.Pdp:
124
+ pageType = "pdp";
125
+ pageId = variantInfo.productId;
126
+ break;
127
+ case require_variantInfo.VariantTypeEnum.Plp:
128
+ pageType = "plp";
129
+ pageId = variantInfo.plpId;
130
+ break;
131
+ case require_variantInfo.VariantTypeEnum.Home:
132
+ pageType = "homepage";
133
+ pageId = variantInfo.url;
134
+ break;
135
+ case require_variantInfo.VariantTypeEnum.Other:
136
+ pageType = "other";
137
+ pageId = variantInfo.url;
138
+ break;
139
+ case require_variantInfo.VariantTypeEnum.PageVisit:
140
+ pageType = "other";
141
+ pageId = variantInfo.url;
142
+ break;
143
+ case require_variantInfo.VariantTypeEnum.FullPageSalesAgent:
144
+ pageType = "full_page_sales_agent";
145
+ pageId = variantInfo.url;
146
+ break;
147
+ default: return;
148
+ }
149
+ pageViewedFired.current = true;
150
+ trackEvent({
151
+ eventName: require_eventNames.EnviveMetricsEventName.PageViewed,
152
+ eventProps: {
153
+ "context.page_type": pageType,
154
+ "context.page_id": pageId,
155
+ "context.supported": isSupported,
156
+ "context.ready": isSupported,
157
+ "context.page_variant_id": resolvedVariantInfo.variantId
158
+ }
159
+ });
160
+ }, [
161
+ isLoading,
162
+ isSupported,
163
+ hasParsedVariantInfo,
164
+ variantInfo,
165
+ resolvedVariantInfo,
166
+ trackEvent
92
167
  ]);
93
168
  const setPageUrlStable = (0, react.useCallback)((url) => {
94
169
  setPageUrl(url);
@@ -122,4 +197,4 @@ const usePage = () => {
122
197
  //#endregion
123
198
  exports.PageProvider = PageProvider;
124
199
  exports.usePage = usePage;
125
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"pageContext.cjs","names":["Logger","PageProvider: React.FC<{\n  children: ReactNode;\n  previewMode: boolean;\n}>","urlResolverAtom","pageVariantInfoAtom","pageUserEventAtom","UserEventCategory","PageVisitCategory","VariantTypeEnum","mapUrlResolverResponseToVariantInfo","mapApiUserEventToUserEvent","CommerceApiClient"],"sources":["../../../src/contexts/pageContext/pageContext.tsx"],"sourcesContent":["import {\n  ReactNode,\n  createContext,\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useState,\n} from 'react';\nimport {\n  UrlResolverResponse,\n  pageUserEventAtom,\n  pageVariantInfoAtom,\n  urlResolverAtom,\n} from 'src/atoms/app/variant';\nimport { useAtom } from 'jotai';\nimport CommerceApiClient from 'src/application/commerce-api';\nimport Logger from 'src/application/logging/logger';\nimport { PageVisitCategory, UserEventCategory } from '@spiffy-ai/commerce-api-client';\nimport { VariantTypeEnum } from 'src/application/models';\nimport { mapApiUserEventToUserEvent, mapUrlResolverResponseToVariantInfo } from './mapping';\nimport { PageDetails } from './types';\n\nconst logger = new Logger('pageContext');\n\nconst PageContext = createContext<\n  | (PageDetails & {\n      isLoading: boolean;\n      isSupported: boolean;\n      setPageUrl: (url: string) => void;\n    })\n  | undefined\n>(undefined);\n\nexport const PageProvider: React.FC<{\n  children: ReactNode;\n  previewMode: boolean;\n}> = ({ children, previewMode = false }) => {\n  const [pageUrl, setPageUrl] = useState<string | undefined>(undefined);\n  const [urlResolverResponse, setUrlResolverResponse] = useAtom(urlResolverAtom);\n  const [isLoading, setIsLoading] = useState(false);\n  const [variantInfo, setVariantInfo] = useAtom(pageVariantInfoAtom);\n  const [userEvent, setUserEvent] = useAtom(pageUserEventAtom);\n  const [isSupported, setIsSupported] = useState(false);\n  useEffect(() => {\n    setPageUrl(window.location.href);\n  }, []);\n\n  // Auto-initialize atoms for preview mode\n  useEffect(() => {\n    if (previewMode && !userEvent && !variantInfo) {\n      const url = typeof window !== 'undefined' ? window.location.href : '';\n      setUserEvent({\n        event_id: 'preview-user-event',\n        created_at: new Date().toISOString(),\n        category: UserEventCategory.PageVisit,\n        attributes: {\n          page_visit_category: PageVisitCategory.Other,\n          url,\n        },\n      } as any);\n\n      setVariantInfo({\n        variantType: VariantTypeEnum.PageVisit,\n        url,\n      });\n    }\n  }, [previewMode, userEvent, variantInfo, setUserEvent, setVariantInfo]);\n\n  const setVariantFromUrlResolver = useCallback(\n    (url: string, response: UrlResolverResponse) => {\n      if (response.ready) {\n        const newVariantInfo = mapUrlResolverResponseToVariantInfo(url, response);\n        const newUserEvent = mapApiUserEventToUserEvent(response.user_event ?? undefined);\n        setVariantInfo(newVariantInfo);\n        setUserEvent(newUserEvent);\n      } else {\n        setUserEvent(undefined);\n        setVariantInfo(undefined);\n      }\n    },\n    [setVariantInfo, setUserEvent],\n  );\n\n  useEffect(() => {\n    const resolvePageUrl = async () => {\n      setIsLoading(true);\n      try {\n        if (!pageUrl) {\n          setIsLoading(false);\n          return;\n        }\n\n        // If in preview mode or atoms are already set (e.g., by storybook), skip API call\n        if (previewMode || userEvent || variantInfo) {\n          setIsLoading(false);\n          return;\n        }\n\n        const cleansedUrl = pageUrl.toLowerCase().trim();\n        const response =\n          urlResolverResponse[cleansedUrl] ??\n          ((await CommerceApiClient.resolveUrl(cleansedUrl)) as UrlResolverResponse);\n        setVariantFromUrlResolver(cleansedUrl, response);\n        setIsLoading(false);\n        setIsSupported(response.ready);\n      } catch (e) {\n        setIsLoading(false);\n        logger.logError('Failed to resolve page URL', e, { pageUrl });\n      }\n    };\n    resolvePageUrl();\n  }, [\n    pageUrl,\n    urlResolverResponse,\n    setUrlResolverResponse,\n    setVariantFromUrlResolver,\n    userEvent,\n    variantInfo,\n    previewMode,\n  ]);\n\n  const setPageUrlStable = useCallback((url: string) => {\n    setPageUrl(url);\n  }, []);\n\n  const value = useMemo(\n    () => ({\n      pageUrl,\n      variantInfo,\n      userEvent,\n      isLoading,\n      isSupported,\n      setPageUrl: setPageUrlStable,\n    }),\n    [pageUrl, userEvent, variantInfo, isLoading, isSupported, setPageUrlStable],\n  );\n\n  return <PageContext.Provider value={value}>{children}</PageContext.Provider>;\n};\n\nexport const usePage = () => {\n  const context = useContext(PageContext);\n  if (!context) {\n    throw new Error('usePage must be used within a PageProvider');\n  }\n  return context;\n};\n"],"mappings":";;;;;;;;;;;;;AAuBA,MAAM,SAAS,IAAIA,uBAAO,cAAc;AAExC,MAAM,uCAOJ,OAAU;AAEZ,MAAaC,gBAGP,EAAE,UAAU,cAAc,YAAY;CAC1C,MAAM,CAAC,SAAS,kCAA2C,OAAU;CACrE,MAAM,CAAC,qBAAqB,6CAAkCC,gCAAgB;CAC9E,MAAM,CAAC,WAAW,oCAAyB,MAAM;CACjD,MAAM,CAAC,aAAa,qCAA0BC,oCAAoB;CAClE,MAAM,CAAC,WAAW,mCAAwBC,kCAAkB;CAC5D,MAAM,CAAC,aAAa,sCAA2B,MAAM;AACrD,4BAAgB;AACd,aAAW,OAAO,SAAS,KAAK;IAC/B,EAAE,CAAC;AAGN,4BAAgB;AACd,MAAI,eAAe,CAAC,aAAa,CAAC,aAAa;GAC7C,MAAM,MAAM,OAAO,WAAW,cAAc,OAAO,SAAS,OAAO;AACnE,gBAAa;IACX,UAAU;IACV,6BAAY,IAAI,MAAM,EAAC,aAAa;IACpC,UAAUC,kDAAkB;IAC5B,YAAY;KACV,qBAAqBC,kDAAkB;KACvC;KACD;IACF,CAAQ;AAET,kBAAe;IACb,aAAaC,oCAAgB;IAC7B;IACD,CAAC;;IAEH;EAAC;EAAa;EAAW;EAAa;EAAc;EAAe,CAAC;CAEvE,MAAM,oDACH,KAAa,aAAkC;AAC9C,MAAI,SAAS,OAAO;GAClB,MAAM,iBAAiBC,oDAAoC,KAAK,SAAS;GACzE,MAAM,eAAeC,2CAA2B,SAAS,cAAc,OAAU;AACjF,kBAAe,eAAe;AAC9B,gBAAa,aAAa;SACrB;AACL,gBAAa,OAAU;AACvB,kBAAe,OAAU;;IAG7B,CAAC,gBAAgB,aAAa,CAC/B;AAED,4BAAgB;EACd,MAAM,iBAAiB,YAAY;AACjC,gBAAa,KAAK;AAClB,OAAI;AACF,QAAI,CAAC,SAAS;AACZ,kBAAa,MAAM;AACnB;;AAIF,QAAI,eAAe,aAAa,aAAa;AAC3C,kBAAa,MAAM;AACnB;;IAGF,MAAM,cAAc,QAAQ,aAAa,CAAC,MAAM;IAChD,MAAM,WACJ,oBAAoB,gBAClB,MAAMC,yCAAkB,WAAW,YAAY;AACnD,8BAA0B,aAAa,SAAS;AAChD,iBAAa,MAAM;AACnB,mBAAe,SAAS,MAAM;YACvB,GAAG;AACV,iBAAa,MAAM;AACnB,WAAO,SAAS,8BAA8B,GAAG,EAAE,SAAS,CAAC;;;AAGjE,kBAAgB;IACf;EACD;EACA;EACA;EACA;EACA;EACA;EACA;EACD,CAAC;CAEF,MAAM,2CAAgC,QAAgB;AACpD,aAAW,IAAI;IACd,EAAE,CAAC;CAEN,MAAM,kCACG;EACL;EACA;EACA;EACA;EACA;EACA,YAAY;EACb,GACD;EAAC;EAAS;EAAW;EAAa;EAAW;EAAa;EAAiB,CAC5E;AAED,QAAO,2CAAC,YAAY;EAAgB;EAAQ;GAAgC;;AAG9E,MAAa,gBAAgB;CAC3B,MAAM,gCAAqB,YAAY;AACvC,KAAI,CAAC,QACH,OAAM,IAAI,MAAM,6CAA6C;AAE/D,QAAO"}
200
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"pageContext.cjs","names":["Logger","PageProvider: React.FC<{\n  children: ReactNode;\n  previewMode: boolean;\n  /** When URL resolver returns `ready: false`, invoked for extension/injection failure reporting. */\n  onUrlResolverNotReady?: (error: Error) => Promise<unknown> | void;\n}>","urlResolverAtom","pageVariantInfoAtom","pageUserEventAtom","UserEventCategory","PageVisitCategory","VariantTypeEnum","mapUrlResolverResponseToVariantInfo","mapApiUserEventToUserEvent","CommerceApiClient","useAmplitude","variantInfoAtom","hasParsedVariantInfoAtom","pageType: string","pageId: string","EnviveMetricsEventName"],"sources":["../../../src/contexts/pageContext/pageContext.tsx"],"sourcesContent":["import {\n  ReactNode,\n  createContext,\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react';\nimport {\n  UrlResolverResponse,\n  hasParsedVariantInfoAtom,\n  pageUserEventAtom,\n  pageVariantInfoAtom,\n  urlResolverAtom,\n  variantInfoAtom,\n} from 'src/atoms/app/variant';\nimport { useAtom, useAtomValue } from 'jotai';\nimport CommerceApiClient from 'src/application/commerce-api';\nimport Logger from 'src/application/logging/logger';\nimport { PageVisitCategory, UserEventCategory } from '@spiffy-ai/commerce-api-client';\nimport { VariantTypeEnum } from 'src/application/models';\nimport { EnviveMetricsEventName, useAmplitude } from 'src/contexts/amplitudeContext';\nimport { mapApiUserEventToUserEvent, mapUrlResolverResponseToVariantInfo } from './mapping';\nimport { PageDetails } from './types';\n\nconst logger = new Logger('pageContext');\n\nconst PageContext = createContext<\n  | (PageDetails & {\n      isLoading: boolean;\n      isSupported: boolean;\n      setPageUrl: (url: string) => void;\n    })\n  | undefined\n>(undefined);\n\nexport const PageProvider: React.FC<{\n  children: ReactNode;\n  previewMode: boolean;\n  /** When URL resolver returns `ready: false`, invoked for extension/injection failure reporting. */\n  onUrlResolverNotReady?: (error: Error) => Promise<unknown> | void;\n}> = ({ children, previewMode = false, onUrlResolverNotReady }) => {\n  const [pageUrl, setPageUrl] = useState<string | undefined>(undefined);\n  const [urlResolverResponse, setUrlResolverResponse] = useAtom(urlResolverAtom);\n  const [isLoading, setIsLoading] = useState(false);\n  const [variantInfo, setVariantInfo] = useAtom(pageVariantInfoAtom);\n  const [userEvent, setUserEvent] = useAtom(pageUserEventAtom);\n  const [isSupported, setIsSupported] = useState(false);\n  const lastNotReadyUrlRef = useRef<string | null>(null);\n\n  useEffect(() => {\n    setPageUrl(window.location.href);\n  }, []);\n\n  // Auto-initialize atoms for preview mode\n  useEffect(() => {\n    if (previewMode && !userEvent && !variantInfo) {\n      const url = typeof window !== 'undefined' ? window.location.href : '';\n      setUserEvent({\n        event_id: 'preview-user-event',\n        created_at: new Date().toISOString(),\n        category: UserEventCategory.PageVisit,\n        attributes: {\n          page_visit_category: PageVisitCategory.Other,\n          url,\n        },\n      } as any);\n\n      setVariantInfo({\n        variantType: VariantTypeEnum.PageVisit,\n        url,\n      });\n    }\n  }, [previewMode, userEvent, variantInfo, setUserEvent, setVariantInfo]);\n\n  const setVariantFromUrlResolver = useCallback(\n    (url: string, response: UrlResolverResponse) => {\n      if (response.ready) {\n        const newVariantInfo = mapUrlResolverResponseToVariantInfo(url, response);\n        const newUserEvent = mapApiUserEventToUserEvent(response.user_event ?? undefined);\n        logger.logDebug('setting variant info and user event', { newVariantInfo, newUserEvent });\n        setVariantInfo(newVariantInfo);\n        setUserEvent(newUserEvent);\n      } else {\n        setUserEvent(undefined);\n        setVariantInfo(undefined);\n        logger.logDebug('set variant info and user event to undefined', { variantInfo, userEvent });\n      }\n    },\n    [setVariantInfo, setUserEvent],\n  );\n\n  useEffect(() => {\n    const resolvePageUrl = async () => {\n      setIsLoading(true);\n      try {\n        if (!pageUrl) {\n          setIsLoading(false);\n          return;\n        }\n\n        // If in preview mode or atoms are already set (e.g., by storybook), skip API call\n        if (previewMode || userEvent || variantInfo) {\n          setIsLoading(false);\n          return;\n        }\n\n        const cleansedUrl = pageUrl.toLowerCase().trim();\n        const response =\n          urlResolverResponse[cleansedUrl] ??\n          ((await CommerceApiClient.resolveUrl(cleansedUrl)) as UrlResolverResponse);\n        setVariantFromUrlResolver(cleansedUrl, response);\n        setIsLoading(false);\n        setIsSupported(response.ready);\n\n        if (response.ready) {\n          lastNotReadyUrlRef.current = null;\n        } else if (onUrlResolverNotReady && lastNotReadyUrlRef.current !== cleansedUrl) {\n          lastNotReadyUrlRef.current = cleansedUrl;\n          const err = new Error(`URL resolver returned ready=false for ${cleansedUrl}`);\n          Promise.resolve(onUrlResolverNotReady(err)).catch(callbackError => {\n            logger.logDebug('onUrlResolverNotReady callback failed', callbackError);\n          });\n        }\n      } catch (e) {\n        setIsLoading(false);\n        logger.logError('Failed to resolve page URL', e, { pageUrl });\n      }\n    };\n    resolvePageUrl();\n  }, [\n    pageUrl,\n    urlResolverResponse,\n    setUrlResolverResponse,\n    setVariantFromUrlResolver,\n    userEvent,\n    variantInfo,\n    previewMode,\n    onUrlResolverNotReady,\n  ]);\n\n  const { trackEvent } = useAmplitude();\n  const resolvedVariantInfo = useAtomValue(variantInfoAtom);\n  const hasParsedVariantInfo = useAtomValue(hasParsedVariantInfoAtom);\n  const pageViewedFired = useRef(false);\n\n  useEffect(() => {\n    if (pageViewedFired.current || isLoading || !hasParsedVariantInfo || !variantInfo) {\n      return;\n    }\n\n    let pageType: string;\n    let pageId: string;\n\n    switch (variantInfo.variantType) {\n      case VariantTypeEnum.Pdp:\n        pageType = 'pdp';\n        pageId = variantInfo.productId;\n        break;\n      case VariantTypeEnum.Plp:\n        pageType = 'plp';\n        pageId = variantInfo.plpId;\n        break;\n      case VariantTypeEnum.Home:\n        pageType = 'homepage';\n        pageId = variantInfo.url;\n        break;\n      case VariantTypeEnum.Other:\n        pageType = 'other';\n        pageId = variantInfo.url;\n        break;\n      case VariantTypeEnum.PageVisit:\n        pageType = 'other';\n        pageId = variantInfo.url;\n        break;\n      case VariantTypeEnum.FullPageSalesAgent:\n        pageType = 'full_page_sales_agent';\n        pageId = variantInfo.url;\n        break;\n      default:\n        return;\n    }\n\n    pageViewedFired.current = true;\n\n    trackEvent({\n      eventName: EnviveMetricsEventName.PageViewed,\n      eventProps: {\n        'context.page_type': pageType,\n        'context.page_id': pageId,\n        'context.supported': isSupported,\n        'context.ready': isSupported,\n        'context.page_variant_id': resolvedVariantInfo.variantId,\n      },\n    });\n  }, [isLoading, isSupported, hasParsedVariantInfo, variantInfo, resolvedVariantInfo, trackEvent]);\n\n  const setPageUrlStable = useCallback((url: string) => {\n    setPageUrl(url);\n  }, []);\n\n  const value = useMemo(\n    () => ({\n      pageUrl,\n      variantInfo,\n      userEvent,\n      isLoading,\n      isSupported,\n      setPageUrl: setPageUrlStable,\n    }),\n    [pageUrl, userEvent, variantInfo, isLoading, isSupported, setPageUrlStable],\n  );\n\n  return <PageContext.Provider value={value}>{children}</PageContext.Provider>;\n};\n\nexport const usePage = () => {\n  const context = useContext(PageContext);\n  if (!context) {\n    throw new Error('usePage must be used within a PageProvider');\n  }\n  return context;\n};\n"],"mappings":"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"}
@@ -5,6 +5,8 @@ import { ReactNode } from "react";
5
5
  declare const PageProvider: React.FC<{
6
6
  children: ReactNode;
7
7
  previewMode: boolean;
8
+ /** When URL resolver returns `ready: false`, invoked for extension/injection failure reporting. */
9
+ onUrlResolverNotReady?: (error: Error) => Promise<unknown> | void;
8
10
  }>;
9
11
  declare const usePage: () => PageDetails & {
10
12
  isLoading: boolean;
@@ -13,4 +15,4 @@ declare const usePage: () => PageDetails & {
13
15
  };
14
16
  //#endregion
15
17
  export { PageProvider, usePage };
16
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFnZUNvbnRleHQuZC5jdHMiLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2NvbnRleHRzL3BhZ2VDb250ZXh0L3BhZ2VDb250ZXh0LmQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUmVhY3ROb2RlIH0gZnJvbSAncmVhY3QnO1xuaW1wb3J0IHsgUGFnZURldGFpbHMgfSBmcm9tICcuL3R5cGVzJztcbmV4cG9ydCBkZWNsYXJlIGNvbnN0IFBhZ2VQcm92aWRlcjogUmVhY3QuRkM8e1xuICAgIGNoaWxkcmVuOiBSZWFjdE5vZGU7XG4gICAgcHJldmlld01vZGU6IGJvb2xlYW47XG59PjtcbmV4cG9ydCBkZWNsYXJlIGNvbnN0IHVzZVBhZ2U6ICgpID0+IFBhZ2VEZXRhaWxzICYge1xuICAgIGlzTG9hZGluZzogYm9vbGVhbjtcbiAgICBpc1N1cHBvcnRlZDogYm9vbGVhbjtcbiAgICBzZXRQYWdlVXJsOiAodXJsOiBzdHJpbmcpID0+IHZvaWQ7XG59O1xuIl0sIm1hcHBpbmdzIjoiOzs7O0FBRUEsSUFBVyxlQUFlO0NBQUM7T0FBVztPQUFNLE1BQUE7Q0FBQTtBQUM1QyxJQUFXLFVBQVUsQ0FBQyxXQUFDLFlBQUEifQ==
18
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFnZUNvbnRleHQuZC5jdHMiLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2NvbnRleHRzL3BhZ2VDb250ZXh0L3BhZ2VDb250ZXh0LmQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUmVhY3ROb2RlIH0gZnJvbSAncmVhY3QnO1xuaW1wb3J0IHsgUGFnZURldGFpbHMgfSBmcm9tICcuL3R5cGVzJztcbmV4cG9ydCBkZWNsYXJlIGNvbnN0IFBhZ2VQcm92aWRlcjogUmVhY3QuRkM8e1xuICAgIGNoaWxkcmVuOiBSZWFjdE5vZGU7XG4gICAgcHJldmlld01vZGU6IGJvb2xlYW47XG4gICAgLyoqIFdoZW4gVVJMIHJlc29sdmVyIHJldHVybnMgYHJlYWR5OiBmYWxzZWAsIGludm9rZWQgZm9yIGV4dGVuc2lvbi9pbmplY3Rpb24gZmFpbHVyZSByZXBvcnRpbmcuICovXG4gICAgb25VcmxSZXNvbHZlck5vdFJlYWR5PzogKGVycm9yOiBFcnJvcikgPT4gUHJvbWlzZTx1bmtub3duPiB8IHZvaWQ7XG59PjtcbmV4cG9ydCBkZWNsYXJlIGNvbnN0IHVzZVBhZ2U6ICgpID0+IFBhZ2VEZXRhaWxzICYge1xuICAgIGlzTG9hZGluZzogYm9vbGVhbjtcbiAgICBpc1N1cHBvcnRlZDogYm9vbGVhbjtcbiAgICBzZXRQYWdlVXJsOiAodXJsOiBzdHJpbmcpID0+IHZvaWQ7XG59O1xuIl0sIm1hcHBpbmdzIjoiOzs7O0FBRUEsSUFBVyxlQUFlO0NBQUM7T0FBVztPQUFNO09BQUE7T0FBQSxNQUFBO0NBQUE7QUFDNUMsSUFBVyxVQUFVLENBQUMsV0FBQyxZQUFBIn0=