@opencx/widget 3.0.52 → 3.0.55

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.
@@ -1,15 +1,15 @@
1
- import O from "openapi-fetch";
2
- import F from "lodash.isequal";
1
+ import M from "openapi-fetch";
2
+ import O from "lodash.isequal";
3
3
  import { v4 as A } from "uuid";
4
- const L = (g) => {
4
+ const F = (g) => {
5
5
  console.log(g.error);
6
- }, B = (g) => {
7
- const n = O({
6
+ }, _ = (g) => {
7
+ const n = M({
8
8
  baseUrl: g.baseUrl
9
9
  }), i = {
10
10
  onRequest: g.onRequest,
11
11
  onResponse: g.onResponse,
12
- onError: g.onError || L
12
+ onError: g.onError || F
13
13
  };
14
14
  return n.use(i), n;
15
15
  };
@@ -29,7 +29,7 @@ class E {
29
29
  }, this.createOpenAPIClient = ({
30
30
  baseUrl: t,
31
31
  headers: s
32
- }) => B({
32
+ }) => _({
33
33
  baseUrl: t,
34
34
  onRequest: ({ request: e }) => {
35
35
  Object.entries(s).forEach(([o, r]) => {
@@ -115,10 +115,10 @@ class E {
115
115
  this.client = this.createOpenAPIClient({ baseUrl: i, headers: h });
116
116
  }
117
117
  }
118
- class b {
118
+ class w {
119
119
  constructor(n) {
120
120
  this.subscribers = /* @__PURE__ */ new Set(), this.get = () => this.state, this.set = (i) => {
121
- F(this.state, i) || (this.state = i, this.notifySubscribers(i));
121
+ O(this.state, i) || (this.state = i, this.notifySubscribers(i));
122
122
  }, this.setPartial = (i) => {
123
123
  if (i == null) return;
124
124
  const h = { ...this.state, ...i };
@@ -140,7 +140,7 @@ class b {
140
140
  }
141
141
  class T {
142
142
  constructor() {
143
- this.state = new b({
143
+ this.state = new w({
144
144
  isPolling: !1,
145
145
  isError: !1
146
146
  }), this.abortController = new AbortController(), this.reset = () => {
@@ -175,7 +175,7 @@ function D(g) {
175
175
  return { error: n };
176
176
  }
177
177
  }
178
- class _ {
178
+ class L {
179
179
  constructor({
180
180
  api: n,
181
181
  config: i,
@@ -265,7 +265,7 @@ class _ {
265
265
  }, this.api = n, this.config = i, this.sessionCtx = h, this.messageCtx = a, this.sessionPollingIntervalSeconds = t, this.registerPolling();
266
266
  }
267
267
  }
268
- class q {
268
+ class B {
269
269
  constructor({
270
270
  config: n,
271
271
  api: i,
@@ -276,25 +276,26 @@ class q {
276
276
  var t;
277
277
  return !!(!((t = this.state.get().contact) != null && t.token) && this.config.collectUserData);
278
278
  }, this.autoCreateUnverifiedUserIfNotExists = async () => {
279
- var t, s, e, o, r, l, c, f, m, C, d, x;
279
+ var t, s, e, o, r, l, c, f, m, C, d, x, P, S;
280
280
  if (!((t = this.config.user) != null && t.token)) {
281
281
  if (this.config.collectUserData && !((e = (s = this.config.user) == null ? void 0 : s.data) != null && e.email)) {
282
282
  if ((o = this.config.extraDataCollectionFields) != null && o.length)
283
283
  return;
284
- const S = await ((r = this.storageCtx) == null ? void 0 : r.getContactToken());
285
- S && await this.setUnverifiedContact(S);
284
+ const v = await ((r = this.storageCtx) == null ? void 0 : r.getContactToken());
285
+ v && await this.setUnverifiedContact(v);
286
286
  return;
287
287
  }
288
288
  if (!((c = (l = this.config.user) == null ? void 0 : l.data) != null && c.email)) {
289
- const S = await ((f = this.storageCtx) == null ? void 0 : f.getContactToken());
290
- if (S) {
291
- await this.setUnverifiedContact(S);
289
+ const v = await ((f = this.storageCtx) == null ? void 0 : f.getContactToken());
290
+ if (v) {
291
+ await this.setUnverifiedContact(v);
292
292
  return;
293
293
  }
294
294
  }
295
295
  await this.createUnverifiedContact({
296
- name: ((C = (m = this.config.user) == null ? void 0 : m.data) == null ? void 0 : C.name) || "Anonymous",
297
- email: (x = (d = this.config.user) == null ? void 0 : d.data) == null ? void 0 : x.email
296
+ email: (C = (m = this.config.user) == null ? void 0 : m.data) == null ? void 0 : C.email,
297
+ non_verified_name: ((x = (d = this.config.user) == null ? void 0 : d.data) == null ? void 0 : x.name) || "Anonymous",
298
+ non_verified_custom_data: (S = (P = this.config.user) == null ? void 0 : P.data) == null ? void 0 : S.customData
298
299
  });
299
300
  }
300
301
  }, this.createUnverifiedContact = async (t, s) => {
@@ -313,7 +314,7 @@ class q {
313
314
  var o, r, l, c;
314
315
  const s = await ((o = this.storageCtx) == null ? void 0 : o.getExternalContactId()), e = ((r = this.config.user) == null ? void 0 : r.externalId) || s || A();
315
316
  this.api.setAuthToken(t), await ((l = this.storageCtx) == null ? void 0 : l.setContactToken(t)), await ((c = this.storageCtx) == null ? void 0 : c.setExternalContactId(e)), this.state.setPartial({ contact: { token: t, externalId: e } });
316
- }, this.config = n, this.storageCtx = h, this.api = i, this.state = new b({
317
+ }, this.config = n, this.storageCtx = h, this.api = i, this.state = new w({
317
318
  contact: (a = n.user) != null && a.token ? {
318
319
  token: n.user.token,
319
320
  // Set optional externalId from config... not local storage
@@ -325,19 +326,19 @@ class q {
325
326
  }), this.autoCreateUnverifiedUserIfNotExists();
326
327
  }
327
328
  }
328
- function I() {
329
+ function y() {
329
330
  return A();
330
331
  }
331
- class $ {
332
+ class q {
332
333
  constructor({
333
334
  api: n,
334
335
  contactCtx: i,
335
336
  sessionsPollingIntervalSeconds: h
336
337
  }) {
337
- this.sessionsRefresher = new T(), this.sessionState = new b({
338
+ this.sessionsRefresher = new T(), this.sessionState = new w({
338
339
  session: null,
339
340
  isCreatingSession: !1
340
- }), this.sessionsState = new b({
341
+ }), this.sessionsState = new w({
341
342
  data: [],
342
343
  cursor: void 0,
343
344
  isLastPage: !1,
@@ -402,14 +403,14 @@ class $ {
402
403
  }, this.api = n, this.contactCtx = i, this.sessionsPollingIntervalSeconds = h, this.registerSessionsRefresherWrapper();
403
404
  }
404
405
  }
405
- class N {
406
+ class $ {
406
407
  constructor({
407
408
  config: n,
408
409
  api: i,
409
410
  sessionCtx: h,
410
411
  contactCtx: a
411
412
  }) {
412
- this.state = new b({
413
+ this.state = new w({
413
414
  messages: [],
414
415
  isSendingMessage: !1,
415
416
  lastAIResMightSolveUserIssue: !1,
@@ -417,7 +418,7 @@ class N {
417
418
  }), this.sendMessageAbortController = new AbortController(), this.reset = () => {
418
419
  this.sendMessageAbortController.abort("Resetting chat"), this.state.reset();
419
420
  }, this.sendMessage = async (t) => {
420
- var l, c, f, m, C, d, x, S, y;
421
+ var l, c, f, m, C, d, x, P;
421
422
  if (!t.content.trim() && (!t.attachments || t.attachments.length === 0)) {
422
423
  console.warn("Cannot send an empty message of no content or attachments");
423
424
  return;
@@ -431,12 +432,12 @@ class N {
431
432
  this.sendMessageAbortController = new AbortController(), this.state.setPartial({ lastAIResMightSolveUserIssue: !1 });
432
433
  try {
433
434
  this.state.setPartial({ isSendingMessage: !0 });
434
- const w = this.toUserMessage(
435
+ const S = this.toUserMessage(
435
436
  t.content.trim(),
436
437
  t.attachments || void 0
437
- ), R = this.state.get().messages;
438
+ ), v = this.state.get().messages;
438
439
  if (this.state.setPartial({
439
- messages: [...R, w]
440
+ messages: [...v, S]
440
441
  }), !((c = this.sessionCtx.sessionState.get().session) != null && c.id)) {
441
442
  if (!await this.sessionCtx.createSession()) {
442
443
  console.error("Failed to create session");
@@ -448,45 +449,44 @@ class N {
448
449
  if (!k) return;
449
450
  const { data: u } = await this.api.sendMessage(
450
451
  {
451
- uuid: w.id,
452
+ uuid: S.id,
452
453
  bot_token: this.config.token,
453
454
  headers: this.config.headers,
454
455
  query_params: this.config.queryParams,
455
456
  session_id: k,
456
- user: (m = this.config.user) == null ? void 0 : m.data,
457
- content: w.content,
457
+ content: S.content,
458
458
  attachments: t.attachments,
459
459
  clientContext: this.config.context
460
460
  },
461
461
  this.sendMessageAbortController.signal
462
462
  );
463
463
  if (u != null && u.success) {
464
- const v = this.toBotMessage(u);
465
- if (v) {
466
- const P = this.state.get().messages;
467
- if (!!P.some(
468
- (M) => M.id === v.id
464
+ const b = this.toBotMessage(u);
465
+ if (b) {
466
+ const I = this.state.get().messages;
467
+ if (!!I.some(
468
+ (R) => R.id === b.id
469
469
  )) {
470
470
  this.state.setPartial({
471
- lastAIResMightSolveUserIssue: ((C = u.autopilotResponse) == null ? void 0 : C.mightSolveUserIssue) || ((d = u.uiResponse) == null ? void 0 : d.mightSolveUserIssue)
471
+ lastAIResMightSolveUserIssue: ((m = u.autopilotResponse) == null ? void 0 : m.mightSolveUserIssue) || ((C = u.uiResponse) == null ? void 0 : C.mightSolveUserIssue)
472
472
  });
473
473
  return;
474
474
  }
475
475
  this.state.setPartial({
476
- messages: [...P, v],
477
- lastAIResMightSolveUserIssue: ((x = u.autopilotResponse) == null ? void 0 : x.mightSolveUserIssue) || ((S = u.uiResponse) == null ? void 0 : S.mightSolveUserIssue)
476
+ messages: [...I, b],
477
+ lastAIResMightSolveUserIssue: ((d = u.autopilotResponse) == null ? void 0 : d.mightSolveUserIssue) || ((x = u.uiResponse) == null ? void 0 : x.mightSolveUserIssue)
478
478
  });
479
479
  }
480
480
  } else {
481
- const v = this.toBotErrorMessage(
482
- ((y = u == null ? void 0 : u.error) == null ? void 0 : y.message) || "Unknown error occurred"
483
- ), P = this.state.get().messages;
481
+ const b = this.toBotErrorMessage(
482
+ ((P = u == null ? void 0 : u.error) == null ? void 0 : P.message) || "Unknown error occurred"
483
+ ), I = this.state.get().messages;
484
484
  this.state.setPartial({
485
- messages: [...P, v]
485
+ messages: [...I, b]
486
486
  });
487
487
  }
488
- } catch (w) {
489
- this.sendMessageAbortController.signal.aborted || console.error("Failed to send message:", w);
488
+ } catch (S) {
489
+ this.sendMessageAbortController.signal.aborted || console.error("Failed to send message:", S);
490
490
  } finally {
491
491
  this.state.setPartial({ isSendingMessage: !1 });
492
492
  }
@@ -499,7 +499,7 @@ class N {
499
499
  ${t}` : t;
500
500
  })();
501
501
  return {
502
- id: I(),
502
+ id: y(),
503
503
  type: "FROM_USER",
504
504
  content: e,
505
505
  deliveredAt: (/* @__PURE__ */ new Date()).toISOString(),
@@ -510,7 +510,7 @@ ${t}` : t;
510
510
  var s;
511
511
  return t.success && t.autopilotResponse ? {
512
512
  type: "FROM_BOT",
513
- id: t.autopilotResponse.id || I(),
513
+ id: t.autopilotResponse.id || y(),
514
514
  timestamp: (/* @__PURE__ */ new Date()).toISOString(),
515
515
  component: "bot_message",
516
516
  agent: this.config.bot ? {
@@ -529,7 +529,7 @@ ${t}` : t;
529
529
  } : null;
530
530
  }, this.toBotErrorMessage = (t) => ({
531
531
  type: "FROM_BOT",
532
- id: I(),
532
+ id: y(),
533
533
  timestamp: (/* @__PURE__ */ new Date()).toISOString(),
534
534
  component: "TEXT",
535
535
  data: {
@@ -540,7 +540,7 @@ ${t}` : t;
540
540
  }), this.config = n, this.api = i, this.sessionCtx = h, this.contactCtx = a;
541
541
  }
542
542
  }
543
- class H {
543
+ class N {
544
544
  constructor({
545
545
  config: n,
546
546
  contactCtx: i,
@@ -565,12 +565,12 @@ class H {
565
565
  this.sessionCtx.sessionState.setPartial({ session: s });
566
566
  }
567
567
  this.state.setPartial({ screen: "chat" });
568
- }, this.state = new b({
568
+ }, this.state = new w({
569
569
  screen: i.shouldCollectData() ? "welcome" : "sessions"
570
570
  }), this.config = n, this.contactCtx = i, this.sessionCtx = h, this.resetChat = a, this.registerRoutingListener();
571
571
  }
572
572
  }
573
- class X {
573
+ class H {
574
574
  constructor({ storage: n }) {
575
575
  this.KEYS = {
576
576
  contactToken: "opencx__widget__contactToken",
@@ -593,26 +593,26 @@ const p = class p {
593
593
  throw Error(
594
594
  "Widget polling values are not defined, did you call WidgetCtx.initialize()"
595
595
  );
596
- this.config = n, this.api = new E({ config: n }), this.storageCtx = i ? new X({ storage: i }) : void 0, this.contactCtx = new q({
596
+ this.config = n, this.api = new E({ config: n }), this.storageCtx = i ? new H({ storage: i }) : void 0, this.contactCtx = new B({
597
597
  api: this.api,
598
598
  config: this.config,
599
599
  storageCtx: this.storageCtx
600
- }), this.sessionCtx = new $({
600
+ }), this.sessionCtx = new q({
601
601
  api: this.api,
602
602
  contactCtx: this.contactCtx,
603
603
  sessionsPollingIntervalSeconds: p.pollingIntervalsSeconds.sessions
604
- }), this.messageCtx = new N({
604
+ }), this.messageCtx = new $({
605
605
  config: this.config,
606
606
  api: this.api,
607
607
  sessionCtx: this.sessionCtx,
608
608
  contactCtx: this.contactCtx
609
- }), this.activeSessionPollingCtx = new _({
609
+ }), this.activeSessionPollingCtx = new L({
610
610
  api: this.api,
611
611
  config: this.config,
612
612
  sessionCtx: this.sessionCtx,
613
613
  messageCtx: this.messageCtx,
614
614
  sessionPollingIntervalSeconds: p.pollingIntervalsSeconds.session
615
- }), this.routerCtx = new H({
615
+ }), this.routerCtx = new N({
616
616
  config: this.config,
617
617
  contactCtx: this.contactCtx,
618
618
  sessionCtx: this.sessionCtx,
@@ -638,7 +638,7 @@ p.pollingIntervalsSeconds = null, p.initialize = async ({
638
638
  };
639
639
  let U = p;
640
640
  export {
641
- b as P,
641
+ w as P,
642
642
  U as W
643
643
  };
644
- //# sourceMappingURL=widget.ctx-DXznvVMa.js.map
644
+ //# sourceMappingURL=widget.ctx-BQOGEEyO.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"widget.ctx-BQOGEEyO.js","sources":["../src/headless/core/api/client.ts","../src/headless/core/api/api-caller.ts","../src/headless/core/utils/PrimitiveState.ts","../src/headless/core/utils/Poller.ts","../src/headless/core/utils/run-catching.ts","../src/headless/core/context/active-session-polling.ctx.ts","../src/headless/core/context/contact.ctx.ts","../src/headless/core/utils/uuid.ts","../src/headless/core/context/session.ctx.ts","../src/headless/core/context/message.ctx.ts","../src/headless/core/context/router.ctx.ts","../src/headless/core/context/storage.ctx.ts","../src/headless/core/context/widget.ctx.ts"],"sourcesContent":["import createClient, { type Middleware } from \"openapi-fetch\";\nimport type { paths } from \"./schema\";\nimport type { components } from \"./schema\";\n\ntype Options = {\n baseUrl: string;\n onRequest?: Middleware[\"onRequest\"];\n onResponse?: Middleware[\"onResponse\"];\n onError?: Middleware[\"onError\"];\n};\n\nconst defaultOnError: Middleware[\"onError\"] = (onErrorOptions) => {\n console.log(onErrorOptions.error);\n};\n\nexport const basicClient = (options: Options) => {\n const client = createClient<paths>({\n baseUrl: options.baseUrl,\n });\n\n const middlewares: Middleware = {\n onRequest: options.onRequest,\n onResponse: options.onResponse,\n onError: options.onError || defaultOnError,\n };\n\n client.use(middlewares);\n return client;\n};\n\nexport type Endpoint = keyof paths;\nexport type Dto = components[\"schemas\"];\n","import { type Dto, type Endpoint, basicClient } from \"./client\";\nimport type { WidgetConfig } from \"../types/widget-config\";\nimport type { SendMessageDto, VoteInputDto } from \"../types/schemas\";\n\nexport class ApiCaller {\n private client: ReturnType<typeof basicClient>;\n private config: WidgetConfig;\n private userToken: string | null = null;\n\n constructor({\n config,\n }: {\n config: WidgetConfig;\n }) {\n this.config = config;\n const { baseUrl, headers } = this.constructClientOptions(\n config.user?.token,\n );\n this.client = this.createOpenAPIClient({ baseUrl, headers });\n }\n\n private constructClientOptions = (token: string | null | undefined) => {\n const baseUrl =\n import.meta.env.MODE === \"test\"\n ? \"http://localhost:8080\"\n : this.config.apiUrl || \"https://api.open.cx\";\n const headers = {\n \"X-Bot-Token\": this.config.token,\n \"Content-Type\": \"application/json\",\n Accept: \"application/json\",\n Authorization: token ? `Bearer ${token}` : undefined,\n };\n\n return { baseUrl, headers };\n };\n\n private createOpenAPIClient = ({\n baseUrl,\n headers,\n }: ReturnType<typeof this.constructClientOptions>) => {\n return basicClient({\n baseUrl,\n onRequest: ({ request }) => {\n Object.entries(headers).forEach(([key, value]) => {\n if (value) {\n request.headers.set(key, value);\n }\n });\n },\n });\n };\n\n setAuthToken = (token: string) => {\n this.userToken = token;\n const { baseUrl, headers } = this.constructClientOptions(token);\n this.client = this.createOpenAPIClient({ baseUrl, headers });\n };\n\n getExternalWidgetConfig = async () => {\n return await this.client.GET(\"/backend/widget/v2/config\", {\n params: { header: { \"X-Bot-Token\": this.config.token } },\n });\n };\n\n widgetPrelude = async () => {\n return await this.client.GET(\"/backend/widget/v2/prelude\", {\n params: { header: { \"X-Bot-Token\": this.config.token } },\n });\n };\n\n sendMessage = async (body: SendMessageDto, abortSignal?: AbortSignal) => {\n return await this.client.POST(\"/backend/widget/v2/chat/send\", {\n body,\n signal: abortSignal,\n });\n };\n\n createUnverifiedContact = async (body: Dto[\"CreateUnverifiedContactDto\"]) => {\n return await this.client.POST(\n \"/backend/widget/v2/contact/create-unverified\",\n {\n params: { header: { \"x-bot-token\": this.config.token } },\n body,\n },\n );\n };\n\n createSession = async (body: Dto[\"CreateWidgetSessionDto\"]) => {\n return await this.client.POST(\"/backend/widget/v2/create-session\", {\n body,\n });\n };\n\n pollSessionAndHistory = async ({\n sessionId,\n lastMessageTimestamp,\n abortSignal,\n }: {\n sessionId: string;\n lastMessageTimestamp?: string;\n abortSignal: AbortSignal;\n }) => {\n const query = lastMessageTimestamp ? { lastMessageTimestamp } : undefined;\n return await this.client.GET(\"/backend/widget/v2/poll/{sessionId}\", {\n params: { path: { sessionId }, query },\n signal: abortSignal,\n });\n };\n\n getSessions = async ({\n cursor,\n filters,\n abortSignal,\n }: {\n cursor: string | undefined;\n filters: Record<string, string>;\n abortSignal?: AbortSignal;\n }) => {\n return await this.client.GET(\"/backend/widget/v2/sessions\", {\n params: { query: { cursor, filters: JSON.stringify(filters) } },\n signal: abortSignal,\n });\n };\n\n /**\n * openapi-fetch usually works fine for file uploads, but this time around it parses the payload in a weird way and results in 413 errors (payload too large)\n * Anyway, good old XHR even does it better with progress events\n */\n uploadFile = async ({\n file,\n abortSignal,\n onProgress,\n }: {\n file: File;\n abortSignal: AbortSignal;\n onProgress?: (percentage: number) => void;\n }): Promise<Dto[\"UploadWidgetFileResponseDto\"]> => {\n return new Promise((resolve, reject) => {\n const formData = new FormData();\n formData.append(\"file\", file);\n\n const xhr = new XMLHttpRequest();\n\n // Set up abort functionality\n if (abortSignal) {\n abortSignal.addEventListener(\"abort\", () => {\n xhr.abort();\n reject(new DOMException(\"Aborted\", \"AbortError\"));\n });\n\n // If already aborted, reject immediately\n if (abortSignal.aborted) {\n reject(new DOMException(\"Aborted\", \"AbortError\"));\n return;\n }\n }\n\n xhr.upload.addEventListener(\"progress\", (event) => {\n if (event.lengthComputable && onProgress) {\n const percentage = Math.round((event.loaded / event.total) * 100);\n onProgress(percentage);\n }\n });\n\n // Handle completion\n xhr.addEventListener(\"load\", () => {\n if (xhr.status >= 200 && xhr.status < 300) {\n try {\n const data = JSON.parse(xhr.responseText);\n resolve(data);\n } catch (error) {\n reject(new Error(`Failed to parse response: ${error}`));\n }\n } else {\n reject(new Error(`Upload failed with status: ${xhr.status}`));\n }\n });\n\n xhr.addEventListener(\"error\", () => {\n reject(new Error(\"Network error occurred\"));\n });\n\n xhr.addEventListener(\"timeout\", () => {\n reject(new Error(\"Upload timed out\"));\n });\n\n const { baseUrl } = this.constructClientOptions(this.userToken);\n\n const path = \"/backend/widget/v2/upload\" satisfies Endpoint\n const uploadUrl = `${baseUrl}${path}`;\n xhr.open(\"POST\", uploadUrl);\n\n xhr.setRequestHeader(\"X-Bot-Token\", this.config.token);\n if (this.userToken) {\n xhr.setRequestHeader(\"Authorization\", `Bearer ${this.userToken}`);\n }\n\n xhr.send(formData);\n });\n };\n\n vote = async (body: VoteInputDto) => {\n return await this.client.POST(\"/backend/widget/v2/chat/vote\", { body });\n };\n}\n","import isEqual from \"lodash.isequal\";\n\nexport type Subscriber<T> = (data: T) => void;\n\nexport class PrimitiveState<S> {\n private subscribers = new Set<Subscriber<S>>();\n private state: S;\n private initialState: S;\n\n constructor(state: S) {\n this.state = state;\n this.initialState = state;\n }\n\n get = (): S => this.state;\n\n set = (newState: S): void => {\n if (!isEqual(this.state, newState)) {\n this.state = newState;\n this.notifySubscribers(newState);\n }\n };\n\n setPartial = (_s: Partial<S>): void => {\n if (_s === undefined || _s === null) return;\n const newState = { ...this.state, ..._s };\n this.set(newState);\n };\n\n reset = (): void => {\n this.set(this.initialState);\n };\n\n private notifySubscribers = (state: S) => {\n const subscribersArray = Array.from(this.subscribers);\n subscribersArray.forEach((callback) => {\n try {\n callback(state);\n } catch (error) {\n if (import.meta.env.MODE !== \"test\") {\n console.error(error);\n }\n }\n });\n };\n\n subscribe = (callback: Subscriber<S>): (() => void) => {\n this.subscribers.add(callback);\n\n return () => {\n this.subscribers.delete(callback);\n };\n };\n}\n","import { PrimitiveState } from \"./PrimitiveState\";\n\nexport type PollingState = {\n isPolling: boolean;\n isError: boolean;\n};\n\nexport class Poller {\n state = new PrimitiveState<PollingState>({\n isPolling: false,\n isError: false,\n });\n private abortController = new AbortController();\n\n reset = () => {\n this.abortController.abort(\"Resetting poller\");\n this.stopPolling?.();\n this.stopPolling = null;\n };\n\n private stopPolling: (() => void) | null = null;\n\n startPolling = (\n cb: (abortSignal: AbortSignal) => Promise<void>,\n intervalMs: number,\n ) => {\n if (this.stopPolling) return;\n\n const timeouts: NodeJS.Timeout[] = [];\n\n const poll = async () => {\n this.abortController = new AbortController();\n this.state.setPartial({ isPolling: true });\n\n try {\n await cb(this.abortController.signal);\n } catch (error) {\n if (this.abortController.signal.aborted) {\n // If aborted, just return and do not schedule the nest poll\n return;\n }\n console.error(\"Failed to poll:\", error);\n this.state.setPartial({ isError: true });\n } finally {\n this.state.setPartial({ isPolling: false });\n }\n\n // Another check to stop scheduling polls in case someone removes the early return in the catch above\n if (this.abortController.signal.aborted) {\n console.log(\"Poller aborted, not scheduling anymore\");\n } else {\n timeouts.push(setTimeout(poll, intervalMs));\n }\n };\n\n poll();\n\n this.stopPolling = () => {\n timeouts.forEach(clearTimeout);\n this.state.reset();\n };\n };\n}\n","export type Result<T, E> =\n | { data: T; error?: undefined }\n | { data?: undefined; error: E };\n\n/**\n * Many thanks to @m-tabaza for this utility... Kotlin vibes\n */\nexport function runCatching<T, E = unknown>(\n callback: () => Promise<T>,\n): Promise<Result<T, E>>;\nexport function runCatching<T, E = unknown>(callback: () => T): Result<T, E>;\nexport function runCatching<T, E = unknown>(\n callback: () => T | Promise<T>,\n): Result<T, E> | Promise<Result<T, E>> {\n try {\n const result = callback();\n\n if (result instanceof Promise) {\n return result.then((data) => ({ data })).catch((error: E) => ({ error }));\n }\n\n return { data: result };\n } catch (error) {\n return { error: error as E };\n }\n}\n","import type { ApiCaller } from \"../api/api-caller\";\nimport type { MessageType } from \"../types/messages\";\nimport type { ActionCallDto, MessageDto } from \"../types/schemas\";\nimport type { WidgetConfig } from \"../types/widget-config\";\nimport { Poller } from \"../utils/Poller\";\nimport { runCatching } from \"../utils/run-catching\";\nimport type { MessageCtx } from \"./message.ctx\";\nimport type { SessionCtx } from \"./session.ctx\";\n\nexport class ActiveSessionPollingCtx {\n private api: ApiCaller;\n private config: WidgetConfig;\n private sessionCtx: SessionCtx;\n private messageCtx: MessageCtx;\n private sessionPollingIntervalSeconds: number;\n\n private poller = new Poller();\n\n constructor({\n api,\n config,\n sessionCtx,\n messageCtx,\n sessionPollingIntervalSeconds,\n }: {\n api: ApiCaller;\n config: WidgetConfig;\n sessionCtx: SessionCtx;\n messageCtx: MessageCtx;\n sessionPollingIntervalSeconds: number;\n }) {\n this.api = api;\n this.config = config;\n this.sessionCtx = sessionCtx;\n this.messageCtx = messageCtx;\n this.sessionPollingIntervalSeconds = sessionPollingIntervalSeconds;\n\n this.registerPolling();\n }\n\n private registerPolling = () => {\n this.sessionCtx.sessionState.subscribe(({ session }) => {\n if (session?.id) {\n this.poller.startPolling(async (abortSignal) => {\n this.hackAndSlash(session.id, abortSignal);\n }, this.sessionPollingIntervalSeconds * 1000);\n } else {\n this.poller.reset();\n }\n });\n };\n\n private hackAndSlash = async (\n sessionId: string,\n abortSignal: AbortSignal,\n ): Promise<void> => {\n /**\n * This is a bit of an implicit contract... there are two cases here\n * 1. If there are no messages in state, it means the user selected a previous session from the sessions screen and got routed to the chat,\n * in this case, we want to show a loading indicator until the initial fetch is done\n * 2. There is a single message in state, which is the optimistically rendered user message,\n * in this case, we don't want to show a loading indicator\n */\n if (this.messageCtx.state.get().messages.length === 0) {\n this.messageCtx.state.setPartial({ isInitialFetchLoading: true });\n }\n\n const messages = this.messageCtx.state.get().messages;\n const lastMessageTimestamp =\n messages.length > 0\n ? messages[messages.length - 1]?.timestamp\n : undefined;\n\n const { data } = await this.api.pollSessionAndHistory({\n sessionId,\n abortSignal,\n lastMessageTimestamp,\n });\n\n if (data?.session) {\n this.sessionCtx.sessionState.setPartial({ session: data.session });\n }\n\n if (data?.history && data.history.length > 0) {\n // Get a fresh reference to current messages after the poll is done\n const prevMessages = this.messageCtx.state.get().messages;\n const newMessages = data.history\n .map(this.mapHistoryToMessage)\n .filter(\n (newMsg) =>\n !prevMessages.some((existingMsg) => existingMsg.id === newMsg.id),\n );\n this.messageCtx.state.setPartial({\n messages: [...prevMessages, ...newMessages],\n });\n }\n\n if (this.messageCtx.state.get().isInitialFetchLoading) {\n this.messageCtx.state.setPartial({ isInitialFetchLoading: false });\n }\n };\n\n mapHistoryToMessage = (history: MessageDto): MessageType => {\n const commonFields = {\n id: history.publicId,\n timestamp: history.sentAt || \"\",\n attachments: history.attachments || undefined,\n };\n\n if (history.sender.kind === \"user\") {\n return {\n ...commonFields,\n type: \"FROM_USER\",\n content: history.content.text || \"\",\n deliveredAt: history.sentAt || \"\",\n };\n }\n\n if (history.sender.kind === \"agent\") {\n return {\n ...commonFields,\n type: \"FROM_AGENT\",\n component: \"agent_message\",\n data: {\n message: history.content.text || \"\",\n },\n agent: {\n name: history.sender.name || \"\",\n avatar: history.sender.avatar || \"\",\n id: null,\n isAi: false,\n },\n };\n }\n\n const action =\n history.actionCalls && history.actionCalls.length > 0\n ? history.actionCalls[history.actionCalls.length - 1]\n : undefined;\n\n return {\n ...commonFields,\n type: \"FROM_BOT\",\n component: \"bot_message\",\n agent: {\n id: null,\n name: this.config.bot?.name || \"\",\n isAi: true,\n avatar: this.config.bot?.avatar || \"\",\n },\n data: {\n message: history.content.text || \"\",\n action: action\n ? {\n name: action.actionName,\n data: this.extractActionResult(action),\n }\n : undefined,\n },\n };\n };\n\n extractActionResult = (action: ActionCallDto) => {\n const result = action.result;\n\n if (result === null) return result;\n if (typeof result !== \"object\") return result;\n\n if (\n \"responseBodyText\" in result &&\n typeof result.responseBodyText === \"string\"\n ) {\n const responseBodyText = result.responseBodyText;\n const parsed = runCatching(() => JSON.parse(responseBodyText)).data;\n if (parsed) return parsed;\n }\n\n return action.result;\n };\n}\n","import { PrimitiveState } from \"../utils/PrimitiveState\";\nimport { ApiCaller } from \"../api/api-caller\";\nimport { type WidgetConfig } from \"../types/widget-config\";\nimport { type Dto } from \"../api/client\";\nimport type { StorageCtx } from \"./storage.ctx\";\nimport { v4 } from \"uuid\";\n\ntype ContactState = {\n contact: {\n token: string;\n externalId: string | undefined;\n } | null;\n extraCollectedData: Record<string, string> | undefined;\n isCreatingUnverifiedContact: boolean;\n isErrorCreatingUnverifiedContact: boolean;\n};\n\nexport class ContactCtx {\n private config: WidgetConfig;\n private storageCtx?: StorageCtx;\n private api: ApiCaller;\n state: PrimitiveState<ContactState>;\n\n constructor({\n config,\n api,\n storageCtx,\n }: {\n api: ApiCaller;\n config: WidgetConfig;\n storageCtx?: StorageCtx;\n }) {\n this.config = config;\n this.storageCtx = storageCtx;\n this.api = api;\n\n this.state = new PrimitiveState<ContactState>({\n contact: config.user?.token\n ? {\n token: config.user.token,\n // Set optional externalId from config... not local storage\n externalId: config.user.externalId,\n }\n : null,\n extraCollectedData: undefined,\n isCreatingUnverifiedContact: false,\n isErrorCreatingUnverifiedContact: false,\n });\n\n this.autoCreateUnverifiedUserIfNotExists();\n }\n\n shouldCollectData = (): boolean => {\n if (!this.state.get().contact?.token && this.config.collectUserData) {\n return true;\n }\n return false;\n };\n\n private autoCreateUnverifiedUserIfNotExists = async () => {\n /**\n * If token is passed in config... we consider it as a verified user and do nothing (we don't force generate an externalId)\n * If a non-verified user take their token and place it in the config... the backend will refuse their requests saying that a non-verified token must have an externalId to create and get sessions\n */\n if (this.config.user?.token) return;\n\n /**\n * If collectUserData is true... we check if the user entered their credentials before, otherwise, show them the welcome screen so they can enter their credentials\n */\n if (this.config.collectUserData && !this.config.user?.data?.email) {\n /**\n * If extra data collection fields are passed,\n * we do not check for a persisted token.\n * This will force the contact to enter the extra data fields every time they visit the page.\n */\n if (this.config.extraDataCollectionFields?.length) {\n return;\n }\n\n const persistedToken = await this.storageCtx?.getContactToken();\n if (persistedToken) {\n await this.setUnverifiedContact(persistedToken);\n }\n // return early whether there is a persisted token or not\n return;\n }\n\n /**\n * If there is no email, then it is an anonymous contact, we check if the contact is persisted or we create a new one\n */\n if (!this.config.user?.data?.email) {\n const persistedToken = await this.storageCtx?.getContactToken();\n if (persistedToken) {\n await this.setUnverifiedContact(persistedToken);\n // return early only if there is a persisted token\n return;\n }\n }\n\n /**\n * If we reach here... then it is one of two\n * 1. There is an email passed in the config, let's just upsert the unverified contact without checking for persistence; maybe the email in the config did change.\n * 2. It is an anonymous contact (without an email) using this device for the first time.\n *\n * This is still safe even if the email in the config is tampered with by the contact, because there is a client-side externalId that will be generated for the current device...\n * So, only sessions created on this device will be accessible.\n */\n await this.createUnverifiedContact({\n email: this.config.user?.data?.email,\n non_verified_name: this.config.user?.data?.name || \"Anonymous\",\n non_verified_custom_data: this.config.user?.data?.customData,\n });\n };\n\n createUnverifiedContact = async (\n payload: Dto[\"CreateUnverifiedContactDto\"],\n extraCollectedData?: Record<string, string>,\n ): Promise<void> => {\n this.state.setPartial({ extraCollectedData });\n\n try {\n this.state.setPartial({\n isCreatingUnverifiedContact: true,\n isErrorCreatingUnverifiedContact: false,\n });\n\n const { data } = await this.api.createUnverifiedContact(payload);\n if (data?.token) {\n await this.setUnverifiedContact(data.token);\n } else {\n this.state.setPartial({ isErrorCreatingUnverifiedContact: true });\n }\n } finally {\n this.state.setPartial({ isCreatingUnverifiedContact: false });\n }\n };\n\n setUnverifiedContact = async (token: string) => {\n const persistedExternalId = await this.storageCtx?.getExternalContactId();\n /** Give priority to `externalId` from the config */\n const externalId: string =\n this.config.user?.externalId || persistedExternalId || v4();\n this.api.setAuthToken(token);\n // Set token in state after setting the token in the api handler\n await this.storageCtx?.setContactToken(token);\n await this.storageCtx?.setExternalContactId(externalId);\n this.state.setPartial({ contact: { token, externalId } });\n };\n}\n","import { v4 as uuidv4 } from \"uuid\";\n\nexport function genUuid() {\n return uuidv4();\n}\n","import type { ApiCaller } from \"../api/api-caller\";\nimport type { SessionDto } from \"../types/schemas\";\nimport { Poller } from \"../utils/Poller\";\nimport { PrimitiveState } from \"../utils/PrimitiveState\";\nimport type { ContactCtx } from \"./contact.ctx\";\n\ntype SessionState = {\n /**\n * The currently selected session.\n * Can be null if no session is selected, or if in chat screen and the session is not created yet.\n */\n session: SessionDto | null;\n isCreatingSession: boolean;\n};\ntype SessionsState = {\n /** List of all user sessions */\n data: SessionDto[];\n /** A cursor to get the next page of sessions */\n cursor: string | undefined;\n /** Indicates if no more pages are left */\n isLastPage: boolean;\n /** Did fetch for the first time */\n didStartInitialFetch: boolean;\n isInitialFetchLoading: boolean;\n};\n\nexport class SessionCtx {\n private api: ApiCaller;\n private contactCtx: ContactCtx;\n private sessionsPollingIntervalSeconds: number;\n private sessionsRefresher = new Poller();\n\n public sessionState = new PrimitiveState<SessionState>({\n session: null,\n isCreatingSession: false,\n });\n public sessionsState = new PrimitiveState<SessionsState>({\n data: [],\n cursor: undefined,\n isLastPage: false,\n didStartInitialFetch: false,\n /**\n * Initialize this as `true` so it always starts loading until the first fetch is done\n */\n isInitialFetchLoading: true,\n });\n\n constructor({\n api,\n contactCtx,\n sessionsPollingIntervalSeconds,\n }: {\n api: ApiCaller;\n contactCtx: ContactCtx;\n sessionsPollingIntervalSeconds: number;\n }) {\n this.api = api;\n this.contactCtx = contactCtx;\n this.sessionsPollingIntervalSeconds = sessionsPollingIntervalSeconds;\n\n this.registerSessionsRefresherWrapper();\n }\n\n /** Clears the session and stops polling */\n reset = async () => {\n // Reset the session only, leave sessions as-is\n this.sessionState.reset();\n };\n\n private registerSessionsRefresherWrapper = () => {\n if (\n // If the widget config was initially provided with a contact token, no state change would be triggered, so we just fetch\n this.contactCtx.state.get().contact?.token &&\n !this.sessionsState.get().didStartInitialFetch\n ) {\n this.registerSessionsRefresher();\n } else {\n // In other cases where auto authenticate is fired, the token would be eventually set in state, so we wait for it\n this.contactCtx.state.subscribe(({ contact }) => {\n if (contact?.token && !this.sessionsState.get().didStartInitialFetch) {\n this.registerSessionsRefresher();\n }\n });\n }\n };\n\n private registerSessionsRefresher = () => {\n this.sessionsRefresher.startPolling(async () => {\n if (this.sessionsState.get().didStartInitialFetch === false) {\n this.sessionsState.setPartial({ didStartInitialFetch: true });\n }\n\n await this.refreshSessions();\n\n if (this.sessionsState.get().isInitialFetchLoading === true) {\n this.sessionsState.setPartial({ isInitialFetchLoading: false });\n }\n }, this.sessionsPollingIntervalSeconds * 1000);\n };\n\n createSession = async () => {\n this.sessionState.setPartial({ session: null, isCreatingSession: true });\n\n const externalId = this.contactCtx.state.get().contact?.externalId;\n const { data: session, error } = await this.api.createSession({\n customData: externalId\n ? {\n external_id: externalId,\n }\n : undefined,\n });\n if (session) {\n this.sessionState.setPartial({ session, isCreatingSession: false });\n return session;\n }\n\n console.error(\"Failed to create session:\", error);\n return null;\n };\n\n loadMoreSessions = async () => {\n if (this.sessionsState.get().isLastPage) return;\n\n const { data } = await this.getSessions({\n cursor: this.sessionsState.get().cursor,\n });\n\n if (data) {\n const allSessions = [...this.sessionsState.get().data, ...data.items];\n // TODO sort by updated at\n const deduped = allSessions.filter(\n (s, i, self) => i === self.findIndex((_s) => s.id === _s.id),\n );\n\n this.sessionsState.setPartial({\n data: deduped,\n cursor: data.next || undefined,\n isLastPage: data.next === null,\n });\n }\n };\n\n private getSessions = async ({ cursor }: { cursor: string | undefined }) => {\n if (!this.contactCtx.state.get().contact?.token) return { data: null };\n\n const externalId = this.contactCtx.state.get().contact?.externalId;\n return await this.api.getSessions({\n cursor,\n filters: externalId\n ? {\n external_id: externalId,\n }\n : {},\n });\n };\n\n refreshSessions = async () => {\n // Get the first page only (pass no `cursor`)\n const { data } = await this.getSessions({ cursor: undefined });\n if (!data) return;\n const sessions = [...data.items, ...this.sessionsState.get().data].filter(\n (s, i, self) => i === self.findIndex((_s) => s.id === _s.id),\n );\n this.sessionsState.setPartial({ data: sessions });\n };\n}\n","import { ApiCaller } from \"../api/api-caller\";\nimport type { WidgetConfig } from \"../types/widget-config\";\nimport type {\n BotMessageType,\n MessageType,\n UserMessageType,\n} from \"../types/messages\";\nimport type {\n MessageAttachmentType,\n SendMessageDto,\n SendMessageOutputDto,\n} from \"../types/schemas\";\nimport { PrimitiveState } from \"../utils/PrimitiveState\";\nimport { genUuid } from \"../utils/uuid\";\nimport { SessionCtx } from \"./session.ctx\";\nimport type { ContactCtx } from \"./contact.ctx\";\n\ntype MessageCtxState = {\n messages: MessageType[];\n isSendingMessage: boolean;\n lastAIResMightSolveUserIssue: boolean;\n isInitialFetchLoading: boolean;\n};\n\nexport class MessageCtx {\n private config: WidgetConfig;\n private api: ApiCaller;\n private contactCtx: ContactCtx;\n private sessionCtx: SessionCtx;\n\n public state = new PrimitiveState<MessageCtxState>({\n messages: [],\n isSendingMessage: false,\n lastAIResMightSolveUserIssue: false,\n isInitialFetchLoading: false,\n });\n\n private sendMessageAbortController = new AbortController();\n\n constructor({\n config,\n api,\n sessionCtx,\n contactCtx,\n }: {\n config: WidgetConfig;\n api: ApiCaller;\n sessionCtx: SessionCtx;\n contactCtx: ContactCtx;\n }) {\n this.config = config;\n this.api = api;\n this.sessionCtx = sessionCtx;\n this.contactCtx = contactCtx;\n }\n\n reset = () => {\n this.sendMessageAbortController.abort(\"Resetting chat\");\n this.state.reset();\n };\n\n sendMessage = async (input: {\n content: SendMessageDto[\"content\"];\n attachments?: SendMessageDto[\"attachments\"];\n }): Promise<void> => {\n /* ------------------------------------------------------ */\n /* Prevent sending if there is no content */\n /* ------------------------------------------------------ */\n if (\n !input.content.trim() &&\n (!input.attachments || input.attachments.length === 0)\n ) {\n console.warn(\"Cannot send an empty message of no content or attachments\");\n return;\n }\n /* ------------------------------------------------------ */\n /* Prevent sending while waiting for AI res */\n /* ------------------------------------------------------ */\n const isSending = this.state.get().isSendingMessage;\n const isAssignedToAI =\n this.sessionCtx.sessionState.get().session?.assignee.kind === \"ai\";\n const _messages = this.state.get().messages;\n const lastMessage =\n _messages.length > 0 ? _messages[_messages.length - 1] : undefined;\n if (\n (isAssignedToAI && isSending) ||\n // If last message is from user, then bot response did not arrive yet\n (isAssignedToAI && lastMessage?.type === \"FROM_USER\")\n ) {\n console.warn(\"Cannot send messages while awaiting AI response\");\n return;\n }\n\n this.sendMessageAbortController = new AbortController();\n\n /* ------------------------------------------------------ */\n /* Clear last AI response might solve user issue */\n /* ------------------------------------------------------ */\n this.state.setPartial({ lastAIResMightSolveUserIssue: false });\n\n try {\n this.state.setPartial({ isSendingMessage: true });\n /* ------------------------------------------------------ */\n /* Optimistically add message to rendered messages */\n /* ------------------------------------------------------ */\n const userMessage = this.toUserMessage(\n input.content.trim(),\n input.attachments || undefined,\n );\n const currentMessages = this.state.get().messages;\n this.state.setPartial({\n messages: [...currentMessages, userMessage],\n });\n\n /* ------------------------------------------------------ */\n /* Create session if not exists */\n /* ------------------------------------------------------ */\n if (!this.sessionCtx.sessionState.get().session?.id) {\n const createdSession = await this.sessionCtx.createSession();\n\n // TODO: apply some retry logic here\n if (!createdSession) {\n console.error(\"Failed to create session\");\n return;\n }\n\n // Refresh sessions list instantly to get the newly created session in the sessions list\n void this.sessionCtx.refreshSessions();\n }\n const sessionId = this.sessionCtx.sessionState.get().session?.id;\n if (!sessionId) return;\n /* ------------------------------------------------------ */\n /* Send and wait for bot response */\n /* ------------------------------------------------------ */\n const { data } = await this.api.sendMessage(\n {\n uuid: userMessage.id,\n bot_token: this.config.token,\n headers: this.config.headers,\n query_params: this.config.queryParams,\n session_id: sessionId,\n content: userMessage.content,\n attachments: input.attachments,\n clientContext: this.config.context,\n },\n this.sendMessageAbortController.signal,\n );\n\n if (data?.success) {\n /* ------------------------------------------------------ */\n /* Append bot reply if not fetched from polling */\n /* ------------------------------------------------------ */\n const botMessage = this.toBotMessage(data);\n if (botMessage) {\n const prevMessages = this.state.get().messages;\n const shouldAppend = !prevMessages.some(\n (m) => m.id === botMessage.id,\n );\n if (!shouldAppend) {\n this.state.setPartial({\n lastAIResMightSolveUserIssue:\n data.autopilotResponse?.mightSolveUserIssue ||\n data.uiResponse?.mightSolveUserIssue,\n });\n return;\n }\n this.state.setPartial({\n messages: [...prevMessages, botMessage],\n lastAIResMightSolveUserIssue:\n data.autopilotResponse?.mightSolveUserIssue ||\n data.uiResponse?.mightSolveUserIssue,\n });\n }\n } else {\n const errorMessage = this.toBotErrorMessage(\n data?.error?.message || \"Unknown error occurred\",\n );\n const currentMessages = this.state.get().messages;\n this.state.setPartial({\n messages: [...currentMessages, errorMessage],\n });\n }\n } catch (error) {\n if (!this.sendMessageAbortController.signal.aborted) {\n console.error(\"Failed to send message:\", error);\n }\n } finally {\n this.state.setPartial({ isSendingMessage: false });\n }\n };\n\n private toUserMessage = (\n content: string,\n attachments?: MessageAttachmentType[],\n ): UserMessageType => {\n const messageContent = (() => {\n const extraCollectedData = this.contactCtx.state.get().extraCollectedData;\n // Prepend extra collected data if this is the first message in the session\n if (\n this.state.get().messages.length === 0 &&\n extraCollectedData &&\n Object.keys(extraCollectedData).length > 0\n ) {\n const data = Object.entries(extraCollectedData)\n .filter(([_, value]) => !!value)\n .map(([key, value]) => `${key}: ${value}`)\n .join(\" \\n\");\n return `${data} \\n\\n${content}`;\n }\n\n return content;\n })();\n\n return {\n id: genUuid(),\n type: \"FROM_USER\",\n content: messageContent,\n deliveredAt: new Date().toISOString(),\n attachments,\n timestamp: new Date().toISOString(),\n };\n };\n\n private toBotMessage = (\n response: SendMessageOutputDto,\n ): BotMessageType | null => {\n if (response.success && response.autopilotResponse) {\n return {\n type: \"FROM_BOT\",\n id: response.autopilotResponse.id || genUuid(),\n timestamp: new Date().toISOString(),\n component: \"bot_message\",\n agent: this.config.bot\n ? {\n name: this.config.bot.name || \"\",\n isAi: true,\n avatar: this.config.bot.avatar || \"\",\n id: null,\n }\n : undefined,\n data: {\n message: response.autopilotResponse.value.content,\n action: response.uiResponse?.value.name\n ? {\n name: response.uiResponse.value.name,\n data: response.uiResponse.value.request_response,\n }\n : undefined,\n },\n };\n }\n\n return null;\n };\n\n private toBotErrorMessage = (message: string): BotMessageType => {\n return {\n type: \"FROM_BOT\",\n id: genUuid(),\n timestamp: new Date().toISOString(),\n component: \"TEXT\",\n data: {\n message,\n variant: \"error\",\n action: undefined,\n },\n };\n };\n}\n","import type { WidgetConfig } from \"../types/widget-config\";\nimport { PrimitiveState } from \"../utils/PrimitiveState\";\nimport type { ContactCtx } from \"./contact.ctx\";\nimport type { SessionCtx } from \"./session.ctx\";\nimport type { WidgetCtx } from \"./widget.ctx\";\n\ntype RouterState = {\n screen:\n | /** A welcome screen to collect user data. Useful in public non-logged-in environments */\n \"welcome\"\n /** Show a list of the user's previous sessions */\n | \"sessions\"\n /** Self-explanatory */\n | \"chat\";\n};\n\nexport class RouterCtx {\n state: PrimitiveState<RouterState>;\n\n private config: WidgetConfig;\n private contactCtx: ContactCtx;\n private sessionCtx: SessionCtx;\n private resetChat: WidgetCtx[\"resetChat\"];\n\n constructor({\n config,\n contactCtx,\n sessionCtx,\n resetChat,\n }: {\n config: WidgetConfig;\n contactCtx: ContactCtx;\n sessionCtx: SessionCtx;\n resetChat: WidgetCtx[\"resetChat\"];\n }) {\n this.state = new PrimitiveState<RouterState>({\n screen: contactCtx.shouldCollectData() ? \"welcome\" : \"sessions\",\n });\n this.config = config;\n this.contactCtx = contactCtx;\n this.sessionCtx = sessionCtx;\n this.resetChat = resetChat;\n\n this.registerRoutingListener();\n }\n\n private registerRoutingListener = () => {\n this.contactCtx.state.subscribe(({ contact }) => {\n // Auto navigate to sessions screen after collecting user data\n if (contact?.token && this.state.get().screen === \"welcome\") {\n this.state.setPartial({ screen: \"sessions\" });\n }\n });\n\n this.sessionCtx.sessionsState.subscribe(\n ({ isInitialFetchLoading, data }) => {\n if (data.length) return;\n if (this.config.router?.goToChatIfNoSessions === false) return;\n\n // Auto navigate to chat screen if contact has no previous sessions\n if (!isInitialFetchLoading && this.state.get().screen !== \"chat\") {\n this.toChatScreen();\n }\n },\n );\n };\n\n toSessionsScreen = () => {\n this.resetChat();\n this.state.setPartial({ screen: \"sessions\" });\n };\n\n /**\n * @param sessionId The ID of the session to open, or `undefined` if it is a new chat session\n */\n toChatScreen = (sessionId?: string) => {\n this.resetChat();\n\n if (sessionId) {\n const session = this.sessionCtx.sessionsState\n .get()\n .data.find((s) => s.id === sessionId);\n // Do not navigate if session is not found (this shouldn't happen, unless a wrong ID is passed)\n if (!session) return;\n this.sessionCtx.sessionState.setPartial({ session });\n }\n\n this.state.setPartial({ screen: \"chat\" });\n };\n}\n","import type { ExternalStorage } from \"../types/external-storage\";\n\nexport class StorageCtx {\n private storage: ExternalStorage;\n private KEYS = {\n contactToken: \"opencx__widget__contactToken\",\n externalContactId: \"opencx__widget__externalContactId\",\n };\n\n constructor({ storage }: { storage: ExternalStorage }) {\n this.storage = storage;\n }\n\n setContactToken = async (token: string) => {\n await this.storage.set(this.KEYS.contactToken, token);\n };\n getContactToken = async () => {\n return this.storage.get(this.KEYS.contactToken);\n };\n\n setExternalContactId = async (id: string) => {\n await this.storage.set(this.KEYS.externalContactId, id);\n };\n getExternalContactId = async () => {\n return this.storage.get(this.KEYS.externalContactId);\n };\n}\n","import { ApiCaller } from \"../api/api-caller\";\nimport type { ExternalStorage } from \"../types/external-storage\";\nimport type { WidgetConfig } from \"../types/widget-config\";\nimport { ActiveSessionPollingCtx } from \"./active-session-polling.ctx\";\nimport { ContactCtx } from \"./contact.ctx\";\nimport { MessageCtx } from \"./message.ctx\";\nimport { RouterCtx } from \"./router.ctx\";\nimport { SessionCtx } from \"./session.ctx\";\nimport { StorageCtx } from \"./storage.ctx\";\n\nexport class WidgetCtx {\n public config: WidgetConfig;\n public api: ApiCaller;\n\n public contactCtx: ContactCtx;\n public sessionCtx: SessionCtx;\n public messageCtx: MessageCtx;\n public routerCtx: RouterCtx;\n public storageCtx?: StorageCtx;\n\n private static pollingIntervalsSeconds: {\n session: number;\n sessions: number;\n } | null = null;\n private activeSessionPollingCtx: ActiveSessionPollingCtx;\n\n private constructor({\n config,\n storage,\n }: { config: WidgetConfig; storage?: ExternalStorage }) {\n if (!WidgetCtx.pollingIntervalsSeconds) {\n throw Error(\n \"Widget polling values are not defined, did you call WidgetCtx.initialize()\",\n );\n }\n\n this.config = config;\n this.api = new ApiCaller({ config });\n this.storageCtx = storage ? new StorageCtx({ storage }) : undefined;\n\n this.contactCtx = new ContactCtx({\n api: this.api,\n config: this.config,\n storageCtx: this.storageCtx,\n });\n\n this.sessionCtx = new SessionCtx({\n api: this.api,\n contactCtx: this.contactCtx,\n sessionsPollingIntervalSeconds:\n WidgetCtx.pollingIntervalsSeconds.sessions,\n });\n\n this.messageCtx = new MessageCtx({\n config: this.config,\n api: this.api,\n sessionCtx: this.sessionCtx,\n contactCtx: this.contactCtx,\n });\n\n this.activeSessionPollingCtx = new ActiveSessionPollingCtx({\n api: this.api,\n config: this.config,\n sessionCtx: this.sessionCtx,\n messageCtx: this.messageCtx,\n sessionPollingIntervalSeconds: WidgetCtx.pollingIntervalsSeconds.session,\n });\n\n this.routerCtx = new RouterCtx({\n config: this.config,\n contactCtx: this.contactCtx,\n sessionCtx: this.sessionCtx,\n resetChat: this.resetChat,\n });\n }\n\n static initialize = async ({\n config,\n storage,\n }: { config: WidgetConfig; storage?: ExternalStorage }) => {\n const externalConfig = await new ApiCaller({\n config,\n }).getExternalWidgetConfig();\n\n this.pollingIntervalsSeconds = {\n session: externalConfig.data?.sessionPollingIntervalSeconds || 10,\n sessions: externalConfig.data?.sessionsPollingIntervalSeconds || 60,\n };\n\n return new WidgetCtx({\n config,\n storage,\n });\n };\n\n resetChat = () => {\n this.sessionCtx.reset();\n this.messageCtx.reset();\n };\n}\n"],"names":["defaultOnError","onErrorOptions","basicClient","options","client","createClient","middlewares","ApiCaller","config","token","baseUrl","headers","request","key","value","body","abortSignal","sessionId","lastMessageTimestamp","query","cursor","filters","file","onProgress","resolve","reject","formData","xhr","event","percentage","data","error","uploadUrl","_a","PrimitiveState","state","newState","isEqual","_s","callback","Poller","cb","intervalMs","timeouts","poll","runCatching","result","ActiveSessionPollingCtx","api","sessionCtx","messageCtx","sessionPollingIntervalSeconds","session","messages","prevMessages","newMessages","newMsg","existingMsg","history","commonFields","action","_b","responseBodyText","parsed","ContactCtx","storageCtx","_c","_d","persistedToken","_e","_g","_f","_h","_j","_i","_l","_k","_n","_m","payload","extraCollectedData","persistedExternalId","externalId","v4","genUuid","uuidv4","SessionCtx","contactCtx","sessionsPollingIntervalSeconds","contact","deduped","s","i","self","sessions","MessageCtx","input","isSending","isAssignedToAI","_messages","lastMessage","userMessage","currentMessages","botMessage","m","errorMessage","content","attachments","messageContent","_","response","message","RouterCtx","resetChat","isInitialFetchLoading","StorageCtx","storage","id","_WidgetCtx","externalConfig","WidgetCtx"],"mappings":"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"}
@@ -1,5 +1,5 @@
1
- "use strict";const O=require("openapi-fetch"),F=require("lodash.isequal"),A=require("uuid"),L=g=>{console.log(g.error)},B=g=>{const n=O({baseUrl:g.baseUrl}),i={onRequest:g.onRequest,onResponse:g.onResponse,onError:g.onError||L};return n.use(i),n};class U{constructor({config:n}){var a;this.userToken=null,this.constructClientOptions=t=>{const s=this.config.apiUrl||"https://api.open.cx",e={"X-Bot-Token":this.config.token,"Content-Type":"application/json",Accept:"application/json",Authorization:t?`Bearer ${t}`:void 0};return{baseUrl:s,headers:e}},this.createOpenAPIClient=({baseUrl:t,headers:s})=>B({baseUrl:t,onRequest:({request:e})=>{Object.entries(s).forEach(([o,r])=>{r&&e.headers.set(o,r)})}}),this.setAuthToken=t=>{this.userToken=t;const{baseUrl:s,headers:e}=this.constructClientOptions(t);this.client=this.createOpenAPIClient({baseUrl:s,headers:e})},this.getExternalWidgetConfig=async()=>await this.client.GET("/backend/widget/v2/config",{params:{header:{"X-Bot-Token":this.config.token}}}),this.widgetPrelude=async()=>await this.client.GET("/backend/widget/v2/prelude",{params:{header:{"X-Bot-Token":this.config.token}}}),this.sendMessage=async(t,s)=>await this.client.POST("/backend/widget/v2/chat/send",{body:t,signal:s}),this.createUnverifiedContact=async t=>await this.client.POST("/backend/widget/v2/contact/create-unverified",{params:{header:{"x-bot-token":this.config.token}},body:t}),this.createSession=async t=>await this.client.POST("/backend/widget/v2/create-session",{body:t}),this.pollSessionAndHistory=async({sessionId:t,lastMessageTimestamp:s,abortSignal:e})=>{const o=s?{lastMessageTimestamp:s}:void 0;return await this.client.GET("/backend/widget/v2/poll/{sessionId}",{params:{path:{sessionId:t},query:o},signal:e})},this.getSessions=async({cursor:t,filters:s,abortSignal:e})=>await this.client.GET("/backend/widget/v2/sessions",{params:{query:{cursor:t,filters:JSON.stringify(s)}},signal:e}),this.uploadFile=async({file:t,abortSignal:s,onProgress:e})=>new Promise((o,r)=>{const l=new FormData;l.append("file",t);const c=new XMLHttpRequest;if(s&&(s.addEventListener("abort",()=>{c.abort(),r(new DOMException("Aborted","AbortError"))}),s.aborted)){r(new DOMException("Aborted","AbortError"));return}c.upload.addEventListener("progress",d=>{if(d.lengthComputable&&e){const x=Math.round(d.loaded/d.total*100);e(x)}}),c.addEventListener("load",()=>{if(c.status>=200&&c.status<300)try{const d=JSON.parse(c.responseText);o(d)}catch(d){r(new Error(`Failed to parse response: ${d}`))}else r(new Error(`Upload failed with status: ${c.status}`))}),c.addEventListener("error",()=>{r(new Error("Network error occurred"))}),c.addEventListener("timeout",()=>{r(new Error("Upload timed out"))});const{baseUrl:f}=this.constructClientOptions(this.userToken),C=`${f}/backend/widget/v2/upload`;c.open("POST",C),c.setRequestHeader("X-Bot-Token",this.config.token),this.userToken&&c.setRequestHeader("Authorization",`Bearer ${this.userToken}`),c.send(l)}),this.vote=async t=>await this.client.POST("/backend/widget/v2/chat/vote",{body:t}),this.config=n;const{baseUrl:i,headers:h}=this.constructClientOptions((a=n.user)==null?void 0:a.token);this.client=this.createOpenAPIClient({baseUrl:i,headers:h})}}class v{constructor(n){this.subscribers=new Set,this.get=()=>this.state,this.set=i=>{F(this.state,i)||(this.state=i,this.notifySubscribers(i))},this.setPartial=i=>{if(i==null)return;const h={...this.state,...i};this.set(h)},this.reset=()=>{this.set(this.initialState)},this.notifySubscribers=i=>{Array.from(this.subscribers).forEach(a=>{try{a(i)}catch(t){console.error(t)}})},this.subscribe=i=>(this.subscribers.add(i),()=>{this.subscribers.delete(i)}),this.state=n,this.initialState=n}}class T{constructor(){this.state=new v({isPolling:!1,isError:!1}),this.abortController=new AbortController,this.reset=()=>{var n;this.abortController.abort("Resetting poller"),(n=this.stopPolling)==null||n.call(this),this.stopPolling=null},this.stopPolling=null,this.startPolling=(n,i)=>{if(this.stopPolling)return;const h=[],a=async()=>{this.abortController=new AbortController,this.state.setPartial({isPolling:!0});try{await n(this.abortController.signal)}catch(t){if(this.abortController.signal.aborted)return;console.error("Failed to poll:",t),this.state.setPartial({isError:!0})}finally{this.state.setPartial({isPolling:!1})}this.abortController.signal.aborted?console.log("Poller aborted, not scheduling anymore"):h.push(setTimeout(a,i))};a(),this.stopPolling=()=>{h.forEach(clearTimeout),this.state.reset()}}}}function D(g){try{const n=g();return n instanceof Promise?n.then(i=>({data:i})).catch(i=>({error:i})):{data:n}}catch(n){return{error:n}}}class _{constructor({api:n,config:i,sessionCtx:h,messageCtx:a,sessionPollingIntervalSeconds:t}){this.poller=new T,this.registerPolling=()=>{this.sessionCtx.sessionState.subscribe(({session:s})=>{s!=null&&s.id?this.poller.startPolling(async e=>{this.hackAndSlash(s.id,e)},this.sessionPollingIntervalSeconds*1e3):this.poller.reset()})},this.hackAndSlash=async(s,e)=>{var c;this.messageCtx.state.get().messages.length===0&&this.messageCtx.state.setPartial({isInitialFetchLoading:!0});const o=this.messageCtx.state.get().messages,r=o.length>0?(c=o[o.length-1])==null?void 0:c.timestamp:void 0,{data:l}=await this.api.pollSessionAndHistory({sessionId:s,abortSignal:e,lastMessageTimestamp:r});if(l!=null&&l.session&&this.sessionCtx.sessionState.setPartial({session:l.session}),l!=null&&l.history&&l.history.length>0){const f=this.messageCtx.state.get().messages,S=l.history.map(this.mapHistoryToMessage).filter(C=>!f.some(d=>d.id===C.id));this.messageCtx.state.setPartial({messages:[...f,...S]})}this.messageCtx.state.get().isInitialFetchLoading&&this.messageCtx.state.setPartial({isInitialFetchLoading:!1})},this.mapHistoryToMessage=s=>{var r,l;const e={id:s.publicId,timestamp:s.sentAt||"",attachments:s.attachments||void 0};if(s.sender.kind==="user")return{...e,type:"FROM_USER",content:s.content.text||"",deliveredAt:s.sentAt||""};if(s.sender.kind==="agent")return{...e,type:"FROM_AGENT",component:"agent_message",data:{message:s.content.text||""},agent:{name:s.sender.name||"",avatar:s.sender.avatar||"",id:null,isAi:!1}};const o=s.actionCalls&&s.actionCalls.length>0?s.actionCalls[s.actionCalls.length-1]:void 0;return{...e,type:"FROM_BOT",component:"bot_message",agent:{id:null,name:((r=this.config.bot)==null?void 0:r.name)||"",isAi:!0,avatar:((l=this.config.bot)==null?void 0:l.avatar)||""},data:{message:s.content.text||"",action:o?{name:o.actionName,data:this.extractActionResult(o)}:void 0}}},this.extractActionResult=s=>{const e=s.result;if(e===null||typeof e!="object")return e;if("responseBodyText"in e&&typeof e.responseBodyText=="string"){const o=e.responseBodyText,r=D(()=>JSON.parse(o)).data;if(r)return r}return s.result},this.api=n,this.config=i,this.sessionCtx=h,this.messageCtx=a,this.sessionPollingIntervalSeconds=t,this.registerPolling()}}class q{constructor({config:n,api:i,storageCtx:h}){var a;this.shouldCollectData=()=>{var t;return!!(!((t=this.state.get().contact)!=null&&t.token)&&this.config.collectUserData)},this.autoCreateUnverifiedUserIfNotExists=async()=>{var t,s,e,o,r,l,c,f,S,C,d,x;if(!((t=this.config.user)!=null&&t.token)){if(this.config.collectUserData&&!((e=(s=this.config.user)==null?void 0:s.data)!=null&&e.email)){if((o=this.config.extraDataCollectionFields)!=null&&o.length)return;const m=await((r=this.storageCtx)==null?void 0:r.getContactToken());m&&await this.setUnverifiedContact(m);return}if(!((c=(l=this.config.user)==null?void 0:l.data)!=null&&c.email)){const m=await((f=this.storageCtx)==null?void 0:f.getContactToken());if(m){await this.setUnverifiedContact(m);return}}await this.createUnverifiedContact({name:((C=(S=this.config.user)==null?void 0:S.data)==null?void 0:C.name)||"Anonymous",email:(x=(d=this.config.user)==null?void 0:d.data)==null?void 0:x.email})}},this.createUnverifiedContact=async(t,s)=>{this.state.setPartial({extraCollectedData:s});try{this.state.setPartial({isCreatingUnverifiedContact:!0,isErrorCreatingUnverifiedContact:!1});const{data:e}=await this.api.createUnverifiedContact(t);e!=null&&e.token?await this.setUnverifiedContact(e.token):this.state.setPartial({isErrorCreatingUnverifiedContact:!0})}finally{this.state.setPartial({isCreatingUnverifiedContact:!1})}},this.setUnverifiedContact=async t=>{var o,r,l,c;const s=await((o=this.storageCtx)==null?void 0:o.getExternalContactId()),e=((r=this.config.user)==null?void 0:r.externalId)||s||A.v4();this.api.setAuthToken(t),await((l=this.storageCtx)==null?void 0:l.setContactToken(t)),await((c=this.storageCtx)==null?void 0:c.setExternalContactId(e)),this.state.setPartial({contact:{token:t,externalId:e}})},this.config=n,this.storageCtx=h,this.api=i,this.state=new v({contact:(a=n.user)!=null&&a.token?{token:n.user.token,externalId:n.user.externalId}:null,extraCollectedData:void 0,isCreatingUnverifiedContact:!1,isErrorCreatingUnverifiedContact:!1}),this.autoCreateUnverifiedUserIfNotExists()}}function I(){return A.v4()}class ${constructor({api:n,contactCtx:i,sessionsPollingIntervalSeconds:h}){this.sessionsRefresher=new T,this.sessionState=new v({session:null,isCreatingSession:!1}),this.sessionsState=new v({data:[],cursor:void 0,isLastPage:!1,didStartInitialFetch:!1,isInitialFetchLoading:!0}),this.reset=async()=>{this.sessionState.reset()},this.registerSessionsRefresherWrapper=()=>{var a;(a=this.contactCtx.state.get().contact)!=null&&a.token&&!this.sessionsState.get().didStartInitialFetch?this.registerSessionsRefresher():this.contactCtx.state.subscribe(({contact:t})=>{t!=null&&t.token&&!this.sessionsState.get().didStartInitialFetch&&this.registerSessionsRefresher()})},this.registerSessionsRefresher=()=>{this.sessionsRefresher.startPolling(async()=>{this.sessionsState.get().didStartInitialFetch===!1&&this.sessionsState.setPartial({didStartInitialFetch:!0}),await this.refreshSessions(),this.sessionsState.get().isInitialFetchLoading===!0&&this.sessionsState.setPartial({isInitialFetchLoading:!1})},this.sessionsPollingIntervalSeconds*1e3)},this.createSession=async()=>{var e;this.sessionState.setPartial({session:null,isCreatingSession:!0});const a=(e=this.contactCtx.state.get().contact)==null?void 0:e.externalId,{data:t,error:s}=await this.api.createSession({customData:a?{external_id:a}:void 0});return t?(this.sessionState.setPartial({session:t,isCreatingSession:!1}),t):(console.error("Failed to create session:",s),null)},this.loadMoreSessions=async()=>{if(this.sessionsState.get().isLastPage)return;const{data:a}=await this.getSessions({cursor:this.sessionsState.get().cursor});if(a){const s=[...this.sessionsState.get().data,...a.items].filter((e,o,r)=>o===r.findIndex(l=>e.id===l.id));this.sessionsState.setPartial({data:s,cursor:a.next||void 0,isLastPage:a.next===null})}},this.getSessions=async({cursor:a})=>{var s,e;if(!((s=this.contactCtx.state.get().contact)!=null&&s.token))return{data:null};const t=(e=this.contactCtx.state.get().contact)==null?void 0:e.externalId;return await this.api.getSessions({cursor:a,filters:t?{external_id:t}:{}})},this.refreshSessions=async()=>{const{data:a}=await this.getSessions({cursor:void 0});if(!a)return;const t=[...a.items,...this.sessionsState.get().data].filter((s,e,o)=>e===o.findIndex(r=>s.id===r.id));this.sessionsState.setPartial({data:t})},this.api=n,this.contactCtx=i,this.sessionsPollingIntervalSeconds=h,this.registerSessionsRefresherWrapper()}}class N{constructor({config:n,api:i,sessionCtx:h,contactCtx:a}){this.state=new v({messages:[],isSendingMessage:!1,lastAIResMightSolveUserIssue:!1,isInitialFetchLoading:!1}),this.sendMessageAbortController=new AbortController,this.reset=()=>{this.sendMessageAbortController.abort("Resetting chat"),this.state.reset()},this.sendMessage=async t=>{var l,c,f,S,C,d,x,m,k;if(!t.content.trim()&&(!t.attachments||t.attachments.length===0)){console.warn("Cannot send an empty message of no content or attachments");return}const s=this.state.get().isSendingMessage,e=((l=this.sessionCtx.sessionState.get().session)==null?void 0:l.assignee.kind)==="ai",o=this.state.get().messages,r=o.length>0?o[o.length-1]:void 0;if(e&&s||e&&(r==null?void 0:r.type)==="FROM_USER"){console.warn("Cannot send messages while awaiting AI response");return}this.sendMessageAbortController=new AbortController,this.state.setPartial({lastAIResMightSolveUserIssue:!1});try{this.state.setPartial({isSendingMessage:!0});const w=this.toUserMessage(t.content.trim(),t.attachments||void 0),R=this.state.get().messages;if(this.state.setPartial({messages:[...R,w]}),!((c=this.sessionCtx.sessionState.get().session)!=null&&c.id)){if(!await this.sessionCtx.createSession()){console.error("Failed to create session");return}this.sessionCtx.refreshSessions()}const E=(f=this.sessionCtx.sessionState.get().session)==null?void 0:f.id;if(!E)return;const{data:u}=await this.api.sendMessage({uuid:w.id,bot_token:this.config.token,headers:this.config.headers,query_params:this.config.queryParams,session_id:E,user:(S=this.config.user)==null?void 0:S.data,content:w.content,attachments:t.attachments,clientContext:this.config.context},this.sendMessageAbortController.signal);if(u!=null&&u.success){const b=this.toBotMessage(u);if(b){const P=this.state.get().messages;if(!!P.some(M=>M.id===b.id)){this.state.setPartial({lastAIResMightSolveUserIssue:((C=u.autopilotResponse)==null?void 0:C.mightSolveUserIssue)||((d=u.uiResponse)==null?void 0:d.mightSolveUserIssue)});return}this.state.setPartial({messages:[...P,b],lastAIResMightSolveUserIssue:((x=u.autopilotResponse)==null?void 0:x.mightSolveUserIssue)||((m=u.uiResponse)==null?void 0:m.mightSolveUserIssue)})}}else{const b=this.toBotErrorMessage(((k=u==null?void 0:u.error)==null?void 0:k.message)||"Unknown error occurred"),P=this.state.get().messages;this.state.setPartial({messages:[...P,b]})}}catch(w){this.sendMessageAbortController.signal.aborted||console.error("Failed to send message:",w)}finally{this.state.setPartial({isSendingMessage:!1})}},this.toUserMessage=(t,s)=>{const e=(()=>{const o=this.contactCtx.state.get().extraCollectedData;return this.state.get().messages.length===0&&o&&Object.keys(o).length>0?`${Object.entries(o).filter(([l,c])=>!!c).map(([l,c])=>`${l}: ${c}`).join(`
1
+ "use strict";const M=require("openapi-fetch"),O=require("lodash.isequal"),A=require("uuid"),F=g=>{console.log(g.error)},_=g=>{const n=M({baseUrl:g.baseUrl}),i={onRequest:g.onRequest,onResponse:g.onResponse,onError:g.onError||F};return n.use(i),n};class U{constructor({config:n}){var a;this.userToken=null,this.constructClientOptions=t=>{const s=this.config.apiUrl||"https://api.open.cx",e={"X-Bot-Token":this.config.token,"Content-Type":"application/json",Accept:"application/json",Authorization:t?`Bearer ${t}`:void 0};return{baseUrl:s,headers:e}},this.createOpenAPIClient=({baseUrl:t,headers:s})=>_({baseUrl:t,onRequest:({request:e})=>{Object.entries(s).forEach(([o,r])=>{r&&e.headers.set(o,r)})}}),this.setAuthToken=t=>{this.userToken=t;const{baseUrl:s,headers:e}=this.constructClientOptions(t);this.client=this.createOpenAPIClient({baseUrl:s,headers:e})},this.getExternalWidgetConfig=async()=>await this.client.GET("/backend/widget/v2/config",{params:{header:{"X-Bot-Token":this.config.token}}}),this.widgetPrelude=async()=>await this.client.GET("/backend/widget/v2/prelude",{params:{header:{"X-Bot-Token":this.config.token}}}),this.sendMessage=async(t,s)=>await this.client.POST("/backend/widget/v2/chat/send",{body:t,signal:s}),this.createUnverifiedContact=async t=>await this.client.POST("/backend/widget/v2/contact/create-unverified",{params:{header:{"x-bot-token":this.config.token}},body:t}),this.createSession=async t=>await this.client.POST("/backend/widget/v2/create-session",{body:t}),this.pollSessionAndHistory=async({sessionId:t,lastMessageTimestamp:s,abortSignal:e})=>{const o=s?{lastMessageTimestamp:s}:void 0;return await this.client.GET("/backend/widget/v2/poll/{sessionId}",{params:{path:{sessionId:t},query:o},signal:e})},this.getSessions=async({cursor:t,filters:s,abortSignal:e})=>await this.client.GET("/backend/widget/v2/sessions",{params:{query:{cursor:t,filters:JSON.stringify(s)}},signal:e}),this.uploadFile=async({file:t,abortSignal:s,onProgress:e})=>new Promise((o,r)=>{const l=new FormData;l.append("file",t);const c=new XMLHttpRequest;if(s&&(s.addEventListener("abort",()=>{c.abort(),r(new DOMException("Aborted","AbortError"))}),s.aborted)){r(new DOMException("Aborted","AbortError"));return}c.upload.addEventListener("progress",d=>{if(d.lengthComputable&&e){const x=Math.round(d.loaded/d.total*100);e(x)}}),c.addEventListener("load",()=>{if(c.status>=200&&c.status<300)try{const d=JSON.parse(c.responseText);o(d)}catch(d){r(new Error(`Failed to parse response: ${d}`))}else r(new Error(`Upload failed with status: ${c.status}`))}),c.addEventListener("error",()=>{r(new Error("Network error occurred"))}),c.addEventListener("timeout",()=>{r(new Error("Upload timed out"))});const{baseUrl:f}=this.constructClientOptions(this.userToken),C=`${f}/backend/widget/v2/upload`;c.open("POST",C),c.setRequestHeader("X-Bot-Token",this.config.token),this.userToken&&c.setRequestHeader("Authorization",`Bearer ${this.userToken}`),c.send(l)}),this.vote=async t=>await this.client.POST("/backend/widget/v2/chat/vote",{body:t}),this.config=n;const{baseUrl:i,headers:h}=this.constructClientOptions((a=n.user)==null?void 0:a.token);this.client=this.createOpenAPIClient({baseUrl:i,headers:h})}}class b{constructor(n){this.subscribers=new Set,this.get=()=>this.state,this.set=i=>{O(this.state,i)||(this.state=i,this.notifySubscribers(i))},this.setPartial=i=>{if(i==null)return;const h={...this.state,...i};this.set(h)},this.reset=()=>{this.set(this.initialState)},this.notifySubscribers=i=>{Array.from(this.subscribers).forEach(a=>{try{a(i)}catch(t){console.error(t)}})},this.subscribe=i=>(this.subscribers.add(i),()=>{this.subscribers.delete(i)}),this.state=n,this.initialState=n}}class T{constructor(){this.state=new b({isPolling:!1,isError:!1}),this.abortController=new AbortController,this.reset=()=>{var n;this.abortController.abort("Resetting poller"),(n=this.stopPolling)==null||n.call(this),this.stopPolling=null},this.stopPolling=null,this.startPolling=(n,i)=>{if(this.stopPolling)return;const h=[],a=async()=>{this.abortController=new AbortController,this.state.setPartial({isPolling:!0});try{await n(this.abortController.signal)}catch(t){if(this.abortController.signal.aborted)return;console.error("Failed to poll:",t),this.state.setPartial({isError:!0})}finally{this.state.setPartial({isPolling:!1})}this.abortController.signal.aborted?console.log("Poller aborted, not scheduling anymore"):h.push(setTimeout(a,i))};a(),this.stopPolling=()=>{h.forEach(clearTimeout),this.state.reset()}}}}function D(g){try{const n=g();return n instanceof Promise?n.then(i=>({data:i})).catch(i=>({error:i})):{data:n}}catch(n){return{error:n}}}class L{constructor({api:n,config:i,sessionCtx:h,messageCtx:a,sessionPollingIntervalSeconds:t}){this.poller=new T,this.registerPolling=()=>{this.sessionCtx.sessionState.subscribe(({session:s})=>{s!=null&&s.id?this.poller.startPolling(async e=>{this.hackAndSlash(s.id,e)},this.sessionPollingIntervalSeconds*1e3):this.poller.reset()})},this.hackAndSlash=async(s,e)=>{var c;this.messageCtx.state.get().messages.length===0&&this.messageCtx.state.setPartial({isInitialFetchLoading:!0});const o=this.messageCtx.state.get().messages,r=o.length>0?(c=o[o.length-1])==null?void 0:c.timestamp:void 0,{data:l}=await this.api.pollSessionAndHistory({sessionId:s,abortSignal:e,lastMessageTimestamp:r});if(l!=null&&l.session&&this.sessionCtx.sessionState.setPartial({session:l.session}),l!=null&&l.history&&l.history.length>0){const f=this.messageCtx.state.get().messages,m=l.history.map(this.mapHistoryToMessage).filter(C=>!f.some(d=>d.id===C.id));this.messageCtx.state.setPartial({messages:[...f,...m]})}this.messageCtx.state.get().isInitialFetchLoading&&this.messageCtx.state.setPartial({isInitialFetchLoading:!1})},this.mapHistoryToMessage=s=>{var r,l;const e={id:s.publicId,timestamp:s.sentAt||"",attachments:s.attachments||void 0};if(s.sender.kind==="user")return{...e,type:"FROM_USER",content:s.content.text||"",deliveredAt:s.sentAt||""};if(s.sender.kind==="agent")return{...e,type:"FROM_AGENT",component:"agent_message",data:{message:s.content.text||""},agent:{name:s.sender.name||"",avatar:s.sender.avatar||"",id:null,isAi:!1}};const o=s.actionCalls&&s.actionCalls.length>0?s.actionCalls[s.actionCalls.length-1]:void 0;return{...e,type:"FROM_BOT",component:"bot_message",agent:{id:null,name:((r=this.config.bot)==null?void 0:r.name)||"",isAi:!0,avatar:((l=this.config.bot)==null?void 0:l.avatar)||""},data:{message:s.content.text||"",action:o?{name:o.actionName,data:this.extractActionResult(o)}:void 0}}},this.extractActionResult=s=>{const e=s.result;if(e===null||typeof e!="object")return e;if("responseBodyText"in e&&typeof e.responseBodyText=="string"){const o=e.responseBodyText,r=D(()=>JSON.parse(o)).data;if(r)return r}return s.result},this.api=n,this.config=i,this.sessionCtx=h,this.messageCtx=a,this.sessionPollingIntervalSeconds=t,this.registerPolling()}}class B{constructor({config:n,api:i,storageCtx:h}){var a;this.shouldCollectData=()=>{var t;return!!(!((t=this.state.get().contact)!=null&&t.token)&&this.config.collectUserData)},this.autoCreateUnverifiedUserIfNotExists=async()=>{var t,s,e,o,r,l,c,f,m,C,d,x,P,S;if(!((t=this.config.user)!=null&&t.token)){if(this.config.collectUserData&&!((e=(s=this.config.user)==null?void 0:s.data)!=null&&e.email)){if((o=this.config.extraDataCollectionFields)!=null&&o.length)return;const v=await((r=this.storageCtx)==null?void 0:r.getContactToken());v&&await this.setUnverifiedContact(v);return}if(!((c=(l=this.config.user)==null?void 0:l.data)!=null&&c.email)){const v=await((f=this.storageCtx)==null?void 0:f.getContactToken());if(v){await this.setUnverifiedContact(v);return}}await this.createUnverifiedContact({email:(C=(m=this.config.user)==null?void 0:m.data)==null?void 0:C.email,non_verified_name:((x=(d=this.config.user)==null?void 0:d.data)==null?void 0:x.name)||"Anonymous",non_verified_custom_data:(S=(P=this.config.user)==null?void 0:P.data)==null?void 0:S.customData})}},this.createUnverifiedContact=async(t,s)=>{this.state.setPartial({extraCollectedData:s});try{this.state.setPartial({isCreatingUnverifiedContact:!0,isErrorCreatingUnverifiedContact:!1});const{data:e}=await this.api.createUnverifiedContact(t);e!=null&&e.token?await this.setUnverifiedContact(e.token):this.state.setPartial({isErrorCreatingUnverifiedContact:!0})}finally{this.state.setPartial({isCreatingUnverifiedContact:!1})}},this.setUnverifiedContact=async t=>{var o,r,l,c;const s=await((o=this.storageCtx)==null?void 0:o.getExternalContactId()),e=((r=this.config.user)==null?void 0:r.externalId)||s||A.v4();this.api.setAuthToken(t),await((l=this.storageCtx)==null?void 0:l.setContactToken(t)),await((c=this.storageCtx)==null?void 0:c.setExternalContactId(e)),this.state.setPartial({contact:{token:t,externalId:e}})},this.config=n,this.storageCtx=h,this.api=i,this.state=new b({contact:(a=n.user)!=null&&a.token?{token:n.user.token,externalId:n.user.externalId}:null,extraCollectedData:void 0,isCreatingUnverifiedContact:!1,isErrorCreatingUnverifiedContact:!1}),this.autoCreateUnverifiedUserIfNotExists()}}function y(){return A.v4()}class q{constructor({api:n,contactCtx:i,sessionsPollingIntervalSeconds:h}){this.sessionsRefresher=new T,this.sessionState=new b({session:null,isCreatingSession:!1}),this.sessionsState=new b({data:[],cursor:void 0,isLastPage:!1,didStartInitialFetch:!1,isInitialFetchLoading:!0}),this.reset=async()=>{this.sessionState.reset()},this.registerSessionsRefresherWrapper=()=>{var a;(a=this.contactCtx.state.get().contact)!=null&&a.token&&!this.sessionsState.get().didStartInitialFetch?this.registerSessionsRefresher():this.contactCtx.state.subscribe(({contact:t})=>{t!=null&&t.token&&!this.sessionsState.get().didStartInitialFetch&&this.registerSessionsRefresher()})},this.registerSessionsRefresher=()=>{this.sessionsRefresher.startPolling(async()=>{this.sessionsState.get().didStartInitialFetch===!1&&this.sessionsState.setPartial({didStartInitialFetch:!0}),await this.refreshSessions(),this.sessionsState.get().isInitialFetchLoading===!0&&this.sessionsState.setPartial({isInitialFetchLoading:!1})},this.sessionsPollingIntervalSeconds*1e3)},this.createSession=async()=>{var e;this.sessionState.setPartial({session:null,isCreatingSession:!0});const a=(e=this.contactCtx.state.get().contact)==null?void 0:e.externalId,{data:t,error:s}=await this.api.createSession({customData:a?{external_id:a}:void 0});return t?(this.sessionState.setPartial({session:t,isCreatingSession:!1}),t):(console.error("Failed to create session:",s),null)},this.loadMoreSessions=async()=>{if(this.sessionsState.get().isLastPage)return;const{data:a}=await this.getSessions({cursor:this.sessionsState.get().cursor});if(a){const s=[...this.sessionsState.get().data,...a.items].filter((e,o,r)=>o===r.findIndex(l=>e.id===l.id));this.sessionsState.setPartial({data:s,cursor:a.next||void 0,isLastPage:a.next===null})}},this.getSessions=async({cursor:a})=>{var s,e;if(!((s=this.contactCtx.state.get().contact)!=null&&s.token))return{data:null};const t=(e=this.contactCtx.state.get().contact)==null?void 0:e.externalId;return await this.api.getSessions({cursor:a,filters:t?{external_id:t}:{}})},this.refreshSessions=async()=>{const{data:a}=await this.getSessions({cursor:void 0});if(!a)return;const t=[...a.items,...this.sessionsState.get().data].filter((s,e,o)=>e===o.findIndex(r=>s.id===r.id));this.sessionsState.setPartial({data:t})},this.api=n,this.contactCtx=i,this.sessionsPollingIntervalSeconds=h,this.registerSessionsRefresherWrapper()}}class ${constructor({config:n,api:i,sessionCtx:h,contactCtx:a}){this.state=new b({messages:[],isSendingMessage:!1,lastAIResMightSolveUserIssue:!1,isInitialFetchLoading:!1}),this.sendMessageAbortController=new AbortController,this.reset=()=>{this.sendMessageAbortController.abort("Resetting chat"),this.state.reset()},this.sendMessage=async t=>{var l,c,f,m,C,d,x,P;if(!t.content.trim()&&(!t.attachments||t.attachments.length===0)){console.warn("Cannot send an empty message of no content or attachments");return}const s=this.state.get().isSendingMessage,e=((l=this.sessionCtx.sessionState.get().session)==null?void 0:l.assignee.kind)==="ai",o=this.state.get().messages,r=o.length>0?o[o.length-1]:void 0;if(e&&s||e&&(r==null?void 0:r.type)==="FROM_USER"){console.warn("Cannot send messages while awaiting AI response");return}this.sendMessageAbortController=new AbortController,this.state.setPartial({lastAIResMightSolveUserIssue:!1});try{this.state.setPartial({isSendingMessage:!0});const S=this.toUserMessage(t.content.trim(),t.attachments||void 0),v=this.state.get().messages;if(this.state.setPartial({messages:[...v,S]}),!((c=this.sessionCtx.sessionState.get().session)!=null&&c.id)){if(!await this.sessionCtx.createSession()){console.error("Failed to create session");return}this.sessionCtx.refreshSessions()}const E=(f=this.sessionCtx.sessionState.get().session)==null?void 0:f.id;if(!E)return;const{data:u}=await this.api.sendMessage({uuid:S.id,bot_token:this.config.token,headers:this.config.headers,query_params:this.config.queryParams,session_id:E,content:S.content,attachments:t.attachments,clientContext:this.config.context},this.sendMessageAbortController.signal);if(u!=null&&u.success){const w=this.toBotMessage(u);if(w){const I=this.state.get().messages;if(!!I.some(R=>R.id===w.id)){this.state.setPartial({lastAIResMightSolveUserIssue:((m=u.autopilotResponse)==null?void 0:m.mightSolveUserIssue)||((C=u.uiResponse)==null?void 0:C.mightSolveUserIssue)});return}this.state.setPartial({messages:[...I,w],lastAIResMightSolveUserIssue:((d=u.autopilotResponse)==null?void 0:d.mightSolveUserIssue)||((x=u.uiResponse)==null?void 0:x.mightSolveUserIssue)})}}else{const w=this.toBotErrorMessage(((P=u==null?void 0:u.error)==null?void 0:P.message)||"Unknown error occurred"),I=this.state.get().messages;this.state.setPartial({messages:[...I,w]})}}catch(S){this.sendMessageAbortController.signal.aborted||console.error("Failed to send message:",S)}finally{this.state.setPartial({isSendingMessage:!1})}},this.toUserMessage=(t,s)=>{const e=(()=>{const o=this.contactCtx.state.get().extraCollectedData;return this.state.get().messages.length===0&&o&&Object.keys(o).length>0?`${Object.entries(o).filter(([l,c])=>!!c).map(([l,c])=>`${l}: ${c}`).join(`
2
2
  `)}
3
3
 
4
- ${t}`:t})();return{id:I(),type:"FROM_USER",content:e,deliveredAt:new Date().toISOString(),attachments:s,timestamp:new Date().toISOString()}},this.toBotMessage=t=>{var s;return t.success&&t.autopilotResponse?{type:"FROM_BOT",id:t.autopilotResponse.id||I(),timestamp:new Date().toISOString(),component:"bot_message",agent:this.config.bot?{name:this.config.bot.name||"",isAi:!0,avatar:this.config.bot.avatar||"",id:null}:void 0,data:{message:t.autopilotResponse.value.content,action:(s=t.uiResponse)!=null&&s.value.name?{name:t.uiResponse.value.name,data:t.uiResponse.value.request_response}:void 0}}:null},this.toBotErrorMessage=t=>({type:"FROM_BOT",id:I(),timestamp:new Date().toISOString(),component:"TEXT",data:{message:t,variant:"error",action:void 0}}),this.config=n,this.api=i,this.sessionCtx=h,this.contactCtx=a}}class H{constructor({config:n,contactCtx:i,sessionCtx:h,resetChat:a}){this.registerRoutingListener=()=>{this.contactCtx.state.subscribe(({contact:t})=>{t!=null&&t.token&&this.state.get().screen==="welcome"&&this.state.setPartial({screen:"sessions"})}),this.sessionCtx.sessionsState.subscribe(({isInitialFetchLoading:t,data:s})=>{var e;s.length||((e=this.config.router)==null?void 0:e.goToChatIfNoSessions)!==!1&&!t&&this.state.get().screen!=="chat"&&this.toChatScreen()})},this.toSessionsScreen=()=>{this.resetChat(),this.state.setPartial({screen:"sessions"})},this.toChatScreen=t=>{if(this.resetChat(),t){const s=this.sessionCtx.sessionsState.get().data.find(e=>e.id===t);if(!s)return;this.sessionCtx.sessionState.setPartial({session:s})}this.state.setPartial({screen:"chat"})},this.state=new v({screen:i.shouldCollectData()?"welcome":"sessions"}),this.config=n,this.contactCtx=i,this.sessionCtx=h,this.resetChat=a,this.registerRoutingListener()}}class X{constructor({storage:n}){this.KEYS={contactToken:"opencx__widget__contactToken",externalContactId:"opencx__widget__externalContactId"},this.setContactToken=async i=>{await this.storage.set(this.KEYS.contactToken,i)},this.getContactToken=async()=>this.storage.get(this.KEYS.contactToken),this.setExternalContactId=async i=>{await this.storage.set(this.KEYS.externalContactId,i)},this.getExternalContactId=async()=>this.storage.get(this.KEYS.externalContactId),this.storage=n}}const p=class p{constructor({config:n,storage:i}){if(this.resetChat=()=>{this.sessionCtx.reset(),this.messageCtx.reset()},!p.pollingIntervalsSeconds)throw Error("Widget polling values are not defined, did you call WidgetCtx.initialize()");this.config=n,this.api=new U({config:n}),this.storageCtx=i?new X({storage:i}):void 0,this.contactCtx=new q({api:this.api,config:this.config,storageCtx:this.storageCtx}),this.sessionCtx=new $({api:this.api,contactCtx:this.contactCtx,sessionsPollingIntervalSeconds:p.pollingIntervalsSeconds.sessions}),this.messageCtx=new N({config:this.config,api:this.api,sessionCtx:this.sessionCtx,contactCtx:this.contactCtx}),this.activeSessionPollingCtx=new _({api:this.api,config:this.config,sessionCtx:this.sessionCtx,messageCtx:this.messageCtx,sessionPollingIntervalSeconds:p.pollingIntervalsSeconds.session}),this.routerCtx=new H({config:this.config,contactCtx:this.contactCtx,sessionCtx:this.sessionCtx,resetChat:this.resetChat})}};p.pollingIntervalsSeconds=null,p.initialize=async({config:n,storage:i})=>{var a,t;const h=await new U({config:n}).getExternalWidgetConfig();return p.pollingIntervalsSeconds={session:((a=h.data)==null?void 0:a.sessionPollingIntervalSeconds)||10,sessions:((t=h.data)==null?void 0:t.sessionsPollingIntervalSeconds)||60},new p({config:n,storage:i})};let y=p;exports.PrimitiveState=v;exports.WidgetCtx=y;
5
- //# sourceMappingURL=widget.ctx-BxnHMoxD.cjs.map
4
+ ${t}`:t})();return{id:y(),type:"FROM_USER",content:e,deliveredAt:new Date().toISOString(),attachments:s,timestamp:new Date().toISOString()}},this.toBotMessage=t=>{var s;return t.success&&t.autopilotResponse?{type:"FROM_BOT",id:t.autopilotResponse.id||y(),timestamp:new Date().toISOString(),component:"bot_message",agent:this.config.bot?{name:this.config.bot.name||"",isAi:!0,avatar:this.config.bot.avatar||"",id:null}:void 0,data:{message:t.autopilotResponse.value.content,action:(s=t.uiResponse)!=null&&s.value.name?{name:t.uiResponse.value.name,data:t.uiResponse.value.request_response}:void 0}}:null},this.toBotErrorMessage=t=>({type:"FROM_BOT",id:y(),timestamp:new Date().toISOString(),component:"TEXT",data:{message:t,variant:"error",action:void 0}}),this.config=n,this.api=i,this.sessionCtx=h,this.contactCtx=a}}class N{constructor({config:n,contactCtx:i,sessionCtx:h,resetChat:a}){this.registerRoutingListener=()=>{this.contactCtx.state.subscribe(({contact:t})=>{t!=null&&t.token&&this.state.get().screen==="welcome"&&this.state.setPartial({screen:"sessions"})}),this.sessionCtx.sessionsState.subscribe(({isInitialFetchLoading:t,data:s})=>{var e;s.length||((e=this.config.router)==null?void 0:e.goToChatIfNoSessions)!==!1&&!t&&this.state.get().screen!=="chat"&&this.toChatScreen()})},this.toSessionsScreen=()=>{this.resetChat(),this.state.setPartial({screen:"sessions"})},this.toChatScreen=t=>{if(this.resetChat(),t){const s=this.sessionCtx.sessionsState.get().data.find(e=>e.id===t);if(!s)return;this.sessionCtx.sessionState.setPartial({session:s})}this.state.setPartial({screen:"chat"})},this.state=new b({screen:i.shouldCollectData()?"welcome":"sessions"}),this.config=n,this.contactCtx=i,this.sessionCtx=h,this.resetChat=a,this.registerRoutingListener()}}class H{constructor({storage:n}){this.KEYS={contactToken:"opencx__widget__contactToken",externalContactId:"opencx__widget__externalContactId"},this.setContactToken=async i=>{await this.storage.set(this.KEYS.contactToken,i)},this.getContactToken=async()=>this.storage.get(this.KEYS.contactToken),this.setExternalContactId=async i=>{await this.storage.set(this.KEYS.externalContactId,i)},this.getExternalContactId=async()=>this.storage.get(this.KEYS.externalContactId),this.storage=n}}const p=class p{constructor({config:n,storage:i}){if(this.resetChat=()=>{this.sessionCtx.reset(),this.messageCtx.reset()},!p.pollingIntervalsSeconds)throw Error("Widget polling values are not defined, did you call WidgetCtx.initialize()");this.config=n,this.api=new U({config:n}),this.storageCtx=i?new H({storage:i}):void 0,this.contactCtx=new B({api:this.api,config:this.config,storageCtx:this.storageCtx}),this.sessionCtx=new q({api:this.api,contactCtx:this.contactCtx,sessionsPollingIntervalSeconds:p.pollingIntervalsSeconds.sessions}),this.messageCtx=new $({config:this.config,api:this.api,sessionCtx:this.sessionCtx,contactCtx:this.contactCtx}),this.activeSessionPollingCtx=new L({api:this.api,config:this.config,sessionCtx:this.sessionCtx,messageCtx:this.messageCtx,sessionPollingIntervalSeconds:p.pollingIntervalsSeconds.session}),this.routerCtx=new N({config:this.config,contactCtx:this.contactCtx,sessionCtx:this.sessionCtx,resetChat:this.resetChat})}};p.pollingIntervalsSeconds=null,p.initialize=async({config:n,storage:i})=>{var a,t;const h=await new U({config:n}).getExternalWidgetConfig();return p.pollingIntervalsSeconds={session:((a=h.data)==null?void 0:a.sessionPollingIntervalSeconds)||10,sessions:((t=h.data)==null?void 0:t.sessionsPollingIntervalSeconds)||60},new p({config:n,storage:i})};let k=p;exports.PrimitiveState=b;exports.WidgetCtx=k;
5
+ //# sourceMappingURL=widget.ctx-CpkDy8NJ.cjs.map