@opencx/widget 3.0.17 → 3.0.18
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/designs.cjs +48 -48
- package/dist/designs.cjs.map +1 -1
- package/dist/designs.js +3515 -3506
- package/dist/designs.js.map +1 -1
- package/dist/index.cjs +1 -1
- package/dist/index.js +1 -1
- package/dist/react.cjs +1 -1
- package/dist/react.js +2 -2
- package/dist/src/designs/react/components/SuggestedReplies.d.ts +1 -6
- package/dist/src/designs/translation/translation.types.d.ts +1 -1
- package/dist/src/headless/core/context/message.ctx.d.ts +8 -6
- package/dist/src/headless/core/sdk/schema.d.ts +2 -5
- package/dist/src/headless/react/hooks/useMessages.d.ts +1 -1
- package/dist/{useUploadFiles-LkWYJwRc.js → useUploadFiles-Bw5PXhkI.js} +3 -3
- package/dist/{useUploadFiles-LkWYJwRc.js.map → useUploadFiles-Bw5PXhkI.js.map} +1 -1
- package/dist/{useUploadFiles-CPuop4TB.cjs → useUploadFiles-DnInAFaL.cjs} +2 -2
- package/dist/{useUploadFiles-CPuop4TB.cjs.map → useUploadFiles-DnInAFaL.cjs.map} +1 -1
- package/dist/{widget.ctx-S-HHGJkM.cjs → widget.ctx-BXzbhG8U.cjs} +2 -2
- package/dist/widget.ctx-BXzbhG8U.cjs.map +1 -0
- package/dist/{widget.ctx-DddwF0U6.js → widget.ctx-m7KQtnuH.js} +49 -42
- package/dist/widget.ctx-m7KQtnuH.js.map +1 -0
- package/dist-embed/script.js +36 -36
- package/dist-embed/script.js.map +1 -1
- package/package.json +1 -1
- package/dist/widget.ctx-DddwF0U6.js.map +0 -1
- package/dist/widget.ctx-S-HHGJkM.cjs.map +0 -1
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
"use strict";var I=Object.defineProperty;var A=a=>{throw TypeError(a)};var O=(a,t,e)=>t in a?I(a,t,{enumerable:!0,configurable:!0,writable:!0,value:e}):a[t]=e;var s=(a,t,e)=>O(a,typeof t!="symbol"?t+"":t,e),k=(a,t,e)=>t.has(a)||A("Cannot "+e);var h=(a,t,e)=>(k(a,t,"read from private field"),e?e.call(a):t.get(a)),P=(a,t,e)=>t.has(a)?A("Cannot add the same private member more than once"):t instanceof WeakSet?t.add(a):t.set(a,e),S=(a,t,e,i)=>(k(a,t,"write to private field"),i?i.call(a,e):t.set(a,e),e);const R=require("axios"),L=require("openapi-fetch"),T=require("lodash.isequal"),E=require("uuid"),D=a=>{console.log(a.error)},q=a=>{const t=L({baseUrl:a.baseUrl}),e={onRequest:a.onRequest,onResponse:a.onResponse,onError:a.onError||D};return t.use(e),t};class _{constructor({config:t}){s(this,"client");s(this,"uploadFileClient");s(this,"config");s(this,"constructClientOptions",t=>{const e=this.config.apiUrl||"https://api.open.cx",i={"X-Bot-Token":this.config.token,"Content-Type":"application/json",Accept:"application/json",Authorization:t?`Bearer ${t}`:void 0};return{baseUrl:e,headers:i}});s(this,"createOpenAPIClient",({baseUrl:t,headers:e})=>q({baseUrl:t,onRequest:({request:i})=>{Object.entries(e).forEach(([n,r])=>{r&&i.headers.set(n,r)})}}));s(this,"createAxiosUploadClient",({baseUrl:t,headers:e})=>R.create({baseURL:`${t}/backend/widget/v2/upload`,headers:e}));s(this,"setAuthToken",t=>{const{baseUrl:e,headers:i}=this.constructClientOptions(t);this.client=this.createOpenAPIClient({baseUrl:e,headers:i}),this.uploadFileClient=this.createAxiosUploadClient({baseUrl:e,headers:i})});s(this,"widgetPrelude",async()=>await this.client.GET("/backend/widget/v2/prelude",{params:{header:{"X-Bot-Token":this.config.token}}}));s(this,"sendMessage",async(t,e)=>await this.client.POST("/backend/widget/v2/chat/send",{body:t,signal:e}));s(this,"getSessionHistory",async({sessionId:t,lastMessageTimestamp:e,abortSignal:i})=>{const n=e?{lastMessageTimestamp:e}:void 0;return await this.client.GET("/backend/widget/v2/session/history/{sessionId}",{params:{path:{sessionId:t},query:n},signal:i})});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}));s(this,"createSession",async t=>await this.client.POST("/backend/widget/v2/create-session",{body:t}));s(this,"getSession",async({sessionId:t,abortSignal:e})=>await this.client.GET("/backend/widget/v2/session/{sessionId}",{params:{path:{sessionId:t}},signal:e}));s(this,"getSessions",async({cursor:t,filters:e,abortSignal:i})=>await this.client.GET("/backend/widget/v2/sessions",{params:{query:{cursor:t,filters:JSON.stringify(e)}},signal:i}));s(this,"uploadFile",async(t,e={})=>{const i=new FormData;i.append("file",t.file);const{data:n}=await this.uploadFileClient.post("",i,{headers:{"Content-Type":"multipart/form-data"},...e});return n});s(this,"vote",async t=>await this.client.POST("/backend/widget/v2/chat/vote",{body:t}));var n;this.config=t;const{baseUrl:e,headers:i}=this.constructClientOptions((n=t.user)==null?void 0:n.token);this.client=this.createOpenAPIClient({baseUrl:e,headers:i}),this.uploadFileClient=this.createAxiosUploadClient({baseUrl:e,headers:i})}}var l,m;class d{constructor(t){s(this,"subscribers",new Set);P(this,l);s(this,"initialState");P(this,m);s(this,"lifecycleListeners",new Map);s(this,"emitLifecycle",(t,e)=>{const i=this.lifecycleListeners.get(t);if(i){const n={type:t,timestamp:Date.now(),data:e};i.forEach(r=>{try{r(n)}catch{}})}});s(this,"notifySubscribers",t=>{Array.from(this.subscribers).forEach(i=>{try{i(t)}catch(n){this.emitLifecycle("error",{error:n})}})});s(this,"subscribe",t=>(this.subscribers.add(t),()=>{this.subscribers.delete(t)}));s(this,"onLifecycle",(t,e)=>{this.lifecycleListeners.has(t)||this.lifecycleListeners.set(t,new Set);const i=this.lifecycleListeners.get(t);return i.add(e),()=>{i.delete(e),i.size===0&&this.lifecycleListeners.delete(t)}});s(this,"get",()=>h(this,l));s(this,"set",t=>{this.emitLifecycle("beforeUpdate",{previousState:h(this,l),nextState:t}),T(h(this,l),t)||(S(this,l,t),S(this,m,Date.now()),this.emitLifecycle("stateChange",{state:t}),this.notifySubscribers(t)),this.emitLifecycle("afterUpdate",{state:t})});s(this,"setPartial",t=>{if(t==null)return;const e={...h(this,l),...t};this.set(e)});s(this,"clear",()=>{this.emitLifecycle("destroy"),this.subscribers=new Set,this.lifecycleListeners=new Map});s(this,"reset",()=>{this.set(this.initialState)});s(this,"lastUpdated",()=>h(this,m));S(this,l,t),this.initialState=t,S(this,m,Date.now()),this.emitLifecycle("init",{initialState:h(this,l)})}}l=new WeakMap,m=new WeakMap;class B{constructor({config:t,api:e}){s(this,"config");s(this,"api");s(this,"state");s(this,"shouldCollectData",()=>{var e;return!!(!((e=this.state.get().contact)!=null&&e.token)&&this.config.collectUserData)});s(this,"autoCreateUnverifiedUser",async()=>{var t,e,i,n;await this.createUnverifiedContact({name:((e=(t=this.config.user)==null?void 0:t.data)==null?void 0:e.name)||"Anonymous",email:(n=(i=this.config.user)==null?void 0:i.data)==null?void 0:n.email})});s(this,"createUnverifiedContact",async t=>{try{this.state.setPartial({isCreatingUnverifiedContact:!0,isErrorCreatingUnverifiedContact:!1});const{data:e}=await this.api.createUnverifiedContact(t);e!=null&&e.token?(this.api.setAuthToken(e.token),this.state.setPartial({contact:{token:e.token}})):this.state.setPartial({isErrorCreatingUnverifiedContact:!0})}finally{this.state.setPartial({isCreatingUnverifiedContact:!1})}});var i,n,r;this.config=t,this.api=e,this.state=new d({contact:(i=t.user)!=null&&i.token?{token:(n=t.user)==null?void 0:n.token}:null,isCreatingUnverifiedContact:!1,isErrorCreatingUnverifiedContact:!1}),!((r=t.user)!=null&&r.token)&&!t.collectUserData&&this.autoCreateUnverifiedUser()}}class M{constructor(){s(this,"state",new d({isPolling:!1,isError:!1}));s(this,"abortController",new AbortController);s(this,"reset",()=>{var t;this.abortController.abort("Resetting poller"),(t=this.stopPolling)==null||t.call(this),this.stopPolling=null});s(this,"stopPolling",null);s(this,"startPolling",(t,e)=>{if(this.stopPolling)return;const i=[],n=async()=>{this.abortController=new AbortController,this.state.setPartial({isPolling:!0});try{await t(this.abortController.signal)}catch(r){if(this.abortController.signal.aborted)return;console.error("Failed to poll:",r),this.state.setPartial({isError:!0})}finally{this.state.setPartial({isPolling:!1})}this.abortController.signal.aborted?console.log("Poller aborted, not scheduling anymore"):i.push(setTimeout(n,e))};n(),this.stopPolling=()=>{i.forEach(clearTimeout),this.state.reset()}})}}function y(){return E.v4()}class H{constructor({config:t,api:e,contactCtx:i}){s(this,"config");s(this,"api");s(this,"contactCtx");s(this,"poller",new M);s(this,"sessionState",new d({session:null,isCreatingSession:!1}));s(this,"sessionsState",new d({data:[],cursor:void 0,isLastPage:!1,didStartInitialFetch:!1,isInitialFetchLoading:!0}));s(this,"reset",async()=>{this.sessionState.reset(),this.poller.reset()});s(this,"registerPolling",()=>{this.sessionState.subscribe(({session:t})=>{t!=null&&t.id?this.poller.startPolling(async e=>{const{data:i}=await this.api.getSession({sessionId:t.id,abortSignal:e});i&&this.sessionState.setPartial({session:i})},1e3):this.poller.reset()})});s(this,"registerInitialSessionsFetch",()=>{var e;const t=async()=>{this.sessionsState.setPartial({didStartInitialFetch:!0}),await this.loadMoreSessions(),this.sessionsState.setPartial({isInitialFetchLoading:!1})};(e=this.contactCtx.state.get().contact)!=null&&e.token&&!this.sessionsState.get().didStartInitialFetch&&t(),this.contactCtx.state.subscribe(({contact:i})=>{i!=null&&i.token&&!this.sessionsState.get().didStartInitialFetch&&t()})});s(this,"createSession",async()=>{var i,n;this.sessionState.setPartial({session:null,isCreatingSession:!0});const{data:t,error:e}=await this.api.createSession({customData:(i=this.config.user)!=null&&i.externalId?{external_id:(n=this.config.user)==null?void 0:n.externalId}:void 0});return t?(this.sessionState.setPartial({session:t,isCreatingSession:!1}),t):(console.error("Failed to create session:",e),null)});s(this,"loadMoreSessions",async()=>{var e,i;if(this.sessionsState.get().isLastPage||!((e=this.contactCtx.state.get().contact)!=null&&e.token))return;const{data:t}=await this.api.getSessions({cursor:this.sessionsState.get().cursor,filters:(i=this.config.user)!=null&&i.externalId?{external_id:this.config.user.externalId}:{}});if(t){const r=[...this.sessionsState.get().data,...t.items].filter((c,g,u)=>g===u.findIndex(f=>c.id===f.id));this.sessionsState.setPartial({data:r,cursor:t.next||void 0,isLastPage:t.next===null})}});this.config=t,this.api=e,this.contactCtx=i,this.registerPolling(),this.registerInitialSessionsFetch()}}class C{constructor({config:t,api:e,sessionCtx:i}){s(this,"config");s(this,"api");s(this,"sessionCtx");s(this,"poller",new M);s(this,"state",new d({messages:[],isSendingMessage:!1,suggestedReplies:null,isInitialFetchLoading:!1}));s(this,"sendMessageAbortController",new AbortController);s(this,"reset",()=>{this.sendMessageAbortController.abort("Resetting chat"),this.state.reset(),this.poller.reset()});s(this,"registerPolling",()=>{this.sessionCtx.sessionState.subscribe(({session:t})=>{t!=null&&t.id?this.poller.startPolling(async e=>{await this.fetchAndSetHistory(t.id,e)},1e3):this.poller.reset()})});s(this,"sendMessage",async t=>{var n,r,c,g,u,f,x;this.sendMessageAbortController=new AbortController;const e=this.state.get().isSendingMessage,i=((n=this.sessionCtx.sessionState.get().session)==null?void 0:n.assignee.kind)==="ai";if(e&&i){console.warn("Cannot send messages while awaiting AI response");return}this.state.setPartial({suggestedReplies:null});try{this.state.setPartial({isSendingMessage:!0});const b=C.toUserMessage(t.content,t.attachments||void 0),U=this.state.get().messages;if(this.state.setPartial({messages:[...U,b]}),!((r=this.sessionCtx.sessionState.get().session)!=null&&r.id)&&!await this.sessionCtx.createSession()){console.error("Failed to create session");return}const v=(c=this.sessionCtx.sessionState.get().session)==null?void 0:c.id;if(!v)return;const{data:o}=await this.api.sendMessage({uuid:b.id,bot_token:this.config.token,headers:this.config.headers,query_params:this.config.queryParams,session_id:v,user:(g=this.config.user)==null?void 0:g.data,...t},this.sendMessageAbortController.signal);if(o!=null&&o.success){const p=C.toBotMessage(o);if(p){const w=this.state.get().messages;if(!!w.some(F=>F.id===p.id))return;this.state.setPartial({messages:[...w,p]})}else(u=o.options)!=null&&u.value&&((f=o.options)==null?void 0:f.value.length)>0&&this.state.setPartial({suggestedReplies:o.options.value})}else{const p=C.toErrorMessage(((x=o==null?void 0:o.error)==null?void 0:x.message)||"Unknown error occurred"),w=this.state.get().messages;this.state.setPartial({messages:[...w,p]})}}catch(b){this.sendMessageAbortController.signal.aborted||console.error("Failed to send message:",b)}finally{this.state.setPartial({isSendingMessage:!1})}});s(this,"fetchAndSetHistory",async(t,e)=>{var r;this.state.get().messages.length===0&&this.state.setPartial({isInitialFetchLoading:!0});const i=(r=this.state.get().messages.at(-1))==null?void 0:r.timestamp,{data:n}=await this.api.getSessionHistory({sessionId:t,lastMessageTimestamp:i,abortSignal:e});if(n&&n.length>0){const c=this.state.get().messages,g=n.map(C.mapHistoryToMessage).filter(u=>!c.some(f=>f.id===u.id));this.state.setPartial({messages:[...c,...g]})}this.state.get().isInitialFetchLoading&&this.state.setPartial({isInitialFetchLoading:!1})});this.config=t,this.api=e,this.sessionCtx=i,this.registerPolling()}static mapHistoryToMessage(t){const e={id:t.publicId,timestamp:t.sentAt||"",attachments:t.attachments||void 0};return t.sender.kind==="user"?{...e,type:"FROM_USER",content:t.content.text||"",deliveredAt:t.sentAt||""}:t.sender.kind==="agent"?{...e,type:"FROM_AGENT",component:"agent_message",data:{message:t.content.text||""}}:{...e,type:"FROM_BOT",component:"bot_message",agent:{id:null,name:t.sender.name||"",isAi:t.sender.kind==="ai",avatar:t.sender.avatar||null},data:{message:t.content.text}}}static toUserMessage(t,e){return{id:y(),type:"FROM_USER",content:t,deliveredAt:new Date().toISOString(),attachments:e,timestamp:new Date().toISOString()}}static toBotMessage(t){if(t.success&&t.autopilotResponse)return{type:"FROM_BOT",id:t.autopilotResponse.id||y(),timestamp:new Date().toISOString(),component:"bot_message",data:{message:t.autopilotResponse.value.content}};if(t.success&&t.uiResponse){const e=t.uiResponse.value;return{type:"FROM_BOT",id:y(),timestamp:new Date().toISOString(),component:e.name,data:e.request_response}}return null}static toErrorMessage(t){return{type:"FROM_BOT",id:y(),timestamp:new Date().toISOString(),component:"TEXT",data:{message:t,variant:"error"}}}}class G{constructor({contactCtx:t,sessionCtx:e,resetChat:i}){s(this,"state");s(this,"contactCtx");s(this,"sessionCtx");s(this,"resetChat");s(this,"registerRoutingListener",()=>{this.contactCtx.state.subscribe(({contact:t})=>{t!=null&&t.token&&this.state.get().screen==="welcome"&&this.state.setPartial({screen:"sessions"})})});s(this,"toSessionsScreen",()=>{this.resetChat(),this.state.setPartial({screen:"sessions"})});s(this,"toChatScreen",t=>{if(this.resetChat(),t){const e=this.sessionCtx.sessionsState.get().data.find(i=>i.id===t);if(!e)return;this.sessionCtx.sessionState.setPartial({session:e})}this.state.setPartial({screen:"chat"})});this.state=new d({screen:t.shouldCollectData()?"welcome":"sessions"}),this.contactCtx=t,this.sessionCtx=e,this.resetChat=i,this.registerRoutingListener()}}class j{constructor({config:t}){s(this,"config");s(this,"api");s(this,"contactCtx");s(this,"sessionCtx");s(this,"messageCtx");s(this,"routerCtx");s(this,"resetChat",()=>{this.sessionCtx.reset(),this.messageCtx.reset()});this.config=t,this.api=new _({config:t}),this.contactCtx=new B({api:this.api,config:this.config}),this.sessionCtx=new H({config:this.config,api:this.api,contactCtx:this.contactCtx}),this.messageCtx=new C({config:this.config,api:this.api,sessionCtx:this.sessionCtx}),this.routerCtx=new G({contactCtx:this.contactCtx,sessionCtx:this.sessionCtx,resetChat:this.resetChat})}}exports.PrimitiveState=d;exports.WidgetCtx=j;
|
|
2
|
-
//# sourceMappingURL=widget.ctx-
|
|
1
|
+
"use strict";var O=Object.defineProperty;var A=a=>{throw TypeError(a)};var L=(a,t,e)=>t in a?O(a,t,{enumerable:!0,configurable:!0,writable:!0,value:e}):a[t]=e;var s=(a,t,e)=>L(a,typeof t!="symbol"?t+"":t,e),M=(a,t,e)=>t.has(a)||A("Cannot "+e);var h=(a,t,e)=>(M(a,t,"read from private field"),e?e.call(a):t.get(a)),P=(a,t,e)=>t.has(a)?A("Cannot add the same private member more than once"):t instanceof WeakSet?t.add(a):t.set(a,e),S=(a,t,e,i)=>(M(a,t,"write to private field"),i?i.call(a,e):t.set(a,e),e);const T=require("axios"),E=require("openapi-fetch"),D=require("lodash.isequal"),q=require("uuid"),_=a=>{console.log(a.error)},B=a=>{const t=E({baseUrl:a.baseUrl}),e={onRequest:a.onRequest,onResponse:a.onResponse,onError:a.onError||_};return t.use(e),t};class H{constructor({config:t}){s(this,"client");s(this,"uploadFileClient");s(this,"config");s(this,"constructClientOptions",t=>{const e=this.config.apiUrl||"https://api.open.cx",i={"X-Bot-Token":this.config.token,"Content-Type":"application/json",Accept:"application/json",Authorization:t?`Bearer ${t}`:void 0};return{baseUrl:e,headers:i}});s(this,"createOpenAPIClient",({baseUrl:t,headers:e})=>B({baseUrl:t,onRequest:({request:i})=>{Object.entries(e).forEach(([n,r])=>{r&&i.headers.set(n,r)})}}));s(this,"createAxiosUploadClient",({baseUrl:t,headers:e})=>T.create({baseURL:`${t}/backend/widget/v2/upload`,headers:e}));s(this,"setAuthToken",t=>{const{baseUrl:e,headers:i}=this.constructClientOptions(t);this.client=this.createOpenAPIClient({baseUrl:e,headers:i}),this.uploadFileClient=this.createAxiosUploadClient({baseUrl:e,headers:i})});s(this,"widgetPrelude",async()=>await this.client.GET("/backend/widget/v2/prelude",{params:{header:{"X-Bot-Token":this.config.token}}}));s(this,"sendMessage",async(t,e)=>await this.client.POST("/backend/widget/v2/chat/send",{body:t,signal:e}));s(this,"getSessionHistory",async({sessionId:t,lastMessageTimestamp:e,abortSignal:i})=>{const n=e?{lastMessageTimestamp:e}:void 0;return await this.client.GET("/backend/widget/v2/session/history/{sessionId}",{params:{path:{sessionId:t},query:n},signal:i})});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}));s(this,"createSession",async t=>await this.client.POST("/backend/widget/v2/create-session",{body:t}));s(this,"getSession",async({sessionId:t,abortSignal:e})=>await this.client.GET("/backend/widget/v2/session/{sessionId}",{params:{path:{sessionId:t}},signal:e}));s(this,"getSessions",async({cursor:t,filters:e,abortSignal:i})=>await this.client.GET("/backend/widget/v2/sessions",{params:{query:{cursor:t,filters:JSON.stringify(e)}},signal:i}));s(this,"uploadFile",async(t,e={})=>{const i=new FormData;i.append("file",t.file);const{data:n}=await this.uploadFileClient.post("",i,{headers:{"Content-Type":"multipart/form-data"},...e});return n});s(this,"vote",async t=>await this.client.POST("/backend/widget/v2/chat/vote",{body:t}));var n;this.config=t;const{baseUrl:e,headers:i}=this.constructClientOptions((n=t.user)==null?void 0:n.token);this.client=this.createOpenAPIClient({baseUrl:e,headers:i}),this.uploadFileClient=this.createAxiosUploadClient({baseUrl:e,headers:i})}}var l,m;class d{constructor(t){s(this,"subscribers",new Set);P(this,l);s(this,"initialState");P(this,m);s(this,"lifecycleListeners",new Map);s(this,"emitLifecycle",(t,e)=>{const i=this.lifecycleListeners.get(t);if(i){const n={type:t,timestamp:Date.now(),data:e};i.forEach(r=>{try{r(n)}catch{}})}});s(this,"notifySubscribers",t=>{Array.from(this.subscribers).forEach(i=>{try{i(t)}catch(n){this.emitLifecycle("error",{error:n})}})});s(this,"subscribe",t=>(this.subscribers.add(t),()=>{this.subscribers.delete(t)}));s(this,"onLifecycle",(t,e)=>{this.lifecycleListeners.has(t)||this.lifecycleListeners.set(t,new Set);const i=this.lifecycleListeners.get(t);return i.add(e),()=>{i.delete(e),i.size===0&&this.lifecycleListeners.delete(t)}});s(this,"get",()=>h(this,l));s(this,"set",t=>{this.emitLifecycle("beforeUpdate",{previousState:h(this,l),nextState:t}),D(h(this,l),t)||(S(this,l,t),S(this,m,Date.now()),this.emitLifecycle("stateChange",{state:t}),this.notifySubscribers(t)),this.emitLifecycle("afterUpdate",{state:t})});s(this,"setPartial",t=>{if(t==null)return;const e={...h(this,l),...t};this.set(e)});s(this,"clear",()=>{this.emitLifecycle("destroy"),this.subscribers=new Set,this.lifecycleListeners=new Map});s(this,"reset",()=>{this.set(this.initialState)});s(this,"lastUpdated",()=>h(this,m));S(this,l,t),this.initialState=t,S(this,m,Date.now()),this.emitLifecycle("init",{initialState:h(this,l)})}}l=new WeakMap,m=new WeakMap;class G{constructor({config:t,api:e}){s(this,"config");s(this,"api");s(this,"state");s(this,"shouldCollectData",()=>{var e;return!!(!((e=this.state.get().contact)!=null&&e.token)&&this.config.collectUserData)});s(this,"autoCreateUnverifiedUser",async()=>{var t,e,i,n;await this.createUnverifiedContact({name:((e=(t=this.config.user)==null?void 0:t.data)==null?void 0:e.name)||"Anonymous",email:(n=(i=this.config.user)==null?void 0:i.data)==null?void 0:n.email})});s(this,"createUnverifiedContact",async t=>{try{this.state.setPartial({isCreatingUnverifiedContact:!0,isErrorCreatingUnverifiedContact:!1});const{data:e}=await this.api.createUnverifiedContact(t);e!=null&&e.token?(this.api.setAuthToken(e.token),this.state.setPartial({contact:{token:e.token}})):this.state.setPartial({isErrorCreatingUnverifiedContact:!0})}finally{this.state.setPartial({isCreatingUnverifiedContact:!1})}});var i,n,r;this.config=t,this.api=e,this.state=new d({contact:(i=t.user)!=null&&i.token?{token:(n=t.user)==null?void 0:n.token}:null,isCreatingUnverifiedContact:!1,isErrorCreatingUnverifiedContact:!1}),!((r=t.user)!=null&&r.token)&&!t.collectUserData&&this.autoCreateUnverifiedUser()}}class k{constructor(){s(this,"state",new d({isPolling:!1,isError:!1}));s(this,"abortController",new AbortController);s(this,"reset",()=>{var t;this.abortController.abort("Resetting poller"),(t=this.stopPolling)==null||t.call(this),this.stopPolling=null});s(this,"stopPolling",null);s(this,"startPolling",(t,e)=>{if(this.stopPolling)return;const i=[],n=async()=>{this.abortController=new AbortController,this.state.setPartial({isPolling:!0});try{await t(this.abortController.signal)}catch(r){if(this.abortController.signal.aborted)return;console.error("Failed to poll:",r),this.state.setPartial({isError:!0})}finally{this.state.setPartial({isPolling:!1})}this.abortController.signal.aborted?console.log("Poller aborted, not scheduling anymore"):i.push(setTimeout(n,e))};n(),this.stopPolling=()=>{i.forEach(clearTimeout),this.state.reset()}})}}function y(){return q.v4()}class j{constructor({config:t,api:e,contactCtx:i}){s(this,"config");s(this,"api");s(this,"contactCtx");s(this,"poller",new k);s(this,"sessionState",new d({session:null,isCreatingSession:!1}));s(this,"sessionsState",new d({data:[],cursor:void 0,isLastPage:!1,didStartInitialFetch:!1,isInitialFetchLoading:!0}));s(this,"reset",async()=>{this.sessionState.reset(),this.poller.reset()});s(this,"registerPolling",()=>{this.sessionState.subscribe(({session:t})=>{t!=null&&t.id?this.poller.startPolling(async e=>{const{data:i}=await this.api.getSession({sessionId:t.id,abortSignal:e});i&&this.sessionState.setPartial({session:i})},1e3):this.poller.reset()})});s(this,"registerInitialSessionsFetch",()=>{var e;const t=async()=>{this.sessionsState.setPartial({didStartInitialFetch:!0}),await this.loadMoreSessions(),this.sessionsState.setPartial({isInitialFetchLoading:!1})};(e=this.contactCtx.state.get().contact)!=null&&e.token&&!this.sessionsState.get().didStartInitialFetch&&t(),this.contactCtx.state.subscribe(({contact:i})=>{i!=null&&i.token&&!this.sessionsState.get().didStartInitialFetch&&t()})});s(this,"createSession",async()=>{var i,n;this.sessionState.setPartial({session:null,isCreatingSession:!0});const{data:t,error:e}=await this.api.createSession({customData:(i=this.config.user)!=null&&i.externalId?{external_id:(n=this.config.user)==null?void 0:n.externalId}:void 0});return t?(this.sessionState.setPartial({session:t,isCreatingSession:!1}),t):(console.error("Failed to create session:",e),null)});s(this,"loadMoreSessions",async()=>{var e,i;if(this.sessionsState.get().isLastPage||!((e=this.contactCtx.state.get().contact)!=null&&e.token))return;const{data:t}=await this.api.getSessions({cursor:this.sessionsState.get().cursor,filters:(i=this.config.user)!=null&&i.externalId?{external_id:this.config.user.externalId}:{}});if(t){const r=[...this.sessionsState.get().data,...t.items].filter((c,g,u)=>g===u.findIndex(f=>c.id===f.id));this.sessionsState.setPartial({data:r,cursor:t.next||void 0,isLastPage:t.next===null})}});this.config=t,this.api=e,this.contactCtx=i,this.registerPolling(),this.registerInitialSessionsFetch()}}class C{constructor({config:t,api:e,sessionCtx:i}){s(this,"config");s(this,"api");s(this,"sessionCtx");s(this,"poller",new k);s(this,"state",new d({messages:[],isSendingMessage:!1,lastAIResMightSolveUserIssue:!1,isInitialFetchLoading:!1}));s(this,"sendMessageAbortController",new AbortController);s(this,"reset",()=>{this.sendMessageAbortController.abort("Resetting chat"),this.state.reset(),this.poller.reset()});s(this,"registerPolling",()=>{this.sessionCtx.sessionState.subscribe(({session:t})=>{t!=null&&t.id?this.poller.startPolling(async e=>{await this.fetchAndSetHistory(t.id,e)},1e3):this.poller.reset()})});s(this,"sendMessage",async t=>{var n,r,c,g,u,f,x,v,U;this.sendMessageAbortController=new AbortController;const e=this.state.get().isSendingMessage,i=((n=this.sessionCtx.sessionState.get().session)==null?void 0:n.assignee.kind)==="ai";if(e&&i){console.warn("Cannot send messages while awaiting AI response");return}this.state.setPartial({lastAIResMightSolveUserIssue:!1});try{this.state.setPartial({isSendingMessage:!0});const b=C.toUserMessage(t.content,t.attachments||void 0),R=this.state.get().messages;if(this.state.setPartial({messages:[...R,b]}),!((r=this.sessionCtx.sessionState.get().session)!=null&&r.id)&&!await this.sessionCtx.createSession()){console.error("Failed to create session");return}const I=(c=this.sessionCtx.sessionState.get().session)==null?void 0:c.id;if(!I)return;const{data:o}=await this.api.sendMessage({uuid:b.id,bot_token:this.config.token,headers:this.config.headers,query_params:this.config.queryParams,session_id:I,user:(g=this.config.user)==null?void 0:g.data,...t},this.sendMessageAbortController.signal);if(o!=null&&o.success){const p=C.toBotMessage(o);if(p){const w=this.state.get().messages;if(!!w.some(F=>F.id===p.id)){this.state.setPartial({lastAIResMightSolveUserIssue:((u=o.autopilotResponse)==null?void 0:u.mightSolveUserIssue)||((f=o.uiResponse)==null?void 0:f.mightSolveUserIssue)});return}this.state.setPartial({messages:[...w,p],lastAIResMightSolveUserIssue:((x=o.autopilotResponse)==null?void 0:x.mightSolveUserIssue)||((v=o.uiResponse)==null?void 0:v.mightSolveUserIssue)})}}else{const p=C.toErrorMessage(((U=o==null?void 0:o.error)==null?void 0:U.message)||"Unknown error occurred"),w=this.state.get().messages;this.state.setPartial({messages:[...w,p]})}}catch(b){this.sendMessageAbortController.signal.aborted||console.error("Failed to send message:",b)}finally{this.state.setPartial({isSendingMessage:!1})}});s(this,"fetchAndSetHistory",async(t,e)=>{var r;this.state.get().messages.length===0&&this.state.setPartial({isInitialFetchLoading:!0});const i=(r=this.state.get().messages.at(-1))==null?void 0:r.timestamp,{data:n}=await this.api.getSessionHistory({sessionId:t,lastMessageTimestamp:i,abortSignal:e});if(n&&n.length>0){const c=this.state.get().messages,g=n.map(C.mapHistoryToMessage).filter(u=>!c.some(f=>f.id===u.id));this.state.setPartial({messages:[...c,...g]})}this.state.get().isInitialFetchLoading&&this.state.setPartial({isInitialFetchLoading:!1})});this.config=t,this.api=e,this.sessionCtx=i,this.registerPolling()}static mapHistoryToMessage(t){const e={id:t.publicId,timestamp:t.sentAt||"",attachments:t.attachments||void 0};return t.sender.kind==="user"?{...e,type:"FROM_USER",content:t.content.text||"",deliveredAt:t.sentAt||""}:t.sender.kind==="agent"?{...e,type:"FROM_AGENT",component:"agent_message",data:{message:t.content.text||""}}:{...e,type:"FROM_BOT",component:"bot_message",agent:{id:null,name:t.sender.name||"",isAi:t.sender.kind==="ai",avatar:t.sender.avatar||null},data:{message:t.content.text}}}static toUserMessage(t,e){return{id:y(),type:"FROM_USER",content:t,deliveredAt:new Date().toISOString(),attachments:e,timestamp:new Date().toISOString()}}static toBotMessage(t){if(t.success&&t.autopilotResponse)return{type:"FROM_BOT",id:t.autopilotResponse.id||y(),timestamp:new Date().toISOString(),component:"bot_message",data:{message:t.autopilotResponse.value.content}};if(t.success&&t.uiResponse){const e=t.uiResponse.value;return{type:"FROM_BOT",id:y(),timestamp:new Date().toISOString(),component:e.name,data:e.request_response}}return null}static toErrorMessage(t){return{type:"FROM_BOT",id:y(),timestamp:new Date().toISOString(),component:"TEXT",data:{message:t,variant:"error"}}}}class X{constructor({contactCtx:t,sessionCtx:e,resetChat:i}){s(this,"state");s(this,"contactCtx");s(this,"sessionCtx");s(this,"resetChat");s(this,"registerRoutingListener",()=>{this.contactCtx.state.subscribe(({contact:t})=>{t!=null&&t.token&&this.state.get().screen==="welcome"&&this.state.setPartial({screen:"sessions"})})});s(this,"toSessionsScreen",()=>{this.resetChat(),this.state.setPartial({screen:"sessions"})});s(this,"toChatScreen",t=>{if(this.resetChat(),t){const e=this.sessionCtx.sessionsState.get().data.find(i=>i.id===t);if(!e)return;this.sessionCtx.sessionState.setPartial({session:e})}this.state.setPartial({screen:"chat"})});this.state=new d({screen:t.shouldCollectData()?"welcome":"sessions"}),this.contactCtx=t,this.sessionCtx=e,this.resetChat=i,this.registerRoutingListener()}}class ${constructor({config:t}){s(this,"config");s(this,"api");s(this,"contactCtx");s(this,"sessionCtx");s(this,"messageCtx");s(this,"routerCtx");s(this,"resetChat",()=>{this.sessionCtx.reset(),this.messageCtx.reset()});this.config=t,this.api=new H({config:t}),this.contactCtx=new G({api:this.api,config:this.config}),this.sessionCtx=new j({config:this.config,api:this.api,contactCtx:this.contactCtx}),this.messageCtx=new C({config:this.config,api:this.api,sessionCtx:this.sessionCtx}),this.routerCtx=new X({contactCtx:this.contactCtx,sessionCtx:this.sessionCtx,resetChat:this.resetChat})}}exports.PrimitiveState=d;exports.WidgetCtx=$;
|
|
2
|
+
//# sourceMappingURL=widget.ctx-BXzbhG8U.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"widget.ctx-BXzbhG8U.cjs","sources":["../src/headless/core/sdk/index.ts","../src/headless/core/api.ts","../src/headless/core/utils/PrimitiveState.ts","../src/headless/core/context/contact.ctx.ts","../src/headless/core/utils/Poller.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/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 axios, { type AxiosInstance, type AxiosRequestConfig } from \"axios\";\nimport { type Dto, type Endpoint, basicClient } from \"./sdk\";\nimport type { WidgetConfig } from \"./types/WidgetConfig\";\nimport type { SendMessageDto, VoteInputDto } from \"./types/schemas\";\n\nexport class ApiCaller {\n private client: ReturnType<typeof basicClient>;\n private uploadFileClient: AxiosInstance;\n private config: WidgetConfig;\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 this.uploadFileClient = this.createAxiosUploadClient({ baseUrl, headers });\n }\n\n private constructClientOptions = (token: string | null | undefined) => {\n const baseUrl = 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 private createAxiosUploadClient = ({\n baseUrl,\n headers,\n }: ReturnType<typeof this.constructClientOptions>) => {\n const uploadPath = \"/backend/widget/v2/upload\" satisfies Endpoint;\n return axios.create({\n baseURL: `${baseUrl}${uploadPath}`,\n headers,\n });\n };\n\n setAuthToken = (token: string) => {\n const { baseUrl, headers } = this.constructClientOptions(token);\n this.client = this.createOpenAPIClient({ baseUrl, headers });\n this.uploadFileClient = this.createAxiosUploadClient({ baseUrl, headers });\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 getSessionHistory = 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(\n \"/backend/widget/v2/session/history/{sessionId}\",\n { params: { path: { sessionId }, query }, 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[\"CreateWidgetChatSessionDto\"]) => {\n return await this.client.POST(\"/backend/widget/v2/create-session\", {\n body,\n });\n };\n\n getSession = async ({\n sessionId,\n abortSignal,\n }: { sessionId: string; abortSignal: AbortSignal }) => {\n return await this.client.GET(\"/backend/widget/v2/session/{sessionId}\", {\n params: { path: { sessionId } },\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 uploadFile = async (\n file: {\n id: string;\n file: File;\n },\n config: Partial<AxiosRequestConfig> = {},\n ) => {\n const formData = new FormData();\n formData.append(\"file\", file.file);\n\n // Couldn't get this to work with the openapi client... dunno why...\n const { data } = await this.uploadFileClient.post<\n Dto[\"UploadWidgetFileResponseDto\"]\n >(\"\", formData, {\n headers: {\n \"Content-Type\": \"multipart/form-data\",\n },\n ...config,\n });\n return data;\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 enum LifecycleEvent {\n INIT = \"init\",\n STATE_CHANGE = \"stateChange\",\n BEFORE_UPDATE = \"beforeUpdate\",\n AFTER_UPDATE = \"afterUpdate\",\n DESTROY = \"destroy\",\n ERROR = \"error\",\n}\n\ntype LifecycleListener = (event: {\n type: LifecycleEvent;\n timestamp: number;\n data?: any;\n}) => void;\n\nexport class PrimitiveState<S> {\n private subscribers = new Set<Subscriber<S>>();\n #state: S;\n private initialState: S;\n #lastUpdated: number | null;\n private lifecycleListeners: Map<LifecycleEvent, Set<LifecycleListener>> =\n new Map();\n\n constructor(state: S) {\n this.#state = state;\n this.initialState = state;\n this.#lastUpdated = Date.now();\n this.emitLifecycle(LifecycleEvent.INIT, { initialState: this.#state });\n }\n\n private emitLifecycle = (event: LifecycleEvent, data?: any) => {\n const listeners = this.lifecycleListeners.get(event);\n if (listeners) {\n const eventData = {\n type: event,\n timestamp: Date.now(),\n data,\n };\n listeners.forEach((listener) => {\n try {\n listener(eventData);\n } catch {\n // ignore error\n }\n });\n }\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 this.emitLifecycle(LifecycleEvent.ERROR, { error });\n }\n });\n };\n\n /**\n * Subscribe to state changes\n * @param callback Function to call when state changes\n * @returns Unsubscribe function\n */\n subscribe = (callback: Subscriber<S>): (() => void) => {\n this.subscribers.add(callback);\n // Don't call the callback immediately with current state\n return () => {\n this.subscribers.delete(callback);\n };\n };\n\n onLifecycle = (\n event: LifecycleEvent,\n listener: LifecycleListener,\n ): (() => void) => {\n if (!this.lifecycleListeners.has(event)) {\n this.lifecycleListeners.set(event, new Set());\n }\n const listeners = this.lifecycleListeners.get(event)!;\n listeners.add(listener);\n\n return () => {\n listeners.delete(listener);\n if (listeners.size === 0) {\n this.lifecycleListeners.delete(event);\n }\n };\n };\n\n /** Get the current state */\n get = (): S => {\n return this.#state;\n };\n\n // TODO make this provide prev state\n /**\n * Set the state and notify subscribers if the state changes\n * @param newState The new state to set\n */\n set = (newState: S): void => {\n this.emitLifecycle(LifecycleEvent.BEFORE_UPDATE, {\n previousState: this.#state,\n nextState: newState,\n });\n\n if (!isEqual(this.#state, newState)) {\n this.#state = newState;\n this.#lastUpdated = Date.now();\n this.emitLifecycle(LifecycleEvent.STATE_CHANGE, { state: newState });\n this.notifySubscribers(newState);\n }\n\n this.emitLifecycle(LifecycleEvent.AFTER_UPDATE, { state: newState });\n };\n\n // TODO make this provide prev state\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 /**\n * Clear all subscriptions\n */\n clear = (): void => {\n this.emitLifecycle(LifecycleEvent.DESTROY);\n this.subscribers = new Set(); // Create a new Set instead of just clearing\n this.lifecycleListeners = new Map();\n };\n\n reset = (): void => {\n this.set(this.initialState);\n };\n\n lastUpdated = (): number | null => {\n return this.#lastUpdated;\n };\n}\n","import { PrimitiveState } from \"../utils/PrimitiveState\";\nimport { ApiCaller } from \"../api\";\nimport { type WidgetConfig } from \"../types/WidgetConfig\";\nimport { type Dto } from \"../sdk\";\n\ntype ContactState = {\n contact: { token: string } | null;\n isCreatingUnverifiedContact: boolean;\n isErrorCreatingUnverifiedContact: boolean;\n};\n\nexport class ContactCtx {\n private config: WidgetConfig;\n private api: ApiCaller;\n state: PrimitiveState<ContactState>;\n\n constructor({\n config,\n api,\n }: {\n api: ApiCaller;\n config: WidgetConfig;\n }) {\n this.config = config;\n this.api = api;\n\n this.state = new PrimitiveState<ContactState>({\n contact: config.user?.token ? { token: config.user?.token } : null,\n isCreatingUnverifiedContact: false,\n isErrorCreatingUnverifiedContact: false,\n });\n\n if (!config.user?.token && !config.collectUserData) {\n this.autoCreateUnverifiedUser();\n }\n }\n\n shouldCollectData = (): boolean => {\n const currentState = this.state.get();\n\n if (!currentState.contact?.token && this.config.collectUserData) {\n return true;\n } else {\n return false;\n }\n };\n\n private autoCreateUnverifiedUser = async () => {\n await this.createUnverifiedContact({\n name: this.config.user?.data?.name || \"Anonymous\",\n email: this.config.user?.data?.email,\n });\n };\n\n createUnverifiedContact = async (\n payload: Dto[\"CreateUnverifiedContactDto\"],\n ): Promise<void> => {\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 this.api.setAuthToken(data.token);\n // Set token in state after setting the token in the api handler\n this.state.setPartial({ contact: { token: data.token } });\n } else {\n this.state.setPartial({ isErrorCreatingUnverifiedContact: true });\n }\n } finally {\n this.state.setPartial({ isCreatingUnverifiedContact: false });\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 stopPolling: (() => void) | null = null;\n\n startPolling = (\n cb: (abortSignal: AbortSignal) => Promise<void>,\n interval: 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 } else {\n console.error(\"Failed to poll:\", error);\n this.state.setPartial({ isError: true });\n }\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, interval));\n }\n };\n\n poll();\n\n this.stopPolling = () => {\n timeouts.forEach(clearTimeout);\n this.state.reset();\n };\n };\n}\n","import { v4 as uuidv4 } from \"uuid\";\n\nexport function genUuid() {\n return uuidv4();\n}\n","import type { ApiCaller } from \"../api\";\nimport type { Dto } from \"../sdk\";\nimport type { SessionDto } from \"../types/schemas\";\nimport type { WidgetConfig } from \"../types/WidgetConfig\";\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 config: WidgetConfig;\n private api: ApiCaller;\n private contactCtx: ContactCtx;\n private poller = 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 config,\n api,\n contactCtx,\n }: { config: WidgetConfig; api: ApiCaller; contactCtx: ContactCtx }) {\n this.config = config;\n this.api = api;\n this.contactCtx = contactCtx;\n\n this.registerPolling();\n this.registerInitialSessionsFetch();\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 // The poller should automatically reset, since we're subscribed to the session state, and whenever it's null, the poller resets... but just in case, let's reset it here as well\n this.poller.reset();\n };\n\n private registerPolling = () => {\n this.sessionState.subscribe(({ session }) => {\n if (session?.id) {\n this.poller.startPolling(async (abortSignal) => {\n const { data } = await this.api.getSession({\n sessionId: session.id,\n abortSignal,\n });\n data && this.sessionState.setPartial({ session: data });\n }, 1000);\n } else {\n this.poller.reset();\n }\n });\n };\n\n private registerInitialSessionsFetch = () => {\n const initialFetch = async () => {\n this.sessionsState.setPartial({ didStartInitialFetch: true });\n // Call this for the first time to get the first page of sessions\n await this.loadMoreSessions();\n this.sessionsState.setPartial({ isInitialFetchLoading: false });\n };\n\n // If the widget config was initially provided with a contact token, no state change would be triggered, so we just fetch\n if (\n this.contactCtx.state.get().contact?.token &&\n !this.sessionsState.get().didStartInitialFetch\n ) {\n initialFetch();\n }\n\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 initialFetch();\n }\n });\n };\n\n createSession = async () => {\n this.sessionState.setPartial({ session: null, isCreatingSession: true });\n\n const { data: session, error } = await this.api.createSession({\n customData: this.config.user?.externalId\n ? {\n external_id: this.config.user?.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 if (!this.contactCtx.state.get().contact?.token) return;\n\n const { data } = await this.api.getSessions({\n cursor: this.sessionsState.get().cursor,\n filters: this.config.user?.externalId\n ? {\n external_id: this.config.user.externalId,\n }\n : {},\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","import { ApiCaller } from \"../api\";\nimport type { WidgetConfig } from \"../types/WidgetConfig\";\nimport type { SafeOmit, SomeOptional } from \"../types/helpers\";\nimport type {\n BotMessageType,\n MessageType,\n UserMessageType,\n} from \"../types/messages\";\nimport type {\n MessageAttachmentType,\n MessageDto,\n SendMessageDto,\n SendMessageOutputDto,\n} from \"../types/schemas\";\nimport { Poller } from \"../utils/Poller\";\nimport { PrimitiveState } from \"../utils/PrimitiveState\";\nimport { genUuid } from \"../utils/uuid\";\nimport { SessionCtx } from \"./session.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 sessionCtx: SessionCtx;\n private poller = new Poller();\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 }: { config: WidgetConfig; api: ApiCaller; sessionCtx: SessionCtx }) {\n this.config = config;\n this.api = api;\n this.sessionCtx = sessionCtx;\n\n this.registerPolling();\n }\n\n reset = () => {\n this.sendMessageAbortController.abort(\"Resetting chat\");\n this.state.reset();\n // The poller should automatically reset, since we're subscribed to the session state, and whenever it's null, the poller resets... but just in case, let's reset it here as well\n this.poller.reset();\n };\n\n private registerPolling = () => {\n this.sessionCtx.sessionState.subscribe(({ session }) => {\n if (session?.id) {\n this.poller.startPolling(async (abortSignal) => {\n await this.fetchAndSetHistory(session.id, abortSignal);\n }, 1000);\n } else {\n this.poller.reset();\n }\n });\n };\n\n sendMessage = async (\n input: SomeOptional<\n SafeOmit<SendMessageDto, \"bot_token\" | \"uuid\">,\n \"session_id\" | \"user\"\n >,\n ): Promise<void> => {\n this.sendMessageAbortController = new AbortController();\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 if (isSending && isAssignedToAI) {\n console.warn(\"Cannot send messages while awaiting AI response\");\n return;\n }\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 = MessageCtx.toUserMessage(\n input.content,\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 const sessionId = this.sessionCtx.sessionState.get().session?.id;\n if (!sessionId) return;\n\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 user: this.config.user?.data,\n ...input,\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 = MessageCtx.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 = MessageCtx.toErrorMessage(\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 fetchAndSetHistory = 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.state.get().messages.length === 0) {\n this.state.setPartial({ isInitialFetchLoading: true });\n }\n\n const lastMessageTimestamp = this.state.get().messages.at(-1)?.timestamp;\n\n const { data: response } = await this.api.getSessionHistory({\n sessionId,\n lastMessageTimestamp,\n abortSignal,\n });\n\n if (response && response.length > 0) {\n // Get a fresh reference to current messages after the poll is done\n const prevMessages = this.state.get().messages;\n const newMessages = response\n .map(MessageCtx.mapHistoryToMessage)\n .filter(\n (newMsg) =>\n !prevMessages.some((existingMsg) => existingMsg.id === newMsg.id),\n );\n this.state.setPartial({\n messages: [...prevMessages, ...newMessages],\n });\n }\n\n if (this.state.get().isInitialFetchLoading) {\n this.state.setPartial({ isInitialFetchLoading: false });\n }\n };\n\n /** Not the best name but whatever */\n private static 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 };\n }\n\n return {\n ...commonFields,\n type: \"FROM_BOT\",\n component: \"bot_message\",\n agent: {\n id: null,\n name: history.sender.name || \"\",\n isAi: history.sender.kind === \"ai\",\n avatar: history.sender.avatar || null,\n },\n data: {\n message: history.content.text,\n },\n };\n }\n\n private static toUserMessage(\n content: string,\n attachments?: MessageAttachmentType[],\n ): UserMessageType {\n return {\n id: genUuid(),\n type: \"FROM_USER\",\n content,\n deliveredAt: new Date().toISOString(),\n attachments,\n timestamp: new Date().toISOString(),\n };\n }\n\n private static toBotMessage(\n response: SendMessageOutputDto,\n ): BotMessageType | null {\n // TODO: sometimes the bot replies with both an autopilotResponse and a uiResponse... handle this case\n\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 data: {\n message: response.autopilotResponse.value.content,\n },\n };\n }\n\n if (response.success && response.uiResponse) {\n const uiVal = response.uiResponse.value;\n return {\n type: \"FROM_BOT\" as const,\n id: genUuid(),\n timestamp: new Date().toISOString(),\n component: uiVal.name,\n data: uiVal.request_response,\n };\n }\n\n return null;\n }\n\n private static toErrorMessage(message: string) {\n return {\n type: \"FROM_BOT\" as const,\n id: genUuid(),\n timestamp: new Date().toISOString(),\n component: \"TEXT\",\n data: {\n message,\n variant: \"error\",\n },\n };\n }\n}\n","import { 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 contactCtx: ContactCtx;\n private sessionCtx: SessionCtx;\n private resetChat: WidgetCtx[\"resetChat\"];\n\n constructor({\n contactCtx,\n sessionCtx,\n resetChat,\n }: {\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.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\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 { ApiCaller } from \"../api\";\nimport type { WidgetConfig } from \"../types/WidgetConfig\";\nimport { ContactCtx } from \"./contact.ctx\";\nimport { MessageCtx } from \"./message.ctx\";\nimport { RouterCtx } from \"./router.ctx\";\nimport { SessionCtx } from \"./session.ctx\";\n\nexport class WidgetCtx {\n public config: WidgetConfig;\n public api: ApiCaller;\n public contactCtx: ContactCtx;\n public sessionCtx: SessionCtx;\n public messageCtx: MessageCtx;\n public routerCtx: RouterCtx;\n\n constructor({ config }: { config: WidgetConfig }) {\n this.config = config;\n this.api = new ApiCaller({ config });\n\n this.contactCtx = new ContactCtx({\n api: this.api,\n config: this.config,\n });\n\n this.sessionCtx = new SessionCtx({\n config: this.config,\n api: this.api,\n contactCtx: this.contactCtx,\n });\n\n this.messageCtx = new MessageCtx({\n config: this.config,\n api: this.api,\n sessionCtx: this.sessionCtx,\n });\n\n this.routerCtx = new RouterCtx({\n contactCtx: this.contactCtx,\n sessionCtx: this.sessionCtx,\n resetChat: this.resetChat,\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","__publicField","token","baseUrl","headers","request","key","value","axios","body","abortSignal","sessionId","lastMessageTimestamp","query","cursor","filters","file","formData","data","_a","PrimitiveState","state","__privateAdd","_state","_lastUpdated","event","listeners","eventData","listener","callback","error","__privateGet","newState","isEqual","__privateSet","_s","ContactCtx","api","_b","_d","_c","payload","Poller","cb","interval","timeouts","poll","genUuid","uuidv4","SessionCtx","contactCtx","session","initialFetch","contact","deduped","s","i","self","MessageCtx","sessionCtx","input","isSending","isAssignedToAI","userMessage","currentMessages","botMessage","prevMessages","m","_e","_f","_g","_h","errorMessage","_i","response","newMessages","newMsg","existingMsg","history","commonFields","content","attachments","uiVal","message","RouterCtx","resetChat","WidgetCtx"],"mappings":"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"}
|
|
@@ -1,27 +1,27 @@
|
|
|
1
|
-
var
|
|
1
|
+
var O = Object.defineProperty;
|
|
2
2
|
var A = (a) => {
|
|
3
3
|
throw TypeError(a);
|
|
4
4
|
};
|
|
5
|
-
var
|
|
6
|
-
var s = (a, t, e) =>
|
|
7
|
-
var h = (a, t, e) => (
|
|
8
|
-
import
|
|
9
|
-
import
|
|
10
|
-
import
|
|
11
|
-
import { v4 as
|
|
12
|
-
const
|
|
5
|
+
var L = (a, t, e) => t in a ? O(a, t, { enumerable: !0, configurable: !0, writable: !0, value: e }) : a[t] = e;
|
|
6
|
+
var s = (a, t, e) => L(a, typeof t != "symbol" ? t + "" : t, e), M = (a, t, e) => t.has(a) || A("Cannot " + e);
|
|
7
|
+
var h = (a, t, e) => (M(a, t, "read from private field"), e ? e.call(a) : t.get(a)), P = (a, t, e) => t.has(a) ? A("Cannot add the same private member more than once") : t instanceof WeakSet ? t.add(a) : t.set(a, e), S = (a, t, e, i) => (M(a, t, "write to private field"), i ? i.call(a, e) : t.set(a, e), e);
|
|
8
|
+
import T from "axios";
|
|
9
|
+
import E from "openapi-fetch";
|
|
10
|
+
import D from "lodash.isequal";
|
|
11
|
+
import { v4 as _ } from "uuid";
|
|
12
|
+
const q = (a) => {
|
|
13
13
|
console.log(a.error);
|
|
14
|
-
},
|
|
15
|
-
const t =
|
|
14
|
+
}, B = (a) => {
|
|
15
|
+
const t = E({
|
|
16
16
|
baseUrl: a.baseUrl
|
|
17
17
|
}), e = {
|
|
18
18
|
onRequest: a.onRequest,
|
|
19
19
|
onResponse: a.onResponse,
|
|
20
|
-
onError: a.onError ||
|
|
20
|
+
onError: a.onError || q
|
|
21
21
|
};
|
|
22
22
|
return t.use(e), t;
|
|
23
23
|
};
|
|
24
|
-
class
|
|
24
|
+
class H {
|
|
25
25
|
constructor({
|
|
26
26
|
config: t
|
|
27
27
|
}) {
|
|
@@ -40,7 +40,7 @@ class q {
|
|
|
40
40
|
s(this, "createOpenAPIClient", ({
|
|
41
41
|
baseUrl: t,
|
|
42
42
|
headers: e
|
|
43
|
-
}) =>
|
|
43
|
+
}) => B({
|
|
44
44
|
baseUrl: t,
|
|
45
45
|
onRequest: ({ request: i }) => {
|
|
46
46
|
Object.entries(e).forEach(([n, r]) => {
|
|
@@ -51,7 +51,7 @@ class q {
|
|
|
51
51
|
s(this, "createAxiosUploadClient", ({
|
|
52
52
|
baseUrl: t,
|
|
53
53
|
headers: e
|
|
54
|
-
}) =>
|
|
54
|
+
}) => T.create({
|
|
55
55
|
baseURL: `${t}/backend/widget/v2/upload`,
|
|
56
56
|
headers: e
|
|
57
57
|
}));
|
|
@@ -181,7 +181,7 @@ class m {
|
|
|
181
181
|
this.emitLifecycle("beforeUpdate", {
|
|
182
182
|
previousState: h(this, l),
|
|
183
183
|
nextState: t
|
|
184
|
-
}),
|
|
184
|
+
}), D(h(this, l), t) || (S(this, l, t), S(this, C, Date.now()), this.emitLifecycle("stateChange", { state: t }), this.notifySubscribers(t)), this.emitLifecycle("afterUpdate", { state: t });
|
|
185
185
|
});
|
|
186
186
|
// TODO make this provide prev state
|
|
187
187
|
s(this, "setPartial", (t) => {
|
|
@@ -206,7 +206,7 @@ class m {
|
|
|
206
206
|
}
|
|
207
207
|
}
|
|
208
208
|
l = new WeakMap(), C = new WeakMap();
|
|
209
|
-
class
|
|
209
|
+
class G {
|
|
210
210
|
constructor({
|
|
211
211
|
config: t,
|
|
212
212
|
api: e
|
|
@@ -245,7 +245,7 @@ class B {
|
|
|
245
245
|
}), !((r = t.user) != null && r.token) && !t.collectUserData && this.autoCreateUnverifiedUser();
|
|
246
246
|
}
|
|
247
247
|
}
|
|
248
|
-
class
|
|
248
|
+
class k {
|
|
249
249
|
constructor() {
|
|
250
250
|
s(this, "state", new m({
|
|
251
251
|
isPolling: !1,
|
|
@@ -279,9 +279,9 @@ class M {
|
|
|
279
279
|
}
|
|
280
280
|
}
|
|
281
281
|
function y() {
|
|
282
|
-
return
|
|
282
|
+
return _();
|
|
283
283
|
}
|
|
284
|
-
class
|
|
284
|
+
class j {
|
|
285
285
|
constructor({
|
|
286
286
|
config: t,
|
|
287
287
|
api: e,
|
|
@@ -290,7 +290,7 @@ class H {
|
|
|
290
290
|
s(this, "config");
|
|
291
291
|
s(this, "api");
|
|
292
292
|
s(this, "contactCtx");
|
|
293
|
-
s(this, "poller", new
|
|
293
|
+
s(this, "poller", new k());
|
|
294
294
|
s(this, "sessionState", new m({
|
|
295
295
|
session: null,
|
|
296
296
|
isCreatingSession: !1
|
|
@@ -371,11 +371,11 @@ class p {
|
|
|
371
371
|
s(this, "config");
|
|
372
372
|
s(this, "api");
|
|
373
373
|
s(this, "sessionCtx");
|
|
374
|
-
s(this, "poller", new
|
|
374
|
+
s(this, "poller", new k());
|
|
375
375
|
s(this, "state", new m({
|
|
376
376
|
messages: [],
|
|
377
377
|
isSendingMessage: !1,
|
|
378
|
-
|
|
378
|
+
lastAIResMightSolveUserIssue: !1,
|
|
379
379
|
isInitialFetchLoading: !1
|
|
380
380
|
}));
|
|
381
381
|
s(this, "sendMessageAbortController", new AbortController());
|
|
@@ -390,35 +390,35 @@ class p {
|
|
|
390
390
|
});
|
|
391
391
|
});
|
|
392
392
|
s(this, "sendMessage", async (t) => {
|
|
393
|
-
var n, r, c, d, g, u, x;
|
|
393
|
+
var n, r, c, d, g, u, x, v, U;
|
|
394
394
|
this.sendMessageAbortController = new AbortController();
|
|
395
395
|
const e = this.state.get().isSendingMessage, i = ((n = this.sessionCtx.sessionState.get().session) == null ? void 0 : n.assignee.kind) === "ai";
|
|
396
396
|
if (e && i) {
|
|
397
397
|
console.warn("Cannot send messages while awaiting AI response");
|
|
398
398
|
return;
|
|
399
399
|
}
|
|
400
|
-
this.state.setPartial({
|
|
400
|
+
this.state.setPartial({ lastAIResMightSolveUserIssue: !1 });
|
|
401
401
|
try {
|
|
402
402
|
this.state.setPartial({ isSendingMessage: !0 });
|
|
403
403
|
const b = p.toUserMessage(
|
|
404
404
|
t.content,
|
|
405
405
|
t.attachments || void 0
|
|
406
|
-
),
|
|
406
|
+
), R = this.state.get().messages;
|
|
407
407
|
if (this.state.setPartial({
|
|
408
|
-
messages: [...
|
|
408
|
+
messages: [...R, b]
|
|
409
409
|
}), !((r = this.sessionCtx.sessionState.get().session) != null && r.id) && !await this.sessionCtx.createSession()) {
|
|
410
410
|
console.error("Failed to create session");
|
|
411
411
|
return;
|
|
412
412
|
}
|
|
413
|
-
const
|
|
414
|
-
if (!
|
|
413
|
+
const I = (c = this.sessionCtx.sessionState.get().session) == null ? void 0 : c.id;
|
|
414
|
+
if (!I) return;
|
|
415
415
|
const { data: o } = await this.api.sendMessage(
|
|
416
416
|
{
|
|
417
417
|
uuid: b.id,
|
|
418
418
|
bot_token: this.config.token,
|
|
419
419
|
headers: this.config.headers,
|
|
420
420
|
query_params: this.config.queryParams,
|
|
421
|
-
session_id:
|
|
421
|
+
session_id: I,
|
|
422
422
|
user: (d = this.config.user) == null ? void 0 : d.data,
|
|
423
423
|
...t
|
|
424
424
|
},
|
|
@@ -430,13 +430,20 @@ class p {
|
|
|
430
430
|
const w = this.state.get().messages;
|
|
431
431
|
if (!!w.some(
|
|
432
432
|
(F) => F.id === f.id
|
|
433
|
-
))
|
|
434
|
-
|
|
435
|
-
|
|
436
|
-
|
|
433
|
+
)) {
|
|
434
|
+
this.state.setPartial({
|
|
435
|
+
lastAIResMightSolveUserIssue: ((g = o.autopilotResponse) == null ? void 0 : g.mightSolveUserIssue) || ((u = o.uiResponse) == null ? void 0 : u.mightSolveUserIssue)
|
|
436
|
+
});
|
|
437
|
+
return;
|
|
438
|
+
}
|
|
439
|
+
this.state.setPartial({
|
|
440
|
+
messages: [...w, f],
|
|
441
|
+
lastAIResMightSolveUserIssue: ((x = o.autopilotResponse) == null ? void 0 : x.mightSolveUserIssue) || ((v = o.uiResponse) == null ? void 0 : v.mightSolveUserIssue)
|
|
442
|
+
});
|
|
443
|
+
}
|
|
437
444
|
} else {
|
|
438
445
|
const f = p.toErrorMessage(
|
|
439
|
-
((
|
|
446
|
+
((U = o == null ? void 0 : o.error) == null ? void 0 : U.message) || "Unknown error occurred"
|
|
440
447
|
), w = this.state.get().messages;
|
|
441
448
|
this.state.setPartial({
|
|
442
449
|
messages: [...w, f]
|
|
@@ -548,7 +555,7 @@ class p {
|
|
|
548
555
|
};
|
|
549
556
|
}
|
|
550
557
|
}
|
|
551
|
-
class
|
|
558
|
+
class X {
|
|
552
559
|
constructor({
|
|
553
560
|
contactCtx: t,
|
|
554
561
|
sessionCtx: e,
|
|
@@ -582,7 +589,7 @@ class G {
|
|
|
582
589
|
}), this.contactCtx = t, this.sessionCtx = e, this.resetChat = i, this.registerRoutingListener();
|
|
583
590
|
}
|
|
584
591
|
}
|
|
585
|
-
class
|
|
592
|
+
class K {
|
|
586
593
|
constructor({ config: t }) {
|
|
587
594
|
s(this, "config");
|
|
588
595
|
s(this, "api");
|
|
@@ -593,10 +600,10 @@ class J {
|
|
|
593
600
|
s(this, "resetChat", () => {
|
|
594
601
|
this.sessionCtx.reset(), this.messageCtx.reset();
|
|
595
602
|
});
|
|
596
|
-
this.config = t, this.api = new
|
|
603
|
+
this.config = t, this.api = new H({ config: t }), this.contactCtx = new G({
|
|
597
604
|
api: this.api,
|
|
598
605
|
config: this.config
|
|
599
|
-
}), this.sessionCtx = new
|
|
606
|
+
}), this.sessionCtx = new j({
|
|
600
607
|
config: this.config,
|
|
601
608
|
api: this.api,
|
|
602
609
|
contactCtx: this.contactCtx
|
|
@@ -604,7 +611,7 @@ class J {
|
|
|
604
611
|
config: this.config,
|
|
605
612
|
api: this.api,
|
|
606
613
|
sessionCtx: this.sessionCtx
|
|
607
|
-
}), this.routerCtx = new
|
|
614
|
+
}), this.routerCtx = new X({
|
|
608
615
|
contactCtx: this.contactCtx,
|
|
609
616
|
sessionCtx: this.sessionCtx,
|
|
610
617
|
resetChat: this.resetChat
|
|
@@ -613,6 +620,6 @@ class J {
|
|
|
613
620
|
}
|
|
614
621
|
export {
|
|
615
622
|
m as P,
|
|
616
|
-
|
|
623
|
+
K as W
|
|
617
624
|
};
|
|
618
|
-
//# sourceMappingURL=widget.ctx-
|
|
625
|
+
//# sourceMappingURL=widget.ctx-m7KQtnuH.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"widget.ctx-m7KQtnuH.js","sources":["../src/headless/core/sdk/index.ts","../src/headless/core/api.ts","../src/headless/core/utils/PrimitiveState.ts","../src/headless/core/context/contact.ctx.ts","../src/headless/core/utils/Poller.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/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 axios, { type AxiosInstance, type AxiosRequestConfig } from \"axios\";\nimport { type Dto, type Endpoint, basicClient } from \"./sdk\";\nimport type { WidgetConfig } from \"./types/WidgetConfig\";\nimport type { SendMessageDto, VoteInputDto } from \"./types/schemas\";\n\nexport class ApiCaller {\n private client: ReturnType<typeof basicClient>;\n private uploadFileClient: AxiosInstance;\n private config: WidgetConfig;\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 this.uploadFileClient = this.createAxiosUploadClient({ baseUrl, headers });\n }\n\n private constructClientOptions = (token: string | null | undefined) => {\n const baseUrl = 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 private createAxiosUploadClient = ({\n baseUrl,\n headers,\n }: ReturnType<typeof this.constructClientOptions>) => {\n const uploadPath = \"/backend/widget/v2/upload\" satisfies Endpoint;\n return axios.create({\n baseURL: `${baseUrl}${uploadPath}`,\n headers,\n });\n };\n\n setAuthToken = (token: string) => {\n const { baseUrl, headers } = this.constructClientOptions(token);\n this.client = this.createOpenAPIClient({ baseUrl, headers });\n this.uploadFileClient = this.createAxiosUploadClient({ baseUrl, headers });\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 getSessionHistory = 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(\n \"/backend/widget/v2/session/history/{sessionId}\",\n { params: { path: { sessionId }, query }, 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[\"CreateWidgetChatSessionDto\"]) => {\n return await this.client.POST(\"/backend/widget/v2/create-session\", {\n body,\n });\n };\n\n getSession = async ({\n sessionId,\n abortSignal,\n }: { sessionId: string; abortSignal: AbortSignal }) => {\n return await this.client.GET(\"/backend/widget/v2/session/{sessionId}\", {\n params: { path: { sessionId } },\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 uploadFile = async (\n file: {\n id: string;\n file: File;\n },\n config: Partial<AxiosRequestConfig> = {},\n ) => {\n const formData = new FormData();\n formData.append(\"file\", file.file);\n\n // Couldn't get this to work with the openapi client... dunno why...\n const { data } = await this.uploadFileClient.post<\n Dto[\"UploadWidgetFileResponseDto\"]\n >(\"\", formData, {\n headers: {\n \"Content-Type\": \"multipart/form-data\",\n },\n ...config,\n });\n return data;\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 enum LifecycleEvent {\n INIT = \"init\",\n STATE_CHANGE = \"stateChange\",\n BEFORE_UPDATE = \"beforeUpdate\",\n AFTER_UPDATE = \"afterUpdate\",\n DESTROY = \"destroy\",\n ERROR = \"error\",\n}\n\ntype LifecycleListener = (event: {\n type: LifecycleEvent;\n timestamp: number;\n data?: any;\n}) => void;\n\nexport class PrimitiveState<S> {\n private subscribers = new Set<Subscriber<S>>();\n #state: S;\n private initialState: S;\n #lastUpdated: number | null;\n private lifecycleListeners: Map<LifecycleEvent, Set<LifecycleListener>> =\n new Map();\n\n constructor(state: S) {\n this.#state = state;\n this.initialState = state;\n this.#lastUpdated = Date.now();\n this.emitLifecycle(LifecycleEvent.INIT, { initialState: this.#state });\n }\n\n private emitLifecycle = (event: LifecycleEvent, data?: any) => {\n const listeners = this.lifecycleListeners.get(event);\n if (listeners) {\n const eventData = {\n type: event,\n timestamp: Date.now(),\n data,\n };\n listeners.forEach((listener) => {\n try {\n listener(eventData);\n } catch {\n // ignore error\n }\n });\n }\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 this.emitLifecycle(LifecycleEvent.ERROR, { error });\n }\n });\n };\n\n /**\n * Subscribe to state changes\n * @param callback Function to call when state changes\n * @returns Unsubscribe function\n */\n subscribe = (callback: Subscriber<S>): (() => void) => {\n this.subscribers.add(callback);\n // Don't call the callback immediately with current state\n return () => {\n this.subscribers.delete(callback);\n };\n };\n\n onLifecycle = (\n event: LifecycleEvent,\n listener: LifecycleListener,\n ): (() => void) => {\n if (!this.lifecycleListeners.has(event)) {\n this.lifecycleListeners.set(event, new Set());\n }\n const listeners = this.lifecycleListeners.get(event)!;\n listeners.add(listener);\n\n return () => {\n listeners.delete(listener);\n if (listeners.size === 0) {\n this.lifecycleListeners.delete(event);\n }\n };\n };\n\n /** Get the current state */\n get = (): S => {\n return this.#state;\n };\n\n // TODO make this provide prev state\n /**\n * Set the state and notify subscribers if the state changes\n * @param newState The new state to set\n */\n set = (newState: S): void => {\n this.emitLifecycle(LifecycleEvent.BEFORE_UPDATE, {\n previousState: this.#state,\n nextState: newState,\n });\n\n if (!isEqual(this.#state, newState)) {\n this.#state = newState;\n this.#lastUpdated = Date.now();\n this.emitLifecycle(LifecycleEvent.STATE_CHANGE, { state: newState });\n this.notifySubscribers(newState);\n }\n\n this.emitLifecycle(LifecycleEvent.AFTER_UPDATE, { state: newState });\n };\n\n // TODO make this provide prev state\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 /**\n * Clear all subscriptions\n */\n clear = (): void => {\n this.emitLifecycle(LifecycleEvent.DESTROY);\n this.subscribers = new Set(); // Create a new Set instead of just clearing\n this.lifecycleListeners = new Map();\n };\n\n reset = (): void => {\n this.set(this.initialState);\n };\n\n lastUpdated = (): number | null => {\n return this.#lastUpdated;\n };\n}\n","import { PrimitiveState } from \"../utils/PrimitiveState\";\nimport { ApiCaller } from \"../api\";\nimport { type WidgetConfig } from \"../types/WidgetConfig\";\nimport { type Dto } from \"../sdk\";\n\ntype ContactState = {\n contact: { token: string } | null;\n isCreatingUnverifiedContact: boolean;\n isErrorCreatingUnverifiedContact: boolean;\n};\n\nexport class ContactCtx {\n private config: WidgetConfig;\n private api: ApiCaller;\n state: PrimitiveState<ContactState>;\n\n constructor({\n config,\n api,\n }: {\n api: ApiCaller;\n config: WidgetConfig;\n }) {\n this.config = config;\n this.api = api;\n\n this.state = new PrimitiveState<ContactState>({\n contact: config.user?.token ? { token: config.user?.token } : null,\n isCreatingUnverifiedContact: false,\n isErrorCreatingUnverifiedContact: false,\n });\n\n if (!config.user?.token && !config.collectUserData) {\n this.autoCreateUnverifiedUser();\n }\n }\n\n shouldCollectData = (): boolean => {\n const currentState = this.state.get();\n\n if (!currentState.contact?.token && this.config.collectUserData) {\n return true;\n } else {\n return false;\n }\n };\n\n private autoCreateUnverifiedUser = async () => {\n await this.createUnverifiedContact({\n name: this.config.user?.data?.name || \"Anonymous\",\n email: this.config.user?.data?.email,\n });\n };\n\n createUnverifiedContact = async (\n payload: Dto[\"CreateUnverifiedContactDto\"],\n ): Promise<void> => {\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 this.api.setAuthToken(data.token);\n // Set token in state after setting the token in the api handler\n this.state.setPartial({ contact: { token: data.token } });\n } else {\n this.state.setPartial({ isErrorCreatingUnverifiedContact: true });\n }\n } finally {\n this.state.setPartial({ isCreatingUnverifiedContact: false });\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 stopPolling: (() => void) | null = null;\n\n startPolling = (\n cb: (abortSignal: AbortSignal) => Promise<void>,\n interval: 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 } else {\n console.error(\"Failed to poll:\", error);\n this.state.setPartial({ isError: true });\n }\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, interval));\n }\n };\n\n poll();\n\n this.stopPolling = () => {\n timeouts.forEach(clearTimeout);\n this.state.reset();\n };\n };\n}\n","import { v4 as uuidv4 } from \"uuid\";\n\nexport function genUuid() {\n return uuidv4();\n}\n","import type { ApiCaller } from \"../api\";\nimport type { Dto } from \"../sdk\";\nimport type { SessionDto } from \"../types/schemas\";\nimport type { WidgetConfig } from \"../types/WidgetConfig\";\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 config: WidgetConfig;\n private api: ApiCaller;\n private contactCtx: ContactCtx;\n private poller = 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 config,\n api,\n contactCtx,\n }: { config: WidgetConfig; api: ApiCaller; contactCtx: ContactCtx }) {\n this.config = config;\n this.api = api;\n this.contactCtx = contactCtx;\n\n this.registerPolling();\n this.registerInitialSessionsFetch();\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 // The poller should automatically reset, since we're subscribed to the session state, and whenever it's null, the poller resets... but just in case, let's reset it here as well\n this.poller.reset();\n };\n\n private registerPolling = () => {\n this.sessionState.subscribe(({ session }) => {\n if (session?.id) {\n this.poller.startPolling(async (abortSignal) => {\n const { data } = await this.api.getSession({\n sessionId: session.id,\n abortSignal,\n });\n data && this.sessionState.setPartial({ session: data });\n }, 1000);\n } else {\n this.poller.reset();\n }\n });\n };\n\n private registerInitialSessionsFetch = () => {\n const initialFetch = async () => {\n this.sessionsState.setPartial({ didStartInitialFetch: true });\n // Call this for the first time to get the first page of sessions\n await this.loadMoreSessions();\n this.sessionsState.setPartial({ isInitialFetchLoading: false });\n };\n\n // If the widget config was initially provided with a contact token, no state change would be triggered, so we just fetch\n if (\n this.contactCtx.state.get().contact?.token &&\n !this.sessionsState.get().didStartInitialFetch\n ) {\n initialFetch();\n }\n\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 initialFetch();\n }\n });\n };\n\n createSession = async () => {\n this.sessionState.setPartial({ session: null, isCreatingSession: true });\n\n const { data: session, error } = await this.api.createSession({\n customData: this.config.user?.externalId\n ? {\n external_id: this.config.user?.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 if (!this.contactCtx.state.get().contact?.token) return;\n\n const { data } = await this.api.getSessions({\n cursor: this.sessionsState.get().cursor,\n filters: this.config.user?.externalId\n ? {\n external_id: this.config.user.externalId,\n }\n : {},\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","import { ApiCaller } from \"../api\";\nimport type { WidgetConfig } from \"../types/WidgetConfig\";\nimport type { SafeOmit, SomeOptional } from \"../types/helpers\";\nimport type {\n BotMessageType,\n MessageType,\n UserMessageType,\n} from \"../types/messages\";\nimport type {\n MessageAttachmentType,\n MessageDto,\n SendMessageDto,\n SendMessageOutputDto,\n} from \"../types/schemas\";\nimport { Poller } from \"../utils/Poller\";\nimport { PrimitiveState } from \"../utils/PrimitiveState\";\nimport { genUuid } from \"../utils/uuid\";\nimport { SessionCtx } from \"./session.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 sessionCtx: SessionCtx;\n private poller = new Poller();\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 }: { config: WidgetConfig; api: ApiCaller; sessionCtx: SessionCtx }) {\n this.config = config;\n this.api = api;\n this.sessionCtx = sessionCtx;\n\n this.registerPolling();\n }\n\n reset = () => {\n this.sendMessageAbortController.abort(\"Resetting chat\");\n this.state.reset();\n // The poller should automatically reset, since we're subscribed to the session state, and whenever it's null, the poller resets... but just in case, let's reset it here as well\n this.poller.reset();\n };\n\n private registerPolling = () => {\n this.sessionCtx.sessionState.subscribe(({ session }) => {\n if (session?.id) {\n this.poller.startPolling(async (abortSignal) => {\n await this.fetchAndSetHistory(session.id, abortSignal);\n }, 1000);\n } else {\n this.poller.reset();\n }\n });\n };\n\n sendMessage = async (\n input: SomeOptional<\n SafeOmit<SendMessageDto, \"bot_token\" | \"uuid\">,\n \"session_id\" | \"user\"\n >,\n ): Promise<void> => {\n this.sendMessageAbortController = new AbortController();\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 if (isSending && isAssignedToAI) {\n console.warn(\"Cannot send messages while awaiting AI response\");\n return;\n }\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 = MessageCtx.toUserMessage(\n input.content,\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 const sessionId = this.sessionCtx.sessionState.get().session?.id;\n if (!sessionId) return;\n\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 user: this.config.user?.data,\n ...input,\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 = MessageCtx.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 = MessageCtx.toErrorMessage(\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 fetchAndSetHistory = 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.state.get().messages.length === 0) {\n this.state.setPartial({ isInitialFetchLoading: true });\n }\n\n const lastMessageTimestamp = this.state.get().messages.at(-1)?.timestamp;\n\n const { data: response } = await this.api.getSessionHistory({\n sessionId,\n lastMessageTimestamp,\n abortSignal,\n });\n\n if (response && response.length > 0) {\n // Get a fresh reference to current messages after the poll is done\n const prevMessages = this.state.get().messages;\n const newMessages = response\n .map(MessageCtx.mapHistoryToMessage)\n .filter(\n (newMsg) =>\n !prevMessages.some((existingMsg) => existingMsg.id === newMsg.id),\n );\n this.state.setPartial({\n messages: [...prevMessages, ...newMessages],\n });\n }\n\n if (this.state.get().isInitialFetchLoading) {\n this.state.setPartial({ isInitialFetchLoading: false });\n }\n };\n\n /** Not the best name but whatever */\n private static 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 };\n }\n\n return {\n ...commonFields,\n type: \"FROM_BOT\",\n component: \"bot_message\",\n agent: {\n id: null,\n name: history.sender.name || \"\",\n isAi: history.sender.kind === \"ai\",\n avatar: history.sender.avatar || null,\n },\n data: {\n message: history.content.text,\n },\n };\n }\n\n private static toUserMessage(\n content: string,\n attachments?: MessageAttachmentType[],\n ): UserMessageType {\n return {\n id: genUuid(),\n type: \"FROM_USER\",\n content,\n deliveredAt: new Date().toISOString(),\n attachments,\n timestamp: new Date().toISOString(),\n };\n }\n\n private static toBotMessage(\n response: SendMessageOutputDto,\n ): BotMessageType | null {\n // TODO: sometimes the bot replies with both an autopilotResponse and a uiResponse... handle this case\n\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 data: {\n message: response.autopilotResponse.value.content,\n },\n };\n }\n\n if (response.success && response.uiResponse) {\n const uiVal = response.uiResponse.value;\n return {\n type: \"FROM_BOT\" as const,\n id: genUuid(),\n timestamp: new Date().toISOString(),\n component: uiVal.name,\n data: uiVal.request_response,\n };\n }\n\n return null;\n }\n\n private static toErrorMessage(message: string) {\n return {\n type: \"FROM_BOT\" as const,\n id: genUuid(),\n timestamp: new Date().toISOString(),\n component: \"TEXT\",\n data: {\n message,\n variant: \"error\",\n },\n };\n }\n}\n","import { 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 contactCtx: ContactCtx;\n private sessionCtx: SessionCtx;\n private resetChat: WidgetCtx[\"resetChat\"];\n\n constructor({\n contactCtx,\n sessionCtx,\n resetChat,\n }: {\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.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\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 { ApiCaller } from \"../api\";\nimport type { WidgetConfig } from \"../types/WidgetConfig\";\nimport { ContactCtx } from \"./contact.ctx\";\nimport { MessageCtx } from \"./message.ctx\";\nimport { RouterCtx } from \"./router.ctx\";\nimport { SessionCtx } from \"./session.ctx\";\n\nexport class WidgetCtx {\n public config: WidgetConfig;\n public api: ApiCaller;\n public contactCtx: ContactCtx;\n public sessionCtx: SessionCtx;\n public messageCtx: MessageCtx;\n public routerCtx: RouterCtx;\n\n constructor({ config }: { config: WidgetConfig }) {\n this.config = config;\n this.api = new ApiCaller({ config });\n\n this.contactCtx = new ContactCtx({\n api: this.api,\n config: this.config,\n });\n\n this.sessionCtx = new SessionCtx({\n config: this.config,\n api: this.api,\n contactCtx: this.contactCtx,\n });\n\n this.messageCtx = new MessageCtx({\n config: this.config,\n api: this.api,\n sessionCtx: this.sessionCtx,\n });\n\n this.routerCtx = new RouterCtx({\n contactCtx: this.contactCtx,\n sessionCtx: this.sessionCtx,\n resetChat: this.resetChat,\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","__publicField","token","baseUrl","headers","request","key","value","axios","body","abortSignal","sessionId","lastMessageTimestamp","query","cursor","filters","file","formData","data","_a","PrimitiveState","state","__privateAdd","_state","_lastUpdated","event","listeners","eventData","listener","callback","error","__privateGet","newState","isEqual","__privateSet","_s","ContactCtx","api","_b","_d","_c","payload","Poller","cb","interval","timeouts","poll","genUuid","uuidv4","SessionCtx","contactCtx","session","initialFetch","contact","deduped","s","i","self","MessageCtx","sessionCtx","input","isSending","isAssignedToAI","userMessage","currentMessages","botMessage","prevMessages","m","_e","_f","_g","_h","errorMessage","_i","response","newMessages","newMsg","existingMsg","history","commonFields","content","attachments","uiVal","message","RouterCtx","resetChat","WidgetCtx"],"mappings":"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"}
|