@rubytech/create-realagent 1.0.866 → 1.0.868

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (35) hide show
  1. package/package.json +1 -1
  2. package/payload/platform/lib/graph-search/dist/index.d.ts +51 -0
  3. package/payload/platform/lib/graph-search/dist/index.d.ts.map +1 -1
  4. package/payload/platform/lib/graph-search/dist/index.js +77 -7
  5. package/payload/platform/lib/graph-search/dist/index.js.map +1 -1
  6. package/payload/platform/lib/graph-search/src/__tests__/bm25-strong-bypass-threshold.test.ts +126 -0
  7. package/payload/platform/lib/graph-search/src/__tests__/vector-threshold.test.ts +170 -0
  8. package/payload/platform/lib/graph-search/src/index.ts +129 -9
  9. package/payload/platform/plugins/admin/skills/unzip-attachment/SKILL.md +20 -7
  10. package/payload/platform/plugins/admin/skills/unzip-attachment/__tests__/preflight.sh +148 -0
  11. package/payload/platform/plugins/admin/skills/unzip-attachment/references/safety.md +53 -18
  12. package/payload/platform/templates/agents/admin/IDENTITY.md +1 -1
  13. package/payload/platform/templates/specialists/agents/database-operator.md +1 -1
  14. package/payload/server/chunk-DHSBEMWW.js +11319 -0
  15. package/payload/server/maxy-edge.js +1 -1
  16. package/payload/server/public/assets/{Checkbox-BySsatDO.js → Checkbox-B9hff9s8.js} +1 -1
  17. package/payload/server/public/assets/{admin-CLp1xGlJ.js → admin-Cpi6L_g7.js} +3 -3
  18. package/payload/server/public/assets/data-Da6iYRW1.js +1 -0
  19. package/payload/server/public/assets/graph-BHq-JYwV.js +1 -0
  20. package/payload/server/public/assets/{useAdminFetch-B3MO55eB.js → graph-labels-ChinGFwI.js} +1 -1
  21. package/payload/server/public/assets/{jsx-runtime-O5ef8xK8.css → jsx-runtime-CVA1ZrPS.css} +1 -1
  22. package/payload/server/public/assets/page-DqPf65sS.js +50 -0
  23. package/payload/server/public/assets/page-OVrxtgOZ.js +1 -0
  24. package/payload/server/public/assets/{public-DRrf63wm.js → public-CJN5KAiK.js} +1 -1
  25. package/payload/server/public/assets/{useVoiceRecorder-CR8gcELb.js → useVoiceRecorder-DyVx7e7a.js} +1 -1
  26. package/payload/server/public/data.html +5 -5
  27. package/payload/server/public/graph.html +6 -6
  28. package/payload/server/public/index.html +8 -8
  29. package/payload/server/public/public.html +5 -5
  30. package/payload/server/server.js +80 -11
  31. package/payload/server/public/assets/data-BuuqlV4L.js +0 -1
  32. package/payload/server/public/assets/graph-CtVITeok.js +0 -1
  33. package/payload/server/public/assets/page-Ddc_nKh8.js +0 -1
  34. package/payload/server/public/assets/page-IQBQoOdT.js +0 -50
  35. /package/payload/server/public/assets/{jsx-runtime-DnY0498s.js → jsx-runtime-nxP_2eNo.js} +0 -0
@@ -0,0 +1 @@
1
+ import{o as e}from"./chunk-DD-I1_y5.js";import{n as t,o as n,r,t as i}from"./jsx-runtime-nxP_2eNo.js";import{a,c as o,d as s,f as c,l,p as u,s as d,t as f,u as p}from"./graph-labels-ChinGFwI.js";var m=r(`circle-arrow-up`,[[`circle`,{cx:`12`,cy:`12`,r:`10`,key:`1mglay`}],[`path`,{d:`m16 12-4-4-4 4`,key:`177agl`}],[`path`,{d:`M12 16V8`,key:`1sbj14`}]]),h=r(`download`,[[`path`,{d:`M12 15V3`,key:`m9g1x1`}],[`path`,{d:`M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4`,key:`ih7n3h`}],[`path`,{d:`m7 10 5 5 5-5`,key:`brsn70`}]]),g=r(`file`,[[`path`,{d:`M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z`,key:`1oefj6`}],[`path`,{d:`M14 2v5a1 1 0 0 0 1 1h5`,key:`wfsgrz`}]]),_=r(`folder`,[[`path`,{d:`M20 20a2 2 0 0 0 2-2V8a2 2 0 0 0-2-2h-7.9a2 2 0 0 1-1.69-.9L9.6 3.9A2 2 0 0 0 7.93 3H4a2 2 0 0 0-2 2v13a2 2 0 0 0 2 2Z`,key:`1kt360`}]]),v=r(`upload`,[[`path`,{d:`M12 3v12`,key:`1x0j5s`}],[`path`,{d:`m17 8-5-5-5 5`,key:`7q97r8`}],[`path`,{d:`M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4`,key:`ih7n3h`}]]);function y(e){let t=new Set,n=[];for(let r of e)for(let e of r.labels)e&&(t.has(e)||f.has(e)&&(t.add(e),n.push(e)));return n}var b=80;function x(e){let t=S(e.properties)??e.labels[0]??``,n=t.length>b?t.slice(0,b):t;return`/graph?${new URLSearchParams({nodeId:e.nodeId,label:n}).toString()}`}function S(e){for(let t of[`title`,`name`,`summary`,`headline`]){let n=e[t];if(typeof n==`string`&&n.length>0)return n}return null}var C=e(n(),1),w=i();function T(){let[e,t]=(0,C.useState)(null),[n,r]=(0,C.useState)(!1);return(0,C.useEffect)(()=>{let e=!1,n=null;try{n=sessionStorage.getItem(`maxy-admin-session-key`)}catch{}if(!n){t(null),r(!0);return}return fetch(`/api/admin/session?session_key=${encodeURIComponent(n)}`).then(i=>{if(!e){if(i.status===401){try{sessionStorage.removeItem(`maxy-admin-session-key`)}catch{}window.location.href=`/`;return}t(n),r(!0)}}).catch(()=>{e||(t(n),r(!0))}),()=>{e=!0}},[]),n?e?(0,w.jsxs)(`div`,{className:`data-page`,children:[(0,w.jsxs)(`header`,{className:`data-header`,children:[(0,w.jsxs)(`h1`,{className:`data-title`,children:[(0,w.jsx)(u,{size:18}),` Data`]}),(0,w.jsx)(D,{})]}),(0,w.jsx)(E,{sessionKey:e})]}):(0,w.jsxs)(`div`,{className:`data-page`,children:[(0,w.jsx)(`header`,{className:`data-header`,children:(0,w.jsxs)(`h1`,{className:`data-title`,children:[(0,w.jsx)(u,{size:18}),` Data`]})}),(0,w.jsxs)(`div`,{className:`data-empty`,children:[(0,w.jsx)(`p`,{children:`You are not signed in.`}),(0,w.jsxs)(`p`,{children:[`Open the `,(0,w.jsx)(`a`,{href:`/`,className:`data-link`,children:`main admin page`}),` and log in, then return here.`]})]})]}):(0,w.jsx)(`div`,{className:`data-page`,children:(0,w.jsxs)(`div`,{className:`data-loading`,children:[(0,w.jsx)(s,{size:18,className:`spin`}),` Loading…`]})})}function E({sessionKey:e,onOpenInGraph:t}){let{adminFetch:n,sessionKey:r,sessionRefetchNonce:i,banner:o,reloadToLogin:s}=a({initialSessionKey:e,surface:`data`});return(0,w.jsxs)(`main`,{className:`data-main`,children:[o&&(0,w.jsx)(`button`,{type:`button`,className:`data-session-banner`,onClick:s,"data-reason":o.reason,children:o.message}),(0,w.jsx)(O,{adminFetch:n,sessionRefetchNonce:i,onOpenInGraph:t}),(0,w.jsx)(N,{adminFetch:n,sessionKey:r,sessionRefetchNonce:i})]})}function D(){let[e,t]=(0,C.useState)(!1),n=(0,C.useRef)(null);(0,C.useEffect)(()=>{if(!e)return;let r=e=>{n.current&&(n.current.contains(e.target)||t(!1))},i=e=>{e.key===`Escape`&&t(!1)};return document.addEventListener(`mousedown`,r),document.addEventListener(`keydown`,i),()=>{document.removeEventListener(`mousedown`,r),document.removeEventListener(`keydown`,i)}},[e]);let r=(0,C.useCallback)(()=>{try{sessionStorage.removeItem(`maxy-admin-session-key`)}catch{}window.location.href=`/`},[]);return(0,w.jsxs)(`div`,{className:`chat-burger-wrap`,ref:n,children:[(0,w.jsx)(`button`,{type:`button`,className:`chat-burger`,onClick:()=>t(e=>!e),"aria-label":`Menu`,"aria-haspopup":`true`,"aria-expanded":e,children:(0,w.jsx)(l,{size:20})}),e&&(0,w.jsxs)(`div`,{className:`chat-menu`,children:[(0,w.jsxs)(`button`,{type:`button`,className:`chat-menu-item`,onClick:()=>{t(!1),window.location.href=`/`},children:[(0,w.jsx)(c,{size:14}),` Chat`]}),(0,w.jsxs)(`button`,{type:`button`,className:`chat-menu-item`,onClick:()=>{t(!1),window.location.href=`/graph`},children:[(0,w.jsx)(d,{size:14}),` Graph`]}),(0,w.jsx)(`div`,{className:`chat-menu-divider`}),(0,w.jsxs)(`button`,{type:`button`,className:`chat-menu-item`,onClick:()=>{t(!1),r()},children:[(0,w.jsx)(p,{size:14}),` Log out`]})]})]})}function O({adminFetch:e,sessionRefetchNonce:t,onOpenInGraph:n}){let[r,i]=(0,C.useState)(``),[a,c]=(0,C.useState)(``),[l,u]=(0,C.useState)(null),[d,f]=(0,C.useState)(`hybrid`),[p,m]=(0,C.useState)(!1),[h,g]=(0,C.useState)(null),[_,v]=(0,C.useState)(null),[b,S]=(0,C.useState)(0),[T,E]=(0,C.useState)(!1);(0,C.useEffect)(()=>{let e=r.trim();if(!e){c(``),u(null);return}E(!1);let t=setTimeout(()=>c(e),300);return()=>clearTimeout(t)},[r]),(0,C.useEffect)(()=>{if(!a)return;let t=!1;return m(!0),g(null),e(`/api/admin/graph-search?q=${encodeURIComponent(a)}&labels=*&limit=20${T?`&threshold=0`:``}`).then(async e=>{let t=await e.json().catch(()=>({error:`HTTP ${e.status}`}));if(!e.ok)throw Error(t.error??`HTTP ${e.status}`);return t}).then(e=>{t||(u(e.results),f(e.mode??`hybrid`),S(typeof e.suppressed==`number`?e.suppressed:0))}).catch(e=>{t||g(e instanceof Error?e.message:String(e))}).finally(()=>{t||m(!1)}),()=>{t=!0}},[a,e,t,T]);let D=(0,C.useMemo)(()=>l?y(l):[],[l]),O=_&&D.includes(_)?_:null,j=(0,C.useMemo)(()=>l?O?l.filter(e=>e.labels.includes(O)):l:null,[l,O]),M=(0,C.useCallback)(e=>{if(n){n(e);return}window.location.href=x(e)},[n]);return(0,w.jsxs)(`section`,{className:`data-panel`,children:[(0,w.jsx)(`h2`,{className:`data-panel-title`,children:`Search`}),(0,w.jsx)(`p`,{className:`data-panel-subtitle`,children:`Find anything in your knowledge base by keyword.`}),(0,w.jsxs)(`div`,{className:`data-search-input`,children:[(0,w.jsx)(o,{size:14}),(0,w.jsx)(`input`,{type:`text`,placeholder:`Search your knowledge…`,value:r,onChange:e=>i(e.target.value),autoFocus:!0,autoComplete:`off`,spellCheck:!1}),p&&(0,w.jsx)(s,{size:14,className:`spin`})]}),h&&(0,w.jsxs)(`div`,{className:`data-error`,children:[`Search failed: `,h]}),D.length>0&&(0,w.jsx)(`div`,{className:`data-chip-row`,role:`group`,"aria-label":`Filter by type`,children:D.map(e=>{let t=e===O;return(0,w.jsx)(`button`,{type:`button`,className:`data-chip`,"data-active":t,onClick:()=>v(t?null:e),children:A([e])??e},e)})}),l&&!p&&(0,w.jsxs)(`div`,{className:`data-results-meta`,children:[j.length,` of `,l.length,` result`,l.length===1?``:`s`]}),l&&!p&&!O&&b>0&&!T&&(0,w.jsxs)(`button`,{type:`button`,className:`data-suppressed-banner`,onClick:()=>E(!0),children:[b,` low-confidence result`,b===1?``:`s`,` hidden — show all`]}),l&&!p&&!O&&T&&(0,w.jsx)(`button`,{type:`button`,className:`data-suppressed-banner`,onClick:()=>E(!1),children:`Showing all results — hide low-confidence`}),l&&l.length===0&&!p&&(0,w.jsx)(`div`,{className:`data-empty-results`,children:`No matches. Try a different keyword.`}),j&&j.length>0&&(0,w.jsx)(`ul`,{className:`data-results`,children:j.map(e=>(0,w.jsx)(k,{hit:e,mode:d,onOpenInGraph:M},e.nodeId))})]})}function k({hit:e,mode:t,onOpenInGraph:n}){let r=j(e.properties),i=M(e.properties),a=A(e.labels),[o,s]=(0,C.useState)(!1),c=i&&i.length>280,l=o||!c?i:i?.slice(0,280)+`…`,u=t===`bm25`?`bm25 ${e.bm25Score.toFixed(2)}`:`vector ${e.vectorScore.toFixed(2)} · bm25 ${e.bm25Score.toFixed(2)} · combined ${e.score.toFixed(2)}`;return(0,w.jsxs)(`li`,{className:`data-result`,children:[(0,w.jsxs)(`div`,{role:`button`,tabIndex:0,className:`data-result-open`,onClick:()=>n(e),onKeyDown:t=>{(t.key===`Enter`||t.key===` `)&&(t.preventDefault(),n(e))},"aria-label":`Open ${r??a??`node`} in graph`,children:[a&&(0,w.jsx)(`div`,{className:`data-result-header`,children:(0,w.jsx)(`span`,{className:`data-result-labels`,children:a})}),r&&(0,w.jsx)(`div`,{className:`data-result-title`,children:r}),i&&(0,w.jsx)(`pre`,{className:`data-result-body`,children:l}),(0,w.jsx)(`div`,{className:`data-result-scores`,children:u})]}),c&&(0,w.jsx)(`button`,{type:`button`,className:`data-result-toggle`,onClick:()=>s(e=>!e),children:o?`Show less`:`Show more`})]})}function A(e){if(e.length===0)return null;let t=e[0].replace(/([a-z])([A-Z])/g,`$1 $2`);return t.charAt(0).toUpperCase()+t.slice(1).toLowerCase()}function j(e){for(let t of[`title`,`name`,`summary`,`headline`]){let n=e[t];if(typeof n==`string`&&n.length>0)return n.length>140?n.slice(0,140)+`…`:n}return null}function M(e){for(let t of[`content`,`summary`,`body`,`description`,`text`]){let n=e[t];if(typeof n==`string`&&n.length>0)return n}return null}function N({adminFetch:e,sessionKey:n,sessionRefetchNonce:r}){let[i,a]=(0,C.useState)(`.`),[o,c]=(0,C.useState)(null),[l,u]=(0,C.useState)([]),[d,f]=(0,C.useState)(!1),[p,y]=(0,C.useState)(null),[b,x]=(0,C.useState)(!1),[S,T]=(0,C.useState)(null),[E,D]=(0,C.useState)(null),[O,k]=(0,C.useState)(null),[A,j]=(0,C.useState)(null),M=(0,C.useRef)(null),[N,I]=(0,C.useState)(0);(0,C.useEffect)(()=>{let t=!1;return f(!0),y(null),e(`/api/admin/files?path=${encodeURIComponent(i===`.`?``:i)}`).then(async e=>{let t=await e.json().catch(()=>({error:`HTTP ${e.status}`}));if(!e.ok)throw Error(t.error??`HTTP ${e.status}`);return t}).then(e=>{t||(c(e.entries),u(e.displayPath??[]))}).catch(e=>{t||(c([]),u([]),y(e instanceof Error?e.message:String(e)))}).finally(()=>{t||f(!1)}),()=>{t=!0}},[e,i,N,r]);let L=(0,C.useCallback)(e=>{a(e)},[]),R=(0,C.useCallback)(()=>{if(i===`.`||i===``)return;let e=i.split(`/`).filter(Boolean);e.pop(),a(e.length===0?`.`:e.join(`/`))},[i]),z=(0,C.useCallback)(e=>{let t=i===`.`?e.name:`${i}/${e.name}`,r=`/api/admin/files/download?session_key=${encodeURIComponent(n)}&path=${encodeURIComponent(t)}`,a=document.createElement(`a`);a.href=r,a.rel=`noopener noreferrer`,document.body.appendChild(a),a.click(),a.remove()},[i,n]),B=(0,C.useCallback)(()=>{M.current?.click()},[]),V=(0,C.useCallback)(e=>{D(null),j(e.name)},[]),H=(0,C.useCallback)(()=>{j(null)},[]),U=(0,C.useCallback)(async t=>{let n=i===`.`?t.name:`${i}/${t.name}`;D(null),j(null),k(t.name);try{let t=await e(`/api/admin/files?path=${encodeURIComponent(n)}`,{method:`DELETE`}),r=await t.json().catch(()=>({error:`HTTP ${t.status}`}));if(!t.ok)throw Error(r.error??`HTTP ${t.status}`);I(e=>e+1)}catch(e){D(e instanceof Error?e.message:String(e))}finally{k(null)}},[i,e]);(0,C.useEffect)(()=>{if(!A)return;let e=e=>{e.key===`Escape`&&j(null)};return document.addEventListener(`keydown`,e),()=>document.removeEventListener(`keydown`,e)},[A]);let W=(0,C.useCallback)(async e=>{T(null),x(!0);try{let t=new FormData;t.append(`session_key`,n),t.append(`file`,e);let r=await fetch(`/api/admin/files/upload`,{method:`POST`,body:t}),i=await r.json().catch(()=>({error:`HTTP ${r.status}`}));if(!r.ok)throw Error(i.error??`HTTP ${r.status}`);a(i.path.split(`/`).slice(0,-1).join(`/`)||`.`),I(e=>e+1)}catch(e){T(e instanceof Error?e.message:String(e))}finally{x(!1),M.current&&(M.current.value=``)}},[n]);return(0,w.jsxs)(`section`,{className:`data-panel`,children:[(0,w.jsxs)(`div`,{className:`data-panel-heading`,children:[(0,w.jsx)(`h2`,{className:`data-panel-title`,children:`Files`}),(0,w.jsxs)(`div`,{className:`data-file-actions`,children:[(0,w.jsxs)(`button`,{type:`button`,className:`data-btn`,onClick:B,disabled:b,children:[b?(0,w.jsx)(s,{size:14,className:`spin`}):(0,w.jsx)(v,{size:14}),` Upload`]}),(0,w.jsx)(`input`,{type:`file`,ref:M,hidden:!0,onChange:e=>{let t=e.target.files?.[0];t&&W(t)}})]})]}),(0,w.jsxs)(`p`,{className:`data-panel-subtitle`,children:[`Everything under your install's `,(0,w.jsx)(`code`,{children:`data/`}),` directory. Click a folder to open it, a file to download it. Uploads go to `,(0,w.jsx)(`code`,{children:`data/uploads/`}),`.`]}),(0,w.jsxs)(`div`,{className:`data-breadcrumbs`,children:[(0,w.jsx)(`button`,{type:`button`,className:`data-crumb`,onClick:()=>L(`.`),children:`data`}),l.map((e,t)=>{let n=l.slice(0,t+1).map(e=>e.name).join(`/`);return(0,w.jsxs)(`span`,{className:`data-crumb-wrap`,children:[(0,w.jsx)(`span`,{className:`data-crumb-sep`,children:`/`}),(0,w.jsx)(`button`,{type:`button`,className:`data-crumb`,onClick:()=>L(n),title:e.name,children:e.displayName??e.name})]},n)}),i!==`.`&&(0,w.jsxs)(`button`,{type:`button`,className:`data-btn data-btn-ghost data-up-btn`,onClick:R,children:[(0,w.jsx)(m,{size:14}),` Up`]})]}),S&&(0,w.jsxs)(`div`,{className:`data-error`,children:[`Upload failed: `,S]}),E&&(0,w.jsxs)(`div`,{className:`data-error`,children:[`Delete failed: `,E]}),p&&(0,w.jsx)(`div`,{className:`data-error`,children:p}),d&&(0,w.jsxs)(`div`,{className:`data-loading`,children:[(0,w.jsx)(s,{size:14,className:`spin`}),` Loading…`]}),!d&&o&&o.length===0&&!p&&(0,w.jsx)(`div`,{className:`data-empty-results`,children:`Empty directory.`}),!d&&o&&o.length>0&&(0,w.jsx)(`ul`,{className:`data-entries`,children:o.map(e=>{let n=e.displayName??e.name,r=O===e.name,a=A===e.name;return(0,w.jsx)(`li`,{className:`data-entry`,children:e.kind===`directory`?(0,w.jsxs)(`button`,{type:`button`,className:`data-entry-btn`,onClick:()=>L(i===`.`?e.name:`${i}/${e.name}`),children:[(0,w.jsx)(_,{size:14}),(0,w.jsx)(`span`,{className:`data-entry-name`,title:e.name,children:n})]}):e.kind===`file`?(0,w.jsxs)(w.Fragment,{children:[(0,w.jsxs)(`button`,{type:`button`,className:`data-entry-btn`,onClick:()=>z(e),title:`Download ${n}`,disabled:a,children:[(0,w.jsx)(g,{size:14}),(0,w.jsx)(`span`,{className:`data-entry-name`,title:e.name,children:n}),(0,w.jsxs)(`span`,{className:`data-entry-meta`,title:`Modified ${F(e.modifiedAt)}`,children:[P(e.sizeBytes),` · `,F(e.modifiedAt)]}),(0,w.jsx)(h,{size:12,className:`data-entry-download-icon`})]}),a?(0,w.jsxs)(`div`,{className:`data-entry-confirm`,role:`group`,"aria-label":`Confirm delete ${n}`,children:[(0,w.jsx)(`span`,{className:`data-entry-confirm-label`,children:`Delete?`}),(0,w.jsx)(`button`,{type:`button`,className:`data-entry-confirm-yes`,onClick:()=>U(e),autoFocus:!0,children:`Yes`}),(0,w.jsx)(`button`,{type:`button`,className:`data-entry-confirm-no`,onClick:H,children:`No`})]}):(0,w.jsx)(`button`,{type:`button`,className:`data-entry-delete`,onClick:()=>V(e),disabled:r,title:`Delete ${n}`,"aria-label":`Delete ${n}`,children:r?(0,w.jsx)(s,{size:14,className:`spin`}):(0,w.jsx)(t,{size:14})})]}):(0,w.jsxs)(`div`,{className:`data-entry-btn data-entry-disabled`,children:[(0,w.jsx)(g,{size:14}),(0,w.jsx)(`span`,{className:`data-entry-name`,children:n}),(0,w.jsx)(`span`,{className:`data-entry-meta`,children:`special`})]})},e.name)})})]})}function P(e){return e==null?``:e<1024?`${e} B`:e<1024*1024?`${(e/1024).toFixed(1)} KB`:e<1024*1024*1024?`${(e/1024/1024).toFixed(1)} MB`:`${(e/1024/1024/1024).toFixed(1)} GB`}function F(e){try{let t=new Date(e);if(isNaN(t.getTime()))return`—`;let n=typeof navigator<`u`?navigator.languages&&navigator.languages.length>0?[...navigator.languages]:[navigator.language]:void 0;return t.toLocaleString(n,{dateStyle:`medium`,timeStyle:`short`})}catch{return`—`}}export{h as a,g as i,E as n,S as r,T as t};
@@ -1,4 +1,4 @@
1
- import{o as e}from"./chunk-DD-I1_y5.js";import{i as t,o as n,r,t as i}from"./jsx-runtime-DnY0498s.js";import{A as a,C as o,D as s,F as c,I as l,N as u,P as d,T as f,_ as p,a as m,b as h,c as g,g as _,i as v,j as y,n as b,o as x,r as S,s as C,t as w,u as T,v as E,w as D,x as O,y as k}from"./useVoiceRecorder-CR8gcELb.js";import{t as A}from"./Checkbox-BySsatDO.js";var j=r(`square-plus`,[[`rect`,{width:`18`,height:`18`,x:`3`,y:`3`,rx:`2`,key:`afitv7`}],[`path`,{d:`M8 12h8`,key:`1wcyev`}],[`path`,{d:`M12 8v8`,key:`napkw2`}]]),M=t(),N=e(n(),1),P=`image/jpeg,image/png,image/gif,image/webp,application/pdf,text/plain,text/markdown,text/csv,text/html,text/calendar,audio/ogg,audio/opus,audio/mp4,audio/mpeg,audio/webm,audio/wav,.opus,.ogg,.m4a,.mp3,.wav,.webm`,F=new Set(P.split(`,`).filter(e=>!e.startsWith(`.`))),I=50*1024*1024,L=typeof window<`u`&&/^(localhost|127\.0\.0\.1|[a-z0-9-]+\.local)(:|$)/.test(window.location.hostname);function R(){let e=crypto.getRandomValues(new Uint8Array(16));e[6]=e[6]&15|64,e[8]=e[8]&63|128;let t=[...e].map(e=>e.toString(16).padStart(2,`0`)).join(``);return`${t.slice(0,8)}-${t.slice(8,12)}-${t.slice(12,16)}-${t.slice(16,20)}-${t.slice(20)}`}function z(){let e=window.location.pathname;if(e.startsWith(`/g/`))return;let t=e.match(/^\/([a-z][a-z0-9-]{2,49})$/);return t?t[1]:void 0}function B(){let e=window.location.pathname.match(/^\/g\/([a-z0-9][a-z0-9-]{0,49})$/);return e?e[1]:void 0}function V(e){return[{key:`length`,label:`At least 8 characters`,met:e.length>=8},{key:`number`,label:`Contains a number`,met:/\d/.test(e)},{key:`special`,label:`Contains a special character`,met:/[^A-Za-z0-9]/.test(e)},{key:`whitespace`,label:`No spaces`,met:e.length>0&&!/\s/.test(e)}]}function H(e,t){if(t===`phone`)return e.length>4?e.slice(0,e.length-4).replace(/\d/g,`•`)+` `+e.slice(-4):e;let n=e.indexOf(`@`);return n<=2?e:e.slice(0,2)+`•••`+e.slice(n)}function ee(e){let[t,n]=(0,N.useState)(null),[r,i]=(0,N.useState)(null),[a,o]=(0,N.useState)(`loading`),[s,c]=(0,N.useState)(``),[l,u]=(0,N.useState)(null),[d,f]=(0,N.useState)(null),[p,m]=(0,N.useState)(!1),[h,g]=(0,N.useState)(null),_=(0,N.useMemo)(()=>z(),[]),v=(0,N.useMemo)(()=>B(),[]),[y,b]=(0,N.useState)(null),x=(0,N.useRef)(_||``),S=(0,N.useRef)(null),C=(0,N.useRef)(null),w=(0,N.useRef)(!1),T=(0,N.useRef)(null),[E,D]=(0,N.useState)(`sign-in`),[O,k]=(0,N.useState)(null),A=(0,N.useRef)(null),j=(0,N.useRef)(null);(0,N.useEffect)(()=>{try{let e=sessionStorage.getItem(`maxy_session`);e&&(j.current=e)}catch{}},[]);let M=(0,N.useCallback)(t=>{S.current=t,n(t);try{sessionStorage.setItem(`maxy_session`,t)}catch{}o(`chat`),e(t)},[e]),P=(0,N.useCallback)(()=>{S.current=null;try{sessionStorage.removeItem(`maxy_session`)}catch{}n(null),o(`auth-required`),D(`sign-in`)},[]),F=(0,N.useCallback)(async e=>{try{let t=x.current,n=await fetch(`/api/access/verify-token`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({token:e,agentSlug:t})}),r=await n.json();n.ok?(A.current=r.session_key,k(r.grant),D(`set-password`),window.history.replaceState({},``,window.location.pathname)):D(`link-expired`)}catch{D(`sign-in`)}},[]),I=(0,N.useCallback)(async()=>{if(S.current)return S.current;if(C.current)return C.current;let e=(async()=>{try{let e=j.current,t=await fetch(`/api/session`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({session_id:R(),..._?{agent:_}:{},...v?{group_slug:v}:{},...e?{session_key:e}:{}})});if(!t.ok){let e=await t.json().catch(()=>({error:``}));return L&&console.error(`[session] POST /api/session failed: ${t.status} ${t.statusText}`,e),t.status===404?i(e.error||`Agent not found`):t.status===503&&i(e.error||`Service unavailable`),null}let r=await t.json();if(r.auth_required){r.agent_id&&(x.current=r.agent_id),c(r.displayName||``),r.agentImage&&u(r.agentImage),r.agentImageShape&&f(r.agentImageShape),r.showAgentName&&m(r.showAgentName),r.branding&&g(r.branding),o(`auth-required`);let e=new URLSearchParams(window.location.search).get(`token`);return e?F(e):D(`sign-in`),null}S.current=r.session_key,n(r.session_key),r.displayName&&c(r.displayName),r.agentImage&&u(r.agentImage),r.agentImageShape&&f(r.agentImageShape),r.showAgentName&&m(r.showAgentName),r.branding&&g(r.branding),o(`chat`),r.resumed&&r.messages&&(w.current=!0,T.current=r.messages),r.group&&b({groupSlug:r.groupSlug,groupName:r.groupName,participants:r.participants??[]});try{sessionStorage.setItem(`maxy_session`,r.session_key)}catch{}return r.session_key}catch(e){return L&&console.error(`[session] fetch /api/session threw:`,e),i(`Unable to connect. Please check your connection and try again.`),null}finally{C.current=null}})();return C.current=e,e},[_,v,F]);return{sessionId:t,sessionKeyRef:S,sessionError:r,pageState:a,setPageState:o,agentDisplayName:s,agentImage:l,agentImageShape:d,showAgentName:p,agentSlug:_,branding:h,resolvedSlugRef:x,ensureSession:I,startNewSession:(0,N.useCallback)(async()=>{S.current=null,C.current=null,j.current=null,w.current=!1,T.current=null,n(null),i(null);try{sessionStorage.removeItem(`maxy_session`)}catch{}let t=await I();t&&e(t)},[I,e]),enterChat:M,enterGate:P,resumedRef:w,resumeMessagesRef:T,gateState:E,setGateState:D,grantInfo:O,setGrantInfo:k,gateSessionKeyRef:A,groupContext:y,groupSlug:v}}function te({sessionKeyRef:e,ensureSession:t,sessionError:n,pageState:r,inputRef:i}){let[a,s]=(0,N.useState)([]),[c,l]=(0,N.useState)(!1),u=(0,N.useRef)(!1),d=e=>{u.current=e,l(e)},f=(0,N.useCallback)(async a=>{if(!u.current){d(!0),s([{role:`maxy`,content:``,timestamp:Date.now()}]);try{let n=await fetch(`/api/chat`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({greeting:!0,session_key:a})});if(n.status===401){e.current=null;try{sessionStorage.removeItem(`maxy_session`)}catch{}L&&console.warn(`[greeting] stale session, retrying with fresh session`);let r=await t();if(!r)return;n=await fetch(`/api/chat`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({greeting:!0,session_key:r})})}if(!n.ok){let e=await n.json().catch(()=>({}));throw Error(e.error||`Something went wrong`)}let r=``,i=``,c=()=>{i&&(r+=i,i=``,s(e=>{let t=[...e];return t[0]={...t[0],content:r},t}))},l=setInterval(c,60);try{for await(let e of o(n)){if(e.error)throw Object.assign(Error(e.error),{fromSSE:!0});e.text&&(i+=e.text),e.type===`component`&&(c(),s(t=>{let n=[...t],r={...n[0]},i=[...r.components||[]];return i.push({name:e.name,data:e.data,submitted:!1}),n[0]={...r,components:i},n}))}}finally{clearInterval(l),c()}}catch(e){if(L&&console.error(`[chat] greeting failed:`,e),r===`auth-required`)return;let t=e instanceof Error?e.message:String(e);s([{role:`maxy`,content:e instanceof Error&&e.fromSSE?t:n??`I'm having trouble connecting right now. Try refreshing the page.`,timestamp:Date.now()}])}finally{d(!1),r===`chat`&&i.current?.focus()}}},[e,t,n,r,i]),p=(0,N.useCallback)(async(a,c)=>{let l=c?.hidden??!1,f=c?.files??[];if(!a&&f.length===0||u.current)return;d(!0);let p=f.map(e=>({filename:e.name,mimeType:e.type})),m={role:`visitor`,content:a,attachments:p.length>0?p:void 0,timestamp:Date.now(),hidden:l},h;s(e=>{let t=[...e,m,{role:`maxy`,content:``,timestamp:Date.now()}];return h=t.length-1,t});try{let n=await t();if(!n)throw Error(`session`);let r=e=>{if(f.length>0){let t=new FormData;t.append(`message`,a),t.append(`session_key`,e);for(let e of f)t.append(`attachments`,e);return{body:t,headers:{}}}return{body:JSON.stringify({message:a,session_key:e}),headers:{"Content-Type":`application/json`}}},{body:i,headers:c}=r(n),l=await fetch(`/api/chat`,{method:`POST`,headers:c,body:i});if(l.status===401){e.current=null;try{sessionStorage.removeItem(`maxy_session`)}catch{}L&&console.warn(`[session-expired] stale key cleared, retrying with fresh session`);let n=await t();if(!n)throw Error(`session`);({body:i,headers:c}=r(n)),l=await fetch(`/api/chat`,{method:`POST`,headers:c,body:i})}if(!l.ok){let e=await l.json().catch(()=>({}));throw Error(e.error||`Something went wrong`)}let u=``,d=``,p=()=>{d&&(u+=d,d=``,s(e=>{let t=[...e];return t[h]={...t[h],content:u},t}))},m=setInterval(p,60);try{for await(let e of o(l)){if(e.error)throw Object.assign(Error(e.error),{fromSSE:!0});e.text&&(d+=e.text),e.type===`component`&&(p(),s(t=>{let n=[...t],r={...n[h]},i=[...r.components||[]];return i.push({name:e.name,data:e.data,submitted:!1}),n[h]={...r,components:i},n}))}}finally{clearInterval(m),p()}}catch(e){if(L&&console.error(`[chat] sendMessage failed:`,e),r===`auth-required`)return;let t=e instanceof Error?e.message:String(e),i=e instanceof Error&&e.fromSSE;s(e=>{let r=[...e];return r[h]={...r[h],content:i?t:t===`session`?n??`I'm having trouble connecting right now. Try refreshing the page.`:`Sorry, I hit a snag. Try again in a moment.`},r})}finally{d(!1),r===`chat`&&i.current?.focus()}},[e,t,n,r,i]);return{messages:a,setMessages:s,isStreaming:c,sendMessage:p,sendGreeting:f,handleComponentSubmit:(0,N.useCallback)((e,t,n)=>{s(n=>{let r=[...n],i={...r[e]},a=[...i.components||[]];return a[t]={...a[t],submitted:!0},r[e]={...i,components:a},r}),p(n,{hidden:!0})},[p])}}function ne({sessionKeyRef:e,groupSlug:t,isStreaming:n,setMessages:r}){let i=(0,N.useRef)(new Date().toISOString()),a=(0,N.useRef)(!0);(0,N.useEffect)(()=>{let e=()=>{a.current=document.visibilityState===`visible`};return document.addEventListener(`visibilitychange`,e),()=>document.removeEventListener(`visibilitychange`,e)},[]);let o=(0,N.useCallback)(async()=>{let n=e.current;if(!(!n||!t))try{let e=await fetch(`/api/group/messages?session_key=${encodeURIComponent(n)}&since=${encodeURIComponent(i.current)}`);if(!e.ok)return;let t=await e.json();if(!t.messages||t.messages.length===0)return;let a=t.messages[t.messages.length-1];a.timestamp&&(i.current=new Date(a.timestamp).toISOString()),r(e=>{let n=new Set(e.filter(e=>e.messageId).map(e=>e.messageId)),r=t.messages.filter(e=>!n.has(e.messageId)).map(e=>({role:e.role,content:e.content,timestamp:e.timestamp,senderName:e.senderName??void 0,senderVisitorId:e.senderVisitorId??void 0,messageId:e.messageId}));if(r.length===0)return e;let i=[...e,...r];return i.sort((e,t)=>e.timestamp-t.timestamp),i})}catch(e){console.error(`[group-poll] failed: ${e instanceof Error?e.message:String(e)}`)}},[e,t,r]);(0,N.useEffect)(()=>{if(!t)return;let e,r=!1;function i(){if(r)return;let t=a.current?3e3:3e4;e=setTimeout(async()=>{n||await o(),i()},t)}return i(),()=>{r=!0,clearTimeout(e)}},[t,n,o])}var U=i();function W({value:e,onChange:t,onComplete:n,disabled:r}){let i=(0,N.useRef)([]);function a(n,r){r.key===`Backspace`?(r.preventDefault(),e[n]?t(e.slice(0,n)+e.slice(n+1)):n>0&&(t(e.slice(0,n-1)+e.slice(n)),i.current[n-1]?.focus())):r.key===`ArrowLeft`&&n>0?i.current[n-1]?.focus():r.key===`ArrowRight`&&n<5&&i.current[n+1]?.focus()}function o(r,a){let o=a.nativeEvent.data;if(!o||!/^\d$/.test(o))return;let s=e.split(``);for(s[r]=o;s.length<r;)s.push(``);let c=s.join(``).replace(/\D/g,``).slice(0,6);t(c),c.length===6?n(c):r<5&&i.current[r+1]?.focus()}function s(e){e.preventDefault();let r=e.clipboardData.getData(`text`).replace(/\D/g,``).slice(0,6);r&&(t(r),r.length===6?n(r):i.current[r.length]?.focus())}return(0,U.jsx)(`div`,{className:`gate-otp-field`,children:Array.from({length:6}).map((t,n)=>(0,U.jsx)(`input`,{ref:e=>{i.current[n]=e},type:`text`,inputMode:`numeric`,className:`pin-box${e[n]?` pin-box-filled`:``}`,value:e[n]||``,onKeyDown:e=>a(n,e),onInput:e=>o(n,e),onPaste:s,onFocus:e=>e.target.select(),autoFocus:n===0,autoComplete:`off`,maxLength:1,disabled:r,"aria-label":`Code digit ${n+1}`},n))})}function re({gateState:e,setGateState:t,grantInfo:n,setGrantInfo:r,gateSessionKeyRef:i,resolvedSlugRef:a,onAuthenticated:o}){let[s,c]=(0,N.useState)(``),[l,u]=(0,N.useState)(``),[d,f]=(0,N.useState)(``),[p,m]=(0,N.useState)(!1),[h,g]=(0,N.useState)(``),[v,y]=(0,N.useState)(``),[b,x]=(0,N.useState)(!1),[S,C]=(0,N.useState)(null),[w,T]=(0,N.useState)(null),[E,D]=(0,N.useState)(!1);async function O(e){if(e.preventDefault(),!E){C(null),D(!0);try{let e=a.current,n=await fetch(`/api/access/login`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({contact:s,password:l,agentSlug:e})}),i=await n.json();n.ok?o(i.session_key):n.status===401?i.error?.includes(`setup not complete`)||i.error?.includes(`invitation link`)?(t(`private-agent`),C(null)):C(i.error||`Invalid credentials`):n.status===403?(t(`access-expired`),i.expiresAt&&r(e=>e?{...e,expiresAt:i.expiresAt}:{displayName:null,contactValue:s,contactMethod:`email`,expiresAt:i.expiresAt,status:`expired`})):n.status===429?C(i.error||`Too many attempts. Please try again later.`):C(i.error||`Something went wrong`)}catch{C(`Unable to connect. Please try again.`)}finally{D(!1)}}}async function k(e){if(!E){C(null),D(!0);try{let n=a.current,o=await fetch(`/api/access/verify-otp`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({phone:v,code:e,agentSlug:n})}),s=await o.json();o.ok?(i.current=s.session_key,r(s.grant),t(`set-password`)):o.status===429?t(`otp-failed`):(C(s.error||`Invalid code`),g(``))}catch{C(`Unable to connect. Please try again.`)}finally{D(!1)}}}async function j(e){if(e.preventDefault(),!E){if(C(null),l!==d){C(`Passwords do not match`);return}if(!V(l).every(e=>e.met)){C(`Password does not meet requirements`);return}D(!0);try{let e=i.current;if(!e){C(`Session expired. Please use your invitation link again.`);return}let t=await fetch(`/api/access/create-credentials`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({session_key:e,password:l})}),n=await t.json();t.ok?o(n.session_key):t.status===400&&n.requirements?C(n.requirements.filter(e=>!e.met).map(e=>e.label).join(`, `)):C(n.error||`Something went wrong`)}catch{C(`Unable to connect. Please try again.`)}finally{D(!1)}}}async function M(e){if(e.preventDefault(),!(E||!s)){C(null),D(!0);try{let e=a.current,t=await fetch(`/api/access/forgot-password`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({contact:s,agentSlug:e})});t.status===429?C((await t.json()).error||`Too many attempts. Please try again later.`):(T(`If an account exists, a reset link has been sent.`),setTimeout(()=>{x(!1),T(null)},4e3))}catch{C(`Unable to connect. Please try again.`)}finally{D(!1)}}}switch(e){case`set-password`:{let e=V(l),t=e.every(e=>e.met),r=l===d,i=t&&r&&d.length>0;return(0,U.jsx)(`div`,{className:`gate-wrap`,children:(0,U.jsxs)(`div`,{className:`gate-card`,children:[n?.expiresAt&&(0,U.jsxs)(`div`,{className:`gate-expiry-badge`,children:[`Access until `,new Date(n.expiresAt).toLocaleDateString(`en-GB`,{day:`numeric`,month:`short`,year:`numeric`})]}),(0,U.jsxs)(`h2`,{className:`gate-title`,children:[`Welcome, `,n?.displayName||`there`]}),(0,U.jsx)(`p`,{className:`gate-subtitle`,children:`Create a password to access this agent in the future.`}),(0,U.jsxs)(`form`,{className:`gate-form`,onSubmit:j,children:[(0,U.jsxs)(`div`,{className:`gate-field`,children:[(0,U.jsx)(`label`,{htmlFor:`gate-display-name`,children:`Display name`}),(0,U.jsx)(`input`,{id:`gate-display-name`,type:`text`,value:n?.displayName||``,disabled:!0})]}),(0,U.jsxs)(`div`,{className:`gate-field gate-pw-row`,children:[(0,U.jsx)(`label`,{htmlFor:`gate-pw`,children:`Password`}),(0,U.jsx)(`div`,{className:`gate-pw-toggle`,children:(0,U.jsx)(A,{checked:p,onChange:m,label:`Show`})}),(0,U.jsx)(`input`,{id:`gate-pw`,type:p?`text`:`password`,value:l,onChange:e=>u(e.target.value),placeholder:`Your password`,autoFocus:!0,autoComplete:`new-password`}),l.length>0&&(0,U.jsx)(`div`,{className:`gate-strength`,children:e.map(e=>(0,U.jsxs)(`span`,{className:`gate-strength-item${e.met?` met`:``}`,children:[e.met?`✓`:`○`,` `,e.label]},e.key))})]}),(0,U.jsxs)(`div`,{className:`gate-field`,children:[(0,U.jsx)(`label`,{htmlFor:`gate-pw-confirm`,children:`Confirm password`}),(0,U.jsx)(`input`,{id:`gate-pw-confirm`,type:p?`text`:`password`,value:d,onChange:e=>f(e.target.value),placeholder:`Confirm your password`,autoComplete:`new-password`}),d.length>0&&!r&&(0,U.jsx)(`div`,{className:`gate-error`,children:`Passwords do not match`})]}),S&&(0,U.jsx)(`div`,{className:`gate-error`,children:S}),(0,U.jsx)(`div`,{className:`gate-submit`,children:(0,U.jsx)(_,{variant:`primary`,type:`submit`,fullWidth:!0,disabled:!i||E,loading:E,children:`Create account & enter chat`})})]})]})})}case`enter-otp`:return(0,U.jsx)(`div`,{className:`gate-wrap`,children:(0,U.jsxs)(`div`,{className:`gate-card`,children:[(0,U.jsx)(`h2`,{className:`gate-title`,children:`Enter your code`}),(0,U.jsxs)(`p`,{className:`gate-subtitle`,children:[`Enter the 6-digit code sent to `,H(v,`phone`)]}),(0,U.jsx)(W,{value:h,onChange:g,onComplete:k,disabled:E}),S&&(0,U.jsx)(`div`,{className:`gate-error`,children:S}),E&&(0,U.jsxs)(`div`,{className:`gate-loading`,children:[(0,U.jsx)(`span`,{className:`spinner`}),`Verifying...`]}),(0,U.jsx)(`div`,{className:`gate-resend`,children:(0,U.jsx)(`button`,{type:`button`,className:`gate-link`,onClick:()=>{t(`sign-in`),g(``),C(null)},children:`Use a different number`})})]})});case`sign-in`:case`private-agent`:return(0,U.jsx)(`div`,{className:`gate-wrap`,children:(0,U.jsxs)(`div`,{className:`gate-card`,children:[(0,U.jsx)(`h2`,{className:`gate-title`,children:e===`private-agent`?`Private Agent`:`Welcome back`}),(0,U.jsx)(`p`,{className:`gate-subtitle`,children:e===`private-agent`?`This agent is invitation-only. If you have an account, sign in below.`:`Sign in to continue your conversation.`}),b?(0,U.jsxs)(`form`,{className:`gate-form`,onSubmit:M,children:[(0,U.jsxs)(`div`,{className:`gate-field`,children:[(0,U.jsx)(`label`,{htmlFor:`gate-forgot-contact`,children:`Email or phone`}),(0,U.jsx)(`input`,{id:`gate-forgot-contact`,type:`text`,value:s,onChange:e=>c(e.target.value),placeholder:`you@example.com`,autoFocus:!0,autoComplete:`email`})]}),S&&(0,U.jsx)(`div`,{className:`gate-error`,children:S}),w&&(0,U.jsx)(`div`,{className:`gate-success`,children:w}),(0,U.jsx)(`div`,{className:`gate-submit`,children:(0,U.jsx)(_,{variant:`primary`,type:`submit`,fullWidth:!0,disabled:!s.trim()||E,loading:E,children:`Send reset link`})}),(0,U.jsx)(`div`,{className:`gate-actions`,children:(0,U.jsx)(`button`,{type:`button`,className:`gate-link`,onClick:()=>{x(!1),C(null),T(null)},children:`Back to sign in`})})]}):(0,U.jsxs)(`form`,{className:`gate-form`,onSubmit:O,children:[(0,U.jsxs)(`div`,{className:`gate-field`,children:[(0,U.jsx)(`label`,{htmlFor:`gate-contact`,children:`Email or phone`}),(0,U.jsx)(`input`,{id:`gate-contact`,type:`text`,value:s,onChange:e=>c(e.target.value),placeholder:`you@example.com`,autoFocus:!0,autoComplete:`email`})]}),(0,U.jsxs)(`div`,{className:`gate-field gate-pw-row`,children:[(0,U.jsx)(`label`,{htmlFor:`gate-login-pw`,children:`Password`}),(0,U.jsx)(`div`,{className:`gate-pw-toggle`,children:(0,U.jsx)(A,{checked:p,onChange:m,label:`Show`})}),(0,U.jsx)(`input`,{id:`gate-login-pw`,type:p?`text`:`password`,value:l,onChange:e=>u(e.target.value),placeholder:`Your password`,autoComplete:`current-password`})]}),S&&(0,U.jsx)(`div`,{className:`gate-error`,children:S}),(0,U.jsx)(`div`,{className:`gate-submit`,children:(0,U.jsx)(_,{variant:`primary`,type:`submit`,fullWidth:!0,disabled:!s.trim()||!l||E,loading:E,children:`Sign in`})}),(0,U.jsx)(`div`,{className:`gate-actions`,children:(0,U.jsx)(`button`,{type:`button`,className:`gate-link`,onClick:()=>{x(!0),C(null)},children:`Forgot password?`})})]}),e===`private-agent`&&(0,U.jsx)(`p`,{className:`gate-hint`,children:`No account? You need an invitation from the agent owner.`})]})});case`access-expired`:return(0,U.jsx)(`div`,{className:`gate-wrap`,children:(0,U.jsxs)(`div`,{className:`gate-card`,children:[(0,U.jsx)(`div`,{className:`gate-icon`,children:`⏰`}),(0,U.jsx)(`h2`,{className:`gate-title`,children:`Access expired`}),(0,U.jsxs)(`p`,{className:`gate-subtitle`,children:[n?.expiresAt?`Your access expired on ${new Date(n.expiresAt).toLocaleDateString(`en-GB`,{day:`numeric`,month:`long`,year:`numeric`})}.`:`Your access to this agent has expired.`,` `,`Contact the agent owner to renew.`]}),(0,U.jsx)(`div`,{className:`gate-actions`,children:(0,U.jsx)(_,{variant:`secondary`,onClick:()=>{t(`sign-in`),C(null),u(``)},children:`Sign in with a different account`})})]})});case`link-expired`:return(0,U.jsx)(`div`,{className:`gate-wrap`,children:(0,U.jsxs)(`div`,{className:`gate-card`,children:[(0,U.jsx)(`div`,{className:`gate-icon`,children:`⚠️`}),(0,U.jsx)(`h2`,{className:`gate-title`,children:`Link expired`}),(0,U.jsx)(`p`,{className:`gate-subtitle`,children:`This invitation link has expired or has already been used. If you already set your password, sign in below. Otherwise, ask the agent owner to send a new invitation.`}),(0,U.jsx)(`div`,{className:`gate-actions`,children:(0,U.jsx)(_,{variant:`primary`,onClick:()=>{t(`sign-in`),C(null)},children:`Go to sign in`})})]})});case`otp-failed`:return(0,U.jsx)(`div`,{className:`gate-wrap`,children:(0,U.jsxs)(`div`,{className:`gate-card`,children:[(0,U.jsx)(`div`,{className:`gate-icon`,children:`⛔`}),(0,U.jsx)(`h2`,{className:`gate-title`,children:`Too many attempts`}),(0,U.jsx)(`p`,{className:`gate-subtitle`,children:`You've entered the wrong code too many times. Ask the agent owner to send a new code.`}),(0,U.jsx)(`div`,{className:`gate-actions`,children:(0,U.jsx)(_,{variant:`secondary`,onClick:()=>{t(`sign-in`),C(null),g(``)},children:`Back to sign in`})})]})})}}var G={"single-select":g,"multi-select":C,"action-buttons":x};function K({name:e,data:t,onSubmit:n,submitted:r}){let i=G[e];return i?(0,U.jsx)(i,{data:t,onSubmit:n,submitted:r}):(console.warn(`[PublicComponentRenderer] Unknown component: "${e}". Registered: ${Object.keys(G).join(`, `)}`),(0,U.jsx)(`div`,{className:`component-card component-card--error`,children:(0,U.jsxs)(`p`,{style:{fontFamily:`var(--font-body)`,fontSize:12,color:`var(--text-secondary)`},children:[`Component “`,e,`” is not available. This may require a platform update.`]})}))}function ie({messages:e,isStreaming:t,sessionError:n,selectionMode:r,selectedItems:i,toggleSelectItem:a,onComponentSubmit:o,remainingSuggestions:s,onSuggestionClick:c,isAtBottom:f,setIsAtBottom:p,isGroup:h,sessionId:g}){let[v,y]=(0,N.useState)(new Set),[b,x]=(0,N.useState)(new Set),S=(0,N.useRef)(null),C=(0,N.useRef)(null),w=(0,N.useRef)(null),E=(0,N.useRef)(!1),D=(0,N.useRef)(!1),O=(0,N.useRef)(!1),k=(0,N.useRef)(0),j=(0,N.useRef)(!1),M=(0,N.useRef)(new Map);j.current=t,(0,N.useEffect)(()=>{t&&!E.current&&f&&(D.current=!0)},[t,f]),(0,N.useEffect)(()=>{if(!D.current&&!f){E.current=t,O.current=!1;return}if(E.current&&!t&&w.current&&C.current){let e=w.current;if(e.offsetHeight>C.current.clientHeight){D.current=!1,O.current=!0,e.scrollIntoView({behavior:`smooth`,block:`start`}),E.current=t;return}}(f||D.current)&&!O.current&&S.current?.scrollIntoView({behavior:`smooth`}),E.current&&!t&&(D.current=!1),E.current=t},[e,f,t]),(0,N.useEffect)(()=>{let e=C.current;if(!e)return;let t=()=>{let t=e.scrollTop;j.current&&t<k.current&&(D.current=!1),k.current=t,p(e.scrollHeight-e.scrollTop-e.clientHeight<80)};return e.addEventListener(`scroll`,t,{passive:!0}),t(),()=>e.removeEventListener(`scroll`,t)},[p]),(0,N.useEffect)(()=>{function e(){x(e=>{let t=new Set;return M.current.forEach((n,r)=>{n.isConnected&&(v.has(r)?e.has(r)&&t.add(r):n.scrollHeight>n.clientHeight+2&&t.add(r))}),e.size===t.size&&[...e].every(e=>t.has(e))?e:t})}e();let t=new ResizeObserver(e);return M.current.forEach(e=>t.observe(e)),()=>t.disconnect()},[e.length,v]);let P=[...e].sort((e,t)=>e.timestamp-t.timestamp);(0,N.useEffect)(()=>{r&&console.log(`[selection] entered`,{conversationId:g??`unknown`,order:P.map(e=>({ts:e.timestamp,role:e.role,id:e.messageId}))})},[r]);let F=P.reduce((e,t,n)=>t.role===`maxy`&&!t.hidden?n:e,-1);return(0,U.jsxs)(`div`,{className:`chat-messages-wrap`,children:[r&&(0,U.jsx)(`div`,{className:`selection-overlay-band`}),(0,U.jsxs)(`div`,{className:`chat-messages`,ref:C,children:[n&&e.length===0&&(0,U.jsx)(`div`,{className:`session-error`,children:(0,U.jsx)(`p`,{children:n})}),P.map((e,n)=>{if(e.hidden)return null;let s=t&&n===P.length-1,c=`${e.timestamp}_${e.role}`,f=i.has(c);return(0,U.jsxs)(`div`,{ref:n===F?w:void 0,className:`message ${e.role}${f?` selected`:``}`,children:[r&&!s&&(0,U.jsx)(`div`,{className:`message-select-check`,children:(0,U.jsx)(A,{checked:f,onChange:()=>a(c)})}),(0,U.jsxs)(`div`,{className:`bubble`,children:[h&&e.role===`visitor`&&e.senderName&&(0,U.jsx)(`span`,{className:`bubble-sender`,children:e.senderName}),e.role===`visitor`&&e.content?(()=>{let t=v.has(n),r=b.has(n);return(0,U.jsxs)(U.Fragment,{children:[(0,U.jsx)(`div`,{ref:e=>{e?M.current.set(n,e):M.current.delete(n)},className:t?`bubble-content`:`bubble-content clamped${r?` overflowing`:``}`,children:e.content}),r&&(0,U.jsx)(`div`,{className:`bubble-expand`,children:(0,U.jsx)(_,{variant:`icon`,className:t?`expanded`:``,onClick:()=>y(e=>{let t=new Set(e);return t.has(n)?t.delete(n):t.add(n),t}),"aria-label":t?`Collapse message`:`Expand message`,children:(0,U.jsx)(l,{size:14})})})]})})():e.content?(0,U.jsx)(T,{content:e.content,timestamp:m(e.timestamp)}):(0,U.jsxs)(`span`,{className:`typing-indicator`,children:[(0,U.jsx)(`span`,{className:`typing-dot`}),(0,U.jsx)(`span`,{className:`typing-dot`}),(0,U.jsx)(`span`,{className:`typing-dot`})]}),e.attachments&&e.attachments.length>0&&(0,U.jsx)(`div`,{className:`message-attachments`,children:e.attachments.map((e,t)=>(0,U.jsxs)(`span`,{className:`message-attachment-chip no-action`,children:[e.mimeType===`application/pdf`?(0,U.jsx)(u,{size:12}):e.mimeType.startsWith(`text/`)?(0,U.jsx)(d,{size:12}):null,(0,U.jsx)(`span`,{children:e.filename})]},t))}),e.role===`visitor`&&(0,U.jsx)(`span`,{className:`message-timestamp`,children:m(e.timestamp)})]}),e.role===`maxy`&&e.components&&e.components.length>0&&(0,U.jsx)(`div`,{className:`public-components`,children:e.components.map((e,t)=>(0,U.jsx)(`div`,{className:`public-component-enter`,children:(0,U.jsx)(K,{name:e.name,data:e.data,onSubmit:e=>o(n,t,e),submitted:e.submitted})},t))})]},n)}),!t&&s.length>0&&(0,U.jsx)(`div`,{className:`chat-suggestions`,children:s.map(e=>(0,U.jsx)(_,{variant:`suggestion`,onClick:()=>c(e),children:e},e))}),(0,U.jsx)(`div`,{ref:S})]}),!f&&(0,U.jsx)(`button`,{className:`scroll-to-bottom`,onClick:()=>S.current?.scrollIntoView({behavior:`smooth`}),"aria-label":`Scroll to bottom`,children:(0,U.jsx)(l,{size:18})})]})}function ae({selectedItems:e,messages:t,isStreaming:n,copySelected:r,selectAll:i,exitSelection:a,showCopyToast:o}){let[l,u]=(0,N.useState)(!1),d=(0,N.useRef)(null),f=(0,N.useRef)(!1);function p(e){let n=e.lastIndexOf(`_`),r=e.slice(0,n),i=e.slice(n+1),a=parseInt(r,10);return t.find(e=>e.timestamp===a&&e.role===i)?.content??``}function m(e,t){return parseInt(e.slice(0,e.lastIndexOf(`_`)),10)-parseInt(t.slice(0,t.lastIndexOf(`_`)),10)}function h(){let e=[...t].sort((e,t)=>e.timestamp-t.timestamp),n=[];for(let t of e){if(t.hidden||!t.content)continue;let e=t.role===`visitor`?`Visitor`:`Maxy`;n.push(`${e}:\n${t.content}`)}return n.join(`
1
+ import{o as e}from"./chunk-DD-I1_y5.js";import{i as t,o as n,r,t as i}from"./jsx-runtime-nxP_2eNo.js";import{A as a,C as o,D as s,F as c,I as l,N as u,P as d,T as f,_ as p,a as m,b as h,c as g,g as _,i as v,j as y,n as b,o as x,r as S,s as C,t as w,u as T,v as E,w as D,x as O,y as k}from"./useVoiceRecorder-DyVx7e7a.js";import{t as A}from"./Checkbox-B9hff9s8.js";var j=r(`square-plus`,[[`rect`,{width:`18`,height:`18`,x:`3`,y:`3`,rx:`2`,key:`afitv7`}],[`path`,{d:`M8 12h8`,key:`1wcyev`}],[`path`,{d:`M12 8v8`,key:`napkw2`}]]),M=t(),N=e(n(),1),P=`image/jpeg,image/png,image/gif,image/webp,application/pdf,text/plain,text/markdown,text/csv,text/html,text/calendar,audio/ogg,audio/opus,audio/mp4,audio/mpeg,audio/webm,audio/wav,.opus,.ogg,.m4a,.mp3,.wav,.webm`,F=new Set(P.split(`,`).filter(e=>!e.startsWith(`.`))),I=50*1024*1024,L=typeof window<`u`&&/^(localhost|127\.0\.0\.1|[a-z0-9-]+\.local)(:|$)/.test(window.location.hostname);function R(){let e=crypto.getRandomValues(new Uint8Array(16));e[6]=e[6]&15|64,e[8]=e[8]&63|128;let t=[...e].map(e=>e.toString(16).padStart(2,`0`)).join(``);return`${t.slice(0,8)}-${t.slice(8,12)}-${t.slice(12,16)}-${t.slice(16,20)}-${t.slice(20)}`}function z(){let e=window.location.pathname;if(e.startsWith(`/g/`))return;let t=e.match(/^\/([a-z][a-z0-9-]{2,49})$/);return t?t[1]:void 0}function B(){let e=window.location.pathname.match(/^\/g\/([a-z0-9][a-z0-9-]{0,49})$/);return e?e[1]:void 0}function V(e){return[{key:`length`,label:`At least 8 characters`,met:e.length>=8},{key:`number`,label:`Contains a number`,met:/\d/.test(e)},{key:`special`,label:`Contains a special character`,met:/[^A-Za-z0-9]/.test(e)},{key:`whitespace`,label:`No spaces`,met:e.length>0&&!/\s/.test(e)}]}function H(e,t){if(t===`phone`)return e.length>4?e.slice(0,e.length-4).replace(/\d/g,`•`)+` `+e.slice(-4):e;let n=e.indexOf(`@`);return n<=2?e:e.slice(0,2)+`•••`+e.slice(n)}function ee(e){let[t,n]=(0,N.useState)(null),[r,i]=(0,N.useState)(null),[a,o]=(0,N.useState)(`loading`),[s,c]=(0,N.useState)(``),[l,u]=(0,N.useState)(null),[d,f]=(0,N.useState)(null),[p,m]=(0,N.useState)(!1),[h,g]=(0,N.useState)(null),_=(0,N.useMemo)(()=>z(),[]),v=(0,N.useMemo)(()=>B(),[]),[y,b]=(0,N.useState)(null),x=(0,N.useRef)(_||``),S=(0,N.useRef)(null),C=(0,N.useRef)(null),w=(0,N.useRef)(!1),T=(0,N.useRef)(null),[E,D]=(0,N.useState)(`sign-in`),[O,k]=(0,N.useState)(null),A=(0,N.useRef)(null),j=(0,N.useRef)(null);(0,N.useEffect)(()=>{try{let e=sessionStorage.getItem(`maxy_session`);e&&(j.current=e)}catch{}},[]);let M=(0,N.useCallback)(t=>{S.current=t,n(t);try{sessionStorage.setItem(`maxy_session`,t)}catch{}o(`chat`),e(t)},[e]),P=(0,N.useCallback)(()=>{S.current=null;try{sessionStorage.removeItem(`maxy_session`)}catch{}n(null),o(`auth-required`),D(`sign-in`)},[]),F=(0,N.useCallback)(async e=>{try{let t=x.current,n=await fetch(`/api/access/verify-token`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({token:e,agentSlug:t})}),r=await n.json();n.ok?(A.current=r.session_key,k(r.grant),D(`set-password`),window.history.replaceState({},``,window.location.pathname)):D(`link-expired`)}catch{D(`sign-in`)}},[]),I=(0,N.useCallback)(async()=>{if(S.current)return S.current;if(C.current)return C.current;let e=(async()=>{try{let e=j.current,t=await fetch(`/api/session`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({session_id:R(),..._?{agent:_}:{},...v?{group_slug:v}:{},...e?{session_key:e}:{}})});if(!t.ok){let e=await t.json().catch(()=>({error:``}));return L&&console.error(`[session] POST /api/session failed: ${t.status} ${t.statusText}`,e),t.status===404?i(e.error||`Agent not found`):t.status===503&&i(e.error||`Service unavailable`),null}let r=await t.json();if(r.auth_required){r.agent_id&&(x.current=r.agent_id),c(r.displayName||``),r.agentImage&&u(r.agentImage),r.agentImageShape&&f(r.agentImageShape),r.showAgentName&&m(r.showAgentName),r.branding&&g(r.branding),o(`auth-required`);let e=new URLSearchParams(window.location.search).get(`token`);return e?F(e):D(`sign-in`),null}S.current=r.session_key,n(r.session_key),r.displayName&&c(r.displayName),r.agentImage&&u(r.agentImage),r.agentImageShape&&f(r.agentImageShape),r.showAgentName&&m(r.showAgentName),r.branding&&g(r.branding),o(`chat`),r.resumed&&r.messages&&(w.current=!0,T.current=r.messages),r.group&&b({groupSlug:r.groupSlug,groupName:r.groupName,participants:r.participants??[]});try{sessionStorage.setItem(`maxy_session`,r.session_key)}catch{}return r.session_key}catch(e){return L&&console.error(`[session] fetch /api/session threw:`,e),i(`Unable to connect. Please check your connection and try again.`),null}finally{C.current=null}})();return C.current=e,e},[_,v,F]);return{sessionId:t,sessionKeyRef:S,sessionError:r,pageState:a,setPageState:o,agentDisplayName:s,agentImage:l,agentImageShape:d,showAgentName:p,agentSlug:_,branding:h,resolvedSlugRef:x,ensureSession:I,startNewSession:(0,N.useCallback)(async()=>{S.current=null,C.current=null,j.current=null,w.current=!1,T.current=null,n(null),i(null);try{sessionStorage.removeItem(`maxy_session`)}catch{}let t=await I();t&&e(t)},[I,e]),enterChat:M,enterGate:P,resumedRef:w,resumeMessagesRef:T,gateState:E,setGateState:D,grantInfo:O,setGrantInfo:k,gateSessionKeyRef:A,groupContext:y,groupSlug:v}}function te({sessionKeyRef:e,ensureSession:t,sessionError:n,pageState:r,inputRef:i}){let[a,s]=(0,N.useState)([]),[c,l]=(0,N.useState)(!1),u=(0,N.useRef)(!1),d=e=>{u.current=e,l(e)},f=(0,N.useCallback)(async a=>{if(!u.current){d(!0),s([{role:`maxy`,content:``,timestamp:Date.now()}]);try{let n=await fetch(`/api/chat`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({greeting:!0,session_key:a})});if(n.status===401){e.current=null;try{sessionStorage.removeItem(`maxy_session`)}catch{}L&&console.warn(`[greeting] stale session, retrying with fresh session`);let r=await t();if(!r)return;n=await fetch(`/api/chat`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({greeting:!0,session_key:r})})}if(!n.ok){let e=await n.json().catch(()=>({}));throw Error(e.error||`Something went wrong`)}let r=``,i=``,c=()=>{i&&(r+=i,i=``,s(e=>{let t=[...e];return t[0]={...t[0],content:r},t}))},l=setInterval(c,60);try{for await(let e of o(n)){if(e.error)throw Object.assign(Error(e.error),{fromSSE:!0});e.text&&(i+=e.text),e.type===`component`&&(c(),s(t=>{let n=[...t],r={...n[0]},i=[...r.components||[]];return i.push({name:e.name,data:e.data,submitted:!1}),n[0]={...r,components:i},n}))}}finally{clearInterval(l),c()}}catch(e){if(L&&console.error(`[chat] greeting failed:`,e),r===`auth-required`)return;let t=e instanceof Error?e.message:String(e);s([{role:`maxy`,content:e instanceof Error&&e.fromSSE?t:n??`I'm having trouble connecting right now. Try refreshing the page.`,timestamp:Date.now()}])}finally{d(!1),r===`chat`&&i.current?.focus()}}},[e,t,n,r,i]),p=(0,N.useCallback)(async(a,c)=>{let l=c?.hidden??!1,f=c?.files??[];if(!a&&f.length===0||u.current)return;d(!0);let p=f.map(e=>({filename:e.name,mimeType:e.type})),m={role:`visitor`,content:a,attachments:p.length>0?p:void 0,timestamp:Date.now(),hidden:l},h;s(e=>{let t=[...e,m,{role:`maxy`,content:``,timestamp:Date.now()}];return h=t.length-1,t});try{let n=await t();if(!n)throw Error(`session`);let r=e=>{if(f.length>0){let t=new FormData;t.append(`message`,a),t.append(`session_key`,e);for(let e of f)t.append(`attachments`,e);return{body:t,headers:{}}}return{body:JSON.stringify({message:a,session_key:e}),headers:{"Content-Type":`application/json`}}},{body:i,headers:c}=r(n),l=await fetch(`/api/chat`,{method:`POST`,headers:c,body:i});if(l.status===401){e.current=null;try{sessionStorage.removeItem(`maxy_session`)}catch{}L&&console.warn(`[session-expired] stale key cleared, retrying with fresh session`);let n=await t();if(!n)throw Error(`session`);({body:i,headers:c}=r(n)),l=await fetch(`/api/chat`,{method:`POST`,headers:c,body:i})}if(!l.ok){let e=await l.json().catch(()=>({}));throw Error(e.error||`Something went wrong`)}let u=``,d=``,p=()=>{d&&(u+=d,d=``,s(e=>{let t=[...e];return t[h]={...t[h],content:u},t}))},m=setInterval(p,60);try{for await(let e of o(l)){if(e.error)throw Object.assign(Error(e.error),{fromSSE:!0});e.text&&(d+=e.text),e.type===`component`&&(p(),s(t=>{let n=[...t],r={...n[h]},i=[...r.components||[]];return i.push({name:e.name,data:e.data,submitted:!1}),n[h]={...r,components:i},n}))}}finally{clearInterval(m),p()}}catch(e){if(L&&console.error(`[chat] sendMessage failed:`,e),r===`auth-required`)return;let t=e instanceof Error?e.message:String(e),i=e instanceof Error&&e.fromSSE;s(e=>{let r=[...e];return r[h]={...r[h],content:i?t:t===`session`?n??`I'm having trouble connecting right now. Try refreshing the page.`:`Sorry, I hit a snag. Try again in a moment.`},r})}finally{d(!1),r===`chat`&&i.current?.focus()}},[e,t,n,r,i]);return{messages:a,setMessages:s,isStreaming:c,sendMessage:p,sendGreeting:f,handleComponentSubmit:(0,N.useCallback)((e,t,n)=>{s(n=>{let r=[...n],i={...r[e]},a=[...i.components||[]];return a[t]={...a[t],submitted:!0},r[e]={...i,components:a},r}),p(n,{hidden:!0})},[p])}}function ne({sessionKeyRef:e,groupSlug:t,isStreaming:n,setMessages:r}){let i=(0,N.useRef)(new Date().toISOString()),a=(0,N.useRef)(!0);(0,N.useEffect)(()=>{let e=()=>{a.current=document.visibilityState===`visible`};return document.addEventListener(`visibilitychange`,e),()=>document.removeEventListener(`visibilitychange`,e)},[]);let o=(0,N.useCallback)(async()=>{let n=e.current;if(!(!n||!t))try{let e=await fetch(`/api/group/messages?session_key=${encodeURIComponent(n)}&since=${encodeURIComponent(i.current)}`);if(!e.ok)return;let t=await e.json();if(!t.messages||t.messages.length===0)return;let a=t.messages[t.messages.length-1];a.timestamp&&(i.current=new Date(a.timestamp).toISOString()),r(e=>{let n=new Set(e.filter(e=>e.messageId).map(e=>e.messageId)),r=t.messages.filter(e=>!n.has(e.messageId)).map(e=>({role:e.role,content:e.content,timestamp:e.timestamp,senderName:e.senderName??void 0,senderVisitorId:e.senderVisitorId??void 0,messageId:e.messageId}));if(r.length===0)return e;let i=[...e,...r];return i.sort((e,t)=>e.timestamp-t.timestamp),i})}catch(e){console.error(`[group-poll] failed: ${e instanceof Error?e.message:String(e)}`)}},[e,t,r]);(0,N.useEffect)(()=>{if(!t)return;let e,r=!1;function i(){if(r)return;let t=a.current?3e3:3e4;e=setTimeout(async()=>{n||await o(),i()},t)}return i(),()=>{r=!0,clearTimeout(e)}},[t,n,o])}var U=i();function W({value:e,onChange:t,onComplete:n,disabled:r}){let i=(0,N.useRef)([]);function a(n,r){r.key===`Backspace`?(r.preventDefault(),e[n]?t(e.slice(0,n)+e.slice(n+1)):n>0&&(t(e.slice(0,n-1)+e.slice(n)),i.current[n-1]?.focus())):r.key===`ArrowLeft`&&n>0?i.current[n-1]?.focus():r.key===`ArrowRight`&&n<5&&i.current[n+1]?.focus()}function o(r,a){let o=a.nativeEvent.data;if(!o||!/^\d$/.test(o))return;let s=e.split(``);for(s[r]=o;s.length<r;)s.push(``);let c=s.join(``).replace(/\D/g,``).slice(0,6);t(c),c.length===6?n(c):r<5&&i.current[r+1]?.focus()}function s(e){e.preventDefault();let r=e.clipboardData.getData(`text`).replace(/\D/g,``).slice(0,6);r&&(t(r),r.length===6?n(r):i.current[r.length]?.focus())}return(0,U.jsx)(`div`,{className:`gate-otp-field`,children:Array.from({length:6}).map((t,n)=>(0,U.jsx)(`input`,{ref:e=>{i.current[n]=e},type:`text`,inputMode:`numeric`,className:`pin-box${e[n]?` pin-box-filled`:``}`,value:e[n]||``,onKeyDown:e=>a(n,e),onInput:e=>o(n,e),onPaste:s,onFocus:e=>e.target.select(),autoFocus:n===0,autoComplete:`off`,maxLength:1,disabled:r,"aria-label":`Code digit ${n+1}`},n))})}function re({gateState:e,setGateState:t,grantInfo:n,setGrantInfo:r,gateSessionKeyRef:i,resolvedSlugRef:a,onAuthenticated:o}){let[s,c]=(0,N.useState)(``),[l,u]=(0,N.useState)(``),[d,f]=(0,N.useState)(``),[p,m]=(0,N.useState)(!1),[h,g]=(0,N.useState)(``),[v,y]=(0,N.useState)(``),[b,x]=(0,N.useState)(!1),[S,C]=(0,N.useState)(null),[w,T]=(0,N.useState)(null),[E,D]=(0,N.useState)(!1);async function O(e){if(e.preventDefault(),!E){C(null),D(!0);try{let e=a.current,n=await fetch(`/api/access/login`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({contact:s,password:l,agentSlug:e})}),i=await n.json();n.ok?o(i.session_key):n.status===401?i.error?.includes(`setup not complete`)||i.error?.includes(`invitation link`)?(t(`private-agent`),C(null)):C(i.error||`Invalid credentials`):n.status===403?(t(`access-expired`),i.expiresAt&&r(e=>e?{...e,expiresAt:i.expiresAt}:{displayName:null,contactValue:s,contactMethod:`email`,expiresAt:i.expiresAt,status:`expired`})):n.status===429?C(i.error||`Too many attempts. Please try again later.`):C(i.error||`Something went wrong`)}catch{C(`Unable to connect. Please try again.`)}finally{D(!1)}}}async function k(e){if(!E){C(null),D(!0);try{let n=a.current,o=await fetch(`/api/access/verify-otp`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({phone:v,code:e,agentSlug:n})}),s=await o.json();o.ok?(i.current=s.session_key,r(s.grant),t(`set-password`)):o.status===429?t(`otp-failed`):(C(s.error||`Invalid code`),g(``))}catch{C(`Unable to connect. Please try again.`)}finally{D(!1)}}}async function j(e){if(e.preventDefault(),!E){if(C(null),l!==d){C(`Passwords do not match`);return}if(!V(l).every(e=>e.met)){C(`Password does not meet requirements`);return}D(!0);try{let e=i.current;if(!e){C(`Session expired. Please use your invitation link again.`);return}let t=await fetch(`/api/access/create-credentials`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({session_key:e,password:l})}),n=await t.json();t.ok?o(n.session_key):t.status===400&&n.requirements?C(n.requirements.filter(e=>!e.met).map(e=>e.label).join(`, `)):C(n.error||`Something went wrong`)}catch{C(`Unable to connect. Please try again.`)}finally{D(!1)}}}async function M(e){if(e.preventDefault(),!(E||!s)){C(null),D(!0);try{let e=a.current,t=await fetch(`/api/access/forgot-password`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({contact:s,agentSlug:e})});t.status===429?C((await t.json()).error||`Too many attempts. Please try again later.`):(T(`If an account exists, a reset link has been sent.`),setTimeout(()=>{x(!1),T(null)},4e3))}catch{C(`Unable to connect. Please try again.`)}finally{D(!1)}}}switch(e){case`set-password`:{let e=V(l),t=e.every(e=>e.met),r=l===d,i=t&&r&&d.length>0;return(0,U.jsx)(`div`,{className:`gate-wrap`,children:(0,U.jsxs)(`div`,{className:`gate-card`,children:[n?.expiresAt&&(0,U.jsxs)(`div`,{className:`gate-expiry-badge`,children:[`Access until `,new Date(n.expiresAt).toLocaleDateString(`en-GB`,{day:`numeric`,month:`short`,year:`numeric`})]}),(0,U.jsxs)(`h2`,{className:`gate-title`,children:[`Welcome, `,n?.displayName||`there`]}),(0,U.jsx)(`p`,{className:`gate-subtitle`,children:`Create a password to access this agent in the future.`}),(0,U.jsxs)(`form`,{className:`gate-form`,onSubmit:j,children:[(0,U.jsxs)(`div`,{className:`gate-field`,children:[(0,U.jsx)(`label`,{htmlFor:`gate-display-name`,children:`Display name`}),(0,U.jsx)(`input`,{id:`gate-display-name`,type:`text`,value:n?.displayName||``,disabled:!0})]}),(0,U.jsxs)(`div`,{className:`gate-field gate-pw-row`,children:[(0,U.jsx)(`label`,{htmlFor:`gate-pw`,children:`Password`}),(0,U.jsx)(`div`,{className:`gate-pw-toggle`,children:(0,U.jsx)(A,{checked:p,onChange:m,label:`Show`})}),(0,U.jsx)(`input`,{id:`gate-pw`,type:p?`text`:`password`,value:l,onChange:e=>u(e.target.value),placeholder:`Your password`,autoFocus:!0,autoComplete:`new-password`}),l.length>0&&(0,U.jsx)(`div`,{className:`gate-strength`,children:e.map(e=>(0,U.jsxs)(`span`,{className:`gate-strength-item${e.met?` met`:``}`,children:[e.met?`✓`:`○`,` `,e.label]},e.key))})]}),(0,U.jsxs)(`div`,{className:`gate-field`,children:[(0,U.jsx)(`label`,{htmlFor:`gate-pw-confirm`,children:`Confirm password`}),(0,U.jsx)(`input`,{id:`gate-pw-confirm`,type:p?`text`:`password`,value:d,onChange:e=>f(e.target.value),placeholder:`Confirm your password`,autoComplete:`new-password`}),d.length>0&&!r&&(0,U.jsx)(`div`,{className:`gate-error`,children:`Passwords do not match`})]}),S&&(0,U.jsx)(`div`,{className:`gate-error`,children:S}),(0,U.jsx)(`div`,{className:`gate-submit`,children:(0,U.jsx)(_,{variant:`primary`,type:`submit`,fullWidth:!0,disabled:!i||E,loading:E,children:`Create account & enter chat`})})]})]})})}case`enter-otp`:return(0,U.jsx)(`div`,{className:`gate-wrap`,children:(0,U.jsxs)(`div`,{className:`gate-card`,children:[(0,U.jsx)(`h2`,{className:`gate-title`,children:`Enter your code`}),(0,U.jsxs)(`p`,{className:`gate-subtitle`,children:[`Enter the 6-digit code sent to `,H(v,`phone`)]}),(0,U.jsx)(W,{value:h,onChange:g,onComplete:k,disabled:E}),S&&(0,U.jsx)(`div`,{className:`gate-error`,children:S}),E&&(0,U.jsxs)(`div`,{className:`gate-loading`,children:[(0,U.jsx)(`span`,{className:`spinner`}),`Verifying...`]}),(0,U.jsx)(`div`,{className:`gate-resend`,children:(0,U.jsx)(`button`,{type:`button`,className:`gate-link`,onClick:()=>{t(`sign-in`),g(``),C(null)},children:`Use a different number`})})]})});case`sign-in`:case`private-agent`:return(0,U.jsx)(`div`,{className:`gate-wrap`,children:(0,U.jsxs)(`div`,{className:`gate-card`,children:[(0,U.jsx)(`h2`,{className:`gate-title`,children:e===`private-agent`?`Private Agent`:`Welcome back`}),(0,U.jsx)(`p`,{className:`gate-subtitle`,children:e===`private-agent`?`This agent is invitation-only. If you have an account, sign in below.`:`Sign in to continue your conversation.`}),b?(0,U.jsxs)(`form`,{className:`gate-form`,onSubmit:M,children:[(0,U.jsxs)(`div`,{className:`gate-field`,children:[(0,U.jsx)(`label`,{htmlFor:`gate-forgot-contact`,children:`Email or phone`}),(0,U.jsx)(`input`,{id:`gate-forgot-contact`,type:`text`,value:s,onChange:e=>c(e.target.value),placeholder:`you@example.com`,autoFocus:!0,autoComplete:`email`})]}),S&&(0,U.jsx)(`div`,{className:`gate-error`,children:S}),w&&(0,U.jsx)(`div`,{className:`gate-success`,children:w}),(0,U.jsx)(`div`,{className:`gate-submit`,children:(0,U.jsx)(_,{variant:`primary`,type:`submit`,fullWidth:!0,disabled:!s.trim()||E,loading:E,children:`Send reset link`})}),(0,U.jsx)(`div`,{className:`gate-actions`,children:(0,U.jsx)(`button`,{type:`button`,className:`gate-link`,onClick:()=>{x(!1),C(null),T(null)},children:`Back to sign in`})})]}):(0,U.jsxs)(`form`,{className:`gate-form`,onSubmit:O,children:[(0,U.jsxs)(`div`,{className:`gate-field`,children:[(0,U.jsx)(`label`,{htmlFor:`gate-contact`,children:`Email or phone`}),(0,U.jsx)(`input`,{id:`gate-contact`,type:`text`,value:s,onChange:e=>c(e.target.value),placeholder:`you@example.com`,autoFocus:!0,autoComplete:`email`})]}),(0,U.jsxs)(`div`,{className:`gate-field gate-pw-row`,children:[(0,U.jsx)(`label`,{htmlFor:`gate-login-pw`,children:`Password`}),(0,U.jsx)(`div`,{className:`gate-pw-toggle`,children:(0,U.jsx)(A,{checked:p,onChange:m,label:`Show`})}),(0,U.jsx)(`input`,{id:`gate-login-pw`,type:p?`text`:`password`,value:l,onChange:e=>u(e.target.value),placeholder:`Your password`,autoComplete:`current-password`})]}),S&&(0,U.jsx)(`div`,{className:`gate-error`,children:S}),(0,U.jsx)(`div`,{className:`gate-submit`,children:(0,U.jsx)(_,{variant:`primary`,type:`submit`,fullWidth:!0,disabled:!s.trim()||!l||E,loading:E,children:`Sign in`})}),(0,U.jsx)(`div`,{className:`gate-actions`,children:(0,U.jsx)(`button`,{type:`button`,className:`gate-link`,onClick:()=>{x(!0),C(null)},children:`Forgot password?`})})]}),e===`private-agent`&&(0,U.jsx)(`p`,{className:`gate-hint`,children:`No account? You need an invitation from the agent owner.`})]})});case`access-expired`:return(0,U.jsx)(`div`,{className:`gate-wrap`,children:(0,U.jsxs)(`div`,{className:`gate-card`,children:[(0,U.jsx)(`div`,{className:`gate-icon`,children:`⏰`}),(0,U.jsx)(`h2`,{className:`gate-title`,children:`Access expired`}),(0,U.jsxs)(`p`,{className:`gate-subtitle`,children:[n?.expiresAt?`Your access expired on ${new Date(n.expiresAt).toLocaleDateString(`en-GB`,{day:`numeric`,month:`long`,year:`numeric`})}.`:`Your access to this agent has expired.`,` `,`Contact the agent owner to renew.`]}),(0,U.jsx)(`div`,{className:`gate-actions`,children:(0,U.jsx)(_,{variant:`secondary`,onClick:()=>{t(`sign-in`),C(null),u(``)},children:`Sign in with a different account`})})]})});case`link-expired`:return(0,U.jsx)(`div`,{className:`gate-wrap`,children:(0,U.jsxs)(`div`,{className:`gate-card`,children:[(0,U.jsx)(`div`,{className:`gate-icon`,children:`⚠️`}),(0,U.jsx)(`h2`,{className:`gate-title`,children:`Link expired`}),(0,U.jsx)(`p`,{className:`gate-subtitle`,children:`This invitation link has expired or has already been used. If you already set your password, sign in below. Otherwise, ask the agent owner to send a new invitation.`}),(0,U.jsx)(`div`,{className:`gate-actions`,children:(0,U.jsx)(_,{variant:`primary`,onClick:()=>{t(`sign-in`),C(null)},children:`Go to sign in`})})]})});case`otp-failed`:return(0,U.jsx)(`div`,{className:`gate-wrap`,children:(0,U.jsxs)(`div`,{className:`gate-card`,children:[(0,U.jsx)(`div`,{className:`gate-icon`,children:`⛔`}),(0,U.jsx)(`h2`,{className:`gate-title`,children:`Too many attempts`}),(0,U.jsx)(`p`,{className:`gate-subtitle`,children:`You've entered the wrong code too many times. Ask the agent owner to send a new code.`}),(0,U.jsx)(`div`,{className:`gate-actions`,children:(0,U.jsx)(_,{variant:`secondary`,onClick:()=>{t(`sign-in`),C(null),g(``)},children:`Back to sign in`})})]})})}}var G={"single-select":g,"multi-select":C,"action-buttons":x};function K({name:e,data:t,onSubmit:n,submitted:r}){let i=G[e];return i?(0,U.jsx)(i,{data:t,onSubmit:n,submitted:r}):(console.warn(`[PublicComponentRenderer] Unknown component: "${e}". Registered: ${Object.keys(G).join(`, `)}`),(0,U.jsx)(`div`,{className:`component-card component-card--error`,children:(0,U.jsxs)(`p`,{style:{fontFamily:`var(--font-body)`,fontSize:12,color:`var(--text-secondary)`},children:[`Component “`,e,`” is not available. This may require a platform update.`]})}))}function ie({messages:e,isStreaming:t,sessionError:n,selectionMode:r,selectedItems:i,toggleSelectItem:a,onComponentSubmit:o,remainingSuggestions:s,onSuggestionClick:c,isAtBottom:f,setIsAtBottom:p,isGroup:h,sessionId:g}){let[v,y]=(0,N.useState)(new Set),[b,x]=(0,N.useState)(new Set),S=(0,N.useRef)(null),C=(0,N.useRef)(null),w=(0,N.useRef)(null),E=(0,N.useRef)(!1),D=(0,N.useRef)(!1),O=(0,N.useRef)(!1),k=(0,N.useRef)(0),j=(0,N.useRef)(!1),M=(0,N.useRef)(new Map);j.current=t,(0,N.useEffect)(()=>{t&&!E.current&&f&&(D.current=!0)},[t,f]),(0,N.useEffect)(()=>{if(!D.current&&!f){E.current=t,O.current=!1;return}if(E.current&&!t&&w.current&&C.current){let e=w.current;if(e.offsetHeight>C.current.clientHeight){D.current=!1,O.current=!0,e.scrollIntoView({behavior:`smooth`,block:`start`}),E.current=t;return}}(f||D.current)&&!O.current&&S.current?.scrollIntoView({behavior:`smooth`}),E.current&&!t&&(D.current=!1),E.current=t},[e,f,t]),(0,N.useEffect)(()=>{let e=C.current;if(!e)return;let t=()=>{let t=e.scrollTop;j.current&&t<k.current&&(D.current=!1),k.current=t,p(e.scrollHeight-e.scrollTop-e.clientHeight<80)};return e.addEventListener(`scroll`,t,{passive:!0}),t(),()=>e.removeEventListener(`scroll`,t)},[p]),(0,N.useEffect)(()=>{function e(){x(e=>{let t=new Set;return M.current.forEach((n,r)=>{n.isConnected&&(v.has(r)?e.has(r)&&t.add(r):n.scrollHeight>n.clientHeight+2&&t.add(r))}),e.size===t.size&&[...e].every(e=>t.has(e))?e:t})}e();let t=new ResizeObserver(e);return M.current.forEach(e=>t.observe(e)),()=>t.disconnect()},[e.length,v]);let P=[...e].sort((e,t)=>e.timestamp-t.timestamp);(0,N.useEffect)(()=>{r&&console.log(`[selection] entered`,{conversationId:g??`unknown`,order:P.map(e=>({ts:e.timestamp,role:e.role,id:e.messageId}))})},[r]);let F=P.reduce((e,t,n)=>t.role===`maxy`&&!t.hidden?n:e,-1);return(0,U.jsxs)(`div`,{className:`chat-messages-wrap`,children:[r&&(0,U.jsx)(`div`,{className:`selection-overlay-band`}),(0,U.jsxs)(`div`,{className:`chat-messages`,ref:C,children:[n&&e.length===0&&(0,U.jsx)(`div`,{className:`session-error`,children:(0,U.jsx)(`p`,{children:n})}),P.map((e,n)=>{if(e.hidden)return null;let s=t&&n===P.length-1,c=`${e.timestamp}_${e.role}`,f=i.has(c);return(0,U.jsxs)(`div`,{ref:n===F?w:void 0,className:`message ${e.role}${f?` selected`:``}`,children:[r&&!s&&(0,U.jsx)(`div`,{className:`message-select-check`,children:(0,U.jsx)(A,{checked:f,onChange:()=>a(c)})}),(0,U.jsxs)(`div`,{className:`bubble`,children:[h&&e.role===`visitor`&&e.senderName&&(0,U.jsx)(`span`,{className:`bubble-sender`,children:e.senderName}),e.role===`visitor`&&e.content?(()=>{let t=v.has(n),r=b.has(n);return(0,U.jsxs)(U.Fragment,{children:[(0,U.jsx)(`div`,{ref:e=>{e?M.current.set(n,e):M.current.delete(n)},className:t?`bubble-content`:`bubble-content clamped${r?` overflowing`:``}`,children:e.content}),r&&(0,U.jsx)(`div`,{className:`bubble-expand`,children:(0,U.jsx)(_,{variant:`icon`,className:t?`expanded`:``,onClick:()=>y(e=>{let t=new Set(e);return t.has(n)?t.delete(n):t.add(n),t}),"aria-label":t?`Collapse message`:`Expand message`,children:(0,U.jsx)(l,{size:14})})})]})})():e.content?(0,U.jsx)(T,{content:e.content,timestamp:m(e.timestamp)}):(0,U.jsxs)(`span`,{className:`typing-indicator`,children:[(0,U.jsx)(`span`,{className:`typing-dot`}),(0,U.jsx)(`span`,{className:`typing-dot`}),(0,U.jsx)(`span`,{className:`typing-dot`})]}),e.attachments&&e.attachments.length>0&&(0,U.jsx)(`div`,{className:`message-attachments`,children:e.attachments.map((e,t)=>(0,U.jsxs)(`span`,{className:`message-attachment-chip no-action`,children:[e.mimeType===`application/pdf`?(0,U.jsx)(u,{size:12}):e.mimeType.startsWith(`text/`)?(0,U.jsx)(d,{size:12}):null,(0,U.jsx)(`span`,{children:e.filename})]},t))}),e.role===`visitor`&&(0,U.jsx)(`span`,{className:`message-timestamp`,children:m(e.timestamp)})]}),e.role===`maxy`&&e.components&&e.components.length>0&&(0,U.jsx)(`div`,{className:`public-components`,children:e.components.map((e,t)=>(0,U.jsx)(`div`,{className:`public-component-enter`,children:(0,U.jsx)(K,{name:e.name,data:e.data,onSubmit:e=>o(n,t,e),submitted:e.submitted})},t))})]},n)}),!t&&s.length>0&&(0,U.jsx)(`div`,{className:`chat-suggestions`,children:s.map(e=>(0,U.jsx)(_,{variant:`suggestion`,onClick:()=>c(e),children:e},e))}),(0,U.jsx)(`div`,{ref:S})]}),!f&&(0,U.jsx)(`button`,{className:`scroll-to-bottom`,onClick:()=>S.current?.scrollIntoView({behavior:`smooth`}),"aria-label":`Scroll to bottom`,children:(0,U.jsx)(l,{size:18})})]})}function ae({selectedItems:e,messages:t,isStreaming:n,copySelected:r,selectAll:i,exitSelection:a,showCopyToast:o}){let[l,u]=(0,N.useState)(!1),d=(0,N.useRef)(null),f=(0,N.useRef)(!1);function p(e){let n=e.lastIndexOf(`_`),r=e.slice(0,n),i=e.slice(n+1),a=parseInt(r,10);return t.find(e=>e.timestamp===a&&e.role===i)?.content??``}function m(e,t){return parseInt(e.slice(0,e.lastIndexOf(`_`)),10)-parseInt(t.slice(0,t.lastIndexOf(`_`)),10)}function h(){let e=[...t].sort((e,t)=>e.timestamp-t.timestamp),n=[];for(let t of e){if(t.hidden||!t.content)continue;let e=t.role===`visitor`?`Visitor`:`Maxy`;n.push(`${e}:\n${t.content}`)}return n.join(`
2
2
 
3
3
  ---
4
4
 
@@ -1,5 +1,5 @@
1
1
  const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["assets/mermaid.core-CCUSwZB_.js","assets/preload-helper-qlgyTAkD.js","assets/dist-CrzV1W3-.js","assets/chunk-DD-I1_y5.js","assets/chunk-U2HBQHQK-BZnA7c4T.js","assets/src-Bo15iQ7w.js","assets/chunk-5PVQY5BW-RhIfPCRB.js","assets/chunk-ICPOFSXX-Di63NBur.js","assets/_baseFor-Dn4GSmI6.js","assets/isEmpty-h-wRi_o9.js","assets/chunk-336JU56O-BpATJiGl.js","assets/chunk-5FUZZQ4R-BoTfWHuW.js","assets/chunk-X2U36JSP-DpQ2OA_c.js","assets/chunk-ZZ45TVLE-DBSm41oP.js","assets/rough.esm-NLRoWnq-.js","assets/chunk-ENJZ2VHE-CNHjq5xK.js","assets/line-DlKKhwkO.js","assets/path-7vUsG-o2.js","assets/array-DJN9YAVf.js","assets/chunk-BSJP7CBP-CTsYuARh.js","assets/chunk-426QAEUC-Wz6Bpsil.js","assets/chunk-XPW4576I-BccP1mlQ.js"])))=>i.map(i=>d[i]);
2
- import{o as e,r as t,t as n}from"./chunk-DD-I1_y5.js";import{n as r,o as i,r as a,t as o}from"./jsx-runtime-DnY0498s.js";import{t as s}from"./Checkbox-BySsatDO.js";import{t as c}from"./preload-helper-qlgyTAkD.js";var l=a(`check`,[[`path`,{d:`M20 6 9 17l-5-5`,key:`1gmf2c`}]]),u=a(`chevron-down`,[[`path`,{d:`m6 9 6 6 6-6`,key:`qrunsl`}]]),d=a(`copy`,[[`rect`,{width:`14`,height:`14`,x:`8`,y:`8`,rx:`2`,ry:`2`,key:`17jyea`}],[`path`,{d:`M4 16c-1.1 0-2-.9-2-2V4c0-1.1.9-2 2-2h10c1.1 0 2 .9 2 2`,key:`zix9uf`}]]),f=a(`file-code`,[[`path`,{d:`M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z`,key:`1oefj6`}],[`path`,{d:`M14 2v5a1 1 0 0 0 1 1h5`,key:`wfsgrz`}],[`path`,{d:`M10 12.5 8 15l2 2.5`,key:`1tg20x`}],[`path`,{d:`m14 12.5 2 2.5-2 2.5`,key:`yinavb`}]]),p=a(`file-text`,[[`path`,{d:`M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z`,key:`1oefj6`}],[`path`,{d:`M14 2v5a1 1 0 0 0 1 1h5`,key:`wfsgrz`}],[`path`,{d:`M10 9H8`,key:`b1mrlr`}],[`path`,{d:`M16 13H8`,key:`t4e002`}],[`path`,{d:`M16 17H8`,key:`z1uh3a`}]]),m=a(`globe`,[[`circle`,{cx:`12`,cy:`12`,r:`10`,key:`1mglay`}],[`path`,{d:`M12 2a14.5 14.5 0 0 0 0 20 14.5 14.5 0 0 0 0-20`,key:`13o1zl`}],[`path`,{d:`M2 12h20`,key:`9i4pu4`}]]),h=a(`mic`,[[`path`,{d:`M12 19v3`,key:`npa21l`}],[`path`,{d:`M19 10v2a7 7 0 0 1-14 0v-2`,key:`1vc78b`}],[`rect`,{x:`9`,y:`2`,width:`6`,height:`13`,rx:`3`,key:`s6n7sd`}]]),g=a(`paperclip`,[[`path`,{d:`m16 6-8.414 8.586a2 2 0 0 0 2.829 2.829l8.414-8.586a4 4 0 1 0-5.657-5.657l-8.379 8.551a6 6 0 1 0 8.485 8.485l8.379-8.551`,key:`1miecu`}]]),_=a(`pause`,[[`rect`,{x:`14`,y:`3`,width:`5`,height:`18`,rx:`1`,key:`kaeet6`}],[`rect`,{x:`5`,y:`3`,width:`5`,height:`18`,rx:`1`,key:`1wsw3u`}]]),v=a(`play`,[[`path`,{d:`M5 5a2 2 0 0 1 3.008-1.728l11.997 6.998a2 2 0 0 1 .003 3.458l-12 7A2 2 0 0 1 5 19z`,key:`10ikf1`}]]),y=a(`square-check-big`,[[`path`,{d:`M21 10.656V19a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h12.344`,key:`2acyp4`}],[`path`,{d:`m9 11 3 3L22 4`,key:`1pflzl`}]]),b=a(`triangle-alert`,[[`path`,{d:`m21.73 18-8-14a2 2 0 0 0-3.48 0l-8 14A2 2 0 0 0 4 21h16a2 2 0 0 0 1.73-3`,key:`wmoenq`}],[`path`,{d:`M12 9v4`,key:`juzpu7`}],[`path`,{d:`M12 17h.01`,key:`p32p05`}]]),x=e(i(),1);async function S(e){if(navigator.clipboard)try{return await navigator.clipboard.writeText(e),!0}catch{}try{let t=document.createElement(`textarea`);t.value=e,t.style.position=`fixed`,t.style.opacity=`0`,document.body.appendChild(t),t.select();let n=document.execCommand(`copy`);return document.body.removeChild(t),n}catch{return!1}}function C(){let[e,t]=(0,x.useState)(!1),[n,r]=(0,x.useState)(new Set),[i,a]=(0,x.useState)(!1),o=(0,x.useRef)(null);(0,x.useEffect)(()=>()=>{o.current!==null&&clearTimeout(o.current)},[]);let s=(0,x.useCallback)(()=>{t(!0),r(new Set)},[]),c=(0,x.useCallback)(()=>{t(!1),r(new Set)},[]),l=(0,x.useCallback)(e=>{r(t=>{let n=new Set(t);return n.has(e)?n.delete(e):n.add(e),n})},[]),u=(0,x.useCallback)(e=>{e.length!==0&&r(t=>e.every(e=>t.has(e))?new Set:new Set(e))},[]),d=(0,x.useCallback)(()=>{o.current!==null&&(clearTimeout(o.current),o.current=null)},[]),f=(0,x.useCallback)(e=>({onPointerDown:()=>{o.current=setTimeout(()=>{t(!0),r(new Set([e])),o.current=null},500)},onPointerUp:d,onPointerLeave:d,onPointerCancel:d}),[d]),p=(0,x.useCallback)(e=>{a(e?`copied`:`failed`),setTimeout(()=>a(!1),1500)},[]);return{selectionMode:e,selectedItems:n,copyToast:i,longPressProps:f,enterSelection:s,exitSelection:c,toggleSelectItem:l,selectAll:u,copySelected:(0,x.useCallback)(async(e,t)=>{let r=[...n].sort(t).map(t=>e(t)).filter(Boolean);r.length>0&&p(await S(r.join(`
2
+ import{o as e,r as t,t as n}from"./chunk-DD-I1_y5.js";import{n as r,o as i,r as a,t as o}from"./jsx-runtime-nxP_2eNo.js";import{t as s}from"./Checkbox-B9hff9s8.js";import{t as c}from"./preload-helper-qlgyTAkD.js";var l=a(`check`,[[`path`,{d:`M20 6 9 17l-5-5`,key:`1gmf2c`}]]),u=a(`chevron-down`,[[`path`,{d:`m6 9 6 6 6-6`,key:`qrunsl`}]]),d=a(`copy`,[[`rect`,{width:`14`,height:`14`,x:`8`,y:`8`,rx:`2`,ry:`2`,key:`17jyea`}],[`path`,{d:`M4 16c-1.1 0-2-.9-2-2V4c0-1.1.9-2 2-2h10c1.1 0 2 .9 2 2`,key:`zix9uf`}]]),f=a(`file-code`,[[`path`,{d:`M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z`,key:`1oefj6`}],[`path`,{d:`M14 2v5a1 1 0 0 0 1 1h5`,key:`wfsgrz`}],[`path`,{d:`M10 12.5 8 15l2 2.5`,key:`1tg20x`}],[`path`,{d:`m14 12.5 2 2.5-2 2.5`,key:`yinavb`}]]),p=a(`file-text`,[[`path`,{d:`M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z`,key:`1oefj6`}],[`path`,{d:`M14 2v5a1 1 0 0 0 1 1h5`,key:`wfsgrz`}],[`path`,{d:`M10 9H8`,key:`b1mrlr`}],[`path`,{d:`M16 13H8`,key:`t4e002`}],[`path`,{d:`M16 17H8`,key:`z1uh3a`}]]),m=a(`globe`,[[`circle`,{cx:`12`,cy:`12`,r:`10`,key:`1mglay`}],[`path`,{d:`M12 2a14.5 14.5 0 0 0 0 20 14.5 14.5 0 0 0 0-20`,key:`13o1zl`}],[`path`,{d:`M2 12h20`,key:`9i4pu4`}]]),h=a(`mic`,[[`path`,{d:`M12 19v3`,key:`npa21l`}],[`path`,{d:`M19 10v2a7 7 0 0 1-14 0v-2`,key:`1vc78b`}],[`rect`,{x:`9`,y:`2`,width:`6`,height:`13`,rx:`3`,key:`s6n7sd`}]]),g=a(`paperclip`,[[`path`,{d:`m16 6-8.414 8.586a2 2 0 0 0 2.829 2.829l8.414-8.586a4 4 0 1 0-5.657-5.657l-8.379 8.551a6 6 0 1 0 8.485 8.485l8.379-8.551`,key:`1miecu`}]]),_=a(`pause`,[[`rect`,{x:`14`,y:`3`,width:`5`,height:`18`,rx:`1`,key:`kaeet6`}],[`rect`,{x:`5`,y:`3`,width:`5`,height:`18`,rx:`1`,key:`1wsw3u`}]]),v=a(`play`,[[`path`,{d:`M5 5a2 2 0 0 1 3.008-1.728l11.997 6.998a2 2 0 0 1 .003 3.458l-12 7A2 2 0 0 1 5 19z`,key:`10ikf1`}]]),y=a(`square-check-big`,[[`path`,{d:`M21 10.656V19a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h12.344`,key:`2acyp4`}],[`path`,{d:`m9 11 3 3L22 4`,key:`1pflzl`}]]),b=a(`triangle-alert`,[[`path`,{d:`m21.73 18-8-14a2 2 0 0 0-3.48 0l-8 14A2 2 0 0 0 4 21h16a2 2 0 0 0 1.73-3`,key:`wmoenq`}],[`path`,{d:`M12 9v4`,key:`juzpu7`}],[`path`,{d:`M12 17h.01`,key:`p32p05`}]]),x=e(i(),1);async function S(e){if(navigator.clipboard)try{return await navigator.clipboard.writeText(e),!0}catch{}try{let t=document.createElement(`textarea`);t.value=e,t.style.position=`fixed`,t.style.opacity=`0`,document.body.appendChild(t),t.select();let n=document.execCommand(`copy`);return document.body.removeChild(t),n}catch{return!1}}function C(){let[e,t]=(0,x.useState)(!1),[n,r]=(0,x.useState)(new Set),[i,a]=(0,x.useState)(!1),o=(0,x.useRef)(null);(0,x.useEffect)(()=>()=>{o.current!==null&&clearTimeout(o.current)},[]);let s=(0,x.useCallback)(()=>{t(!0),r(new Set)},[]),c=(0,x.useCallback)(()=>{t(!1),r(new Set)},[]),l=(0,x.useCallback)(e=>{r(t=>{let n=new Set(t);return n.has(e)?n.delete(e):n.add(e),n})},[]),u=(0,x.useCallback)(e=>{e.length!==0&&r(t=>e.every(e=>t.has(e))?new Set:new Set(e))},[]),d=(0,x.useCallback)(()=>{o.current!==null&&(clearTimeout(o.current),o.current=null)},[]),f=(0,x.useCallback)(e=>({onPointerDown:()=>{o.current=setTimeout(()=>{t(!0),r(new Set([e])),o.current=null},500)},onPointerUp:d,onPointerLeave:d,onPointerCancel:d}),[d]),p=(0,x.useCallback)(e=>{a(e?`copied`:`failed`),setTimeout(()=>a(!1),1500)},[]);return{selectionMode:e,selectedItems:n,copyToast:i,longPressProps:f,enterSelection:s,exitSelection:c,toggleSelectItem:l,selectAll:u,copySelected:(0,x.useCallback)(async(e,t)=>{let r=[...n].sort(t).map(t=>e(t)).filter(Boolean);r.length>0&&p(await S(r.join(`
3
3
 
4
4
  ---
5
5
 
@@ -5,12 +5,12 @@
5
5
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
6
6
  <title>Data — Maxy</title>
7
7
  <link rel="icon" href="/favicon.ico">
8
- <script type="module" crossorigin src="/assets/data-BuuqlV4L.js"></script>
8
+ <script type="module" crossorigin src="/assets/data-Da6iYRW1.js"></script>
9
9
  <link rel="modulepreload" crossorigin href="/assets/chunk-DD-I1_y5.js">
10
- <link rel="modulepreload" crossorigin href="/assets/jsx-runtime-DnY0498s.js">
11
- <link rel="modulepreload" crossorigin href="/assets/useAdminFetch-B3MO55eB.js">
12
- <link rel="modulepreload" crossorigin href="/assets/page-Ddc_nKh8.js">
13
- <link rel="stylesheet" crossorigin href="/assets/jsx-runtime-O5ef8xK8.css">
10
+ <link rel="modulepreload" crossorigin href="/assets/jsx-runtime-nxP_2eNo.js">
11
+ <link rel="modulepreload" crossorigin href="/assets/graph-labels-ChinGFwI.js">
12
+ <link rel="modulepreload" crossorigin href="/assets/page-OVrxtgOZ.js">
13
+ <link rel="stylesheet" crossorigin href="/assets/jsx-runtime-CVA1ZrPS.css">
14
14
  </head>
15
15
  <body>
16
16
  <div id="root"></div>
@@ -5,13 +5,13 @@
5
5
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
6
6
  <title>Graph — Maxy</title>
7
7
  <link rel="icon" href="/favicon.ico">
8
- <script type="module" crossorigin src="/assets/graph-CtVITeok.js"></script>
8
+ <script type="module" crossorigin src="/assets/graph-BHq-JYwV.js"></script>
9
9
  <link rel="modulepreload" crossorigin href="/assets/chunk-DD-I1_y5.js">
10
- <link rel="modulepreload" crossorigin href="/assets/jsx-runtime-DnY0498s.js">
11
- <link rel="modulepreload" crossorigin href="/assets/Checkbox-BySsatDO.js">
12
- <link rel="modulepreload" crossorigin href="/assets/useAdminFetch-B3MO55eB.js">
13
- <link rel="modulepreload" crossorigin href="/assets/page-IQBQoOdT.js">
14
- <link rel="stylesheet" crossorigin href="/assets/jsx-runtime-O5ef8xK8.css">
10
+ <link rel="modulepreload" crossorigin href="/assets/jsx-runtime-nxP_2eNo.js">
11
+ <link rel="modulepreload" crossorigin href="/assets/Checkbox-B9hff9s8.js">
12
+ <link rel="modulepreload" crossorigin href="/assets/graph-labels-ChinGFwI.js">
13
+ <link rel="modulepreload" crossorigin href="/assets/page-DqPf65sS.js">
14
+ <link rel="stylesheet" crossorigin href="/assets/jsx-runtime-CVA1ZrPS.css">
15
15
  </head>
16
16
  <body>
17
17
  <div id="root"></div>
@@ -5,16 +5,16 @@
5
5
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
6
6
  <title>Real Agent</title>
7
7
  <link rel="icon" href="/favicon.ico">
8
- <script type="module" crossorigin src="/assets/admin-CLp1xGlJ.js"></script>
8
+ <script type="module" crossorigin src="/assets/admin-Cpi6L_g7.js"></script>
9
9
  <link rel="modulepreload" crossorigin href="/assets/chunk-DD-I1_y5.js">
10
- <link rel="modulepreload" crossorigin href="/assets/jsx-runtime-DnY0498s.js">
10
+ <link rel="modulepreload" crossorigin href="/assets/jsx-runtime-nxP_2eNo.js">
11
11
  <link rel="modulepreload" crossorigin href="/assets/preload-helper-qlgyTAkD.js">
12
- <link rel="modulepreload" crossorigin href="/assets/Checkbox-BySsatDO.js">
13
- <link rel="modulepreload" crossorigin href="/assets/useVoiceRecorder-CR8gcELb.js">
14
- <link rel="modulepreload" crossorigin href="/assets/useAdminFetch-B3MO55eB.js">
15
- <link rel="modulepreload" crossorigin href="/assets/page-Ddc_nKh8.js">
16
- <link rel="modulepreload" crossorigin href="/assets/page-IQBQoOdT.js">
17
- <link rel="stylesheet" crossorigin href="/assets/jsx-runtime-O5ef8xK8.css">
12
+ <link rel="modulepreload" crossorigin href="/assets/Checkbox-B9hff9s8.js">
13
+ <link rel="modulepreload" crossorigin href="/assets/useVoiceRecorder-DyVx7e7a.js">
14
+ <link rel="modulepreload" crossorigin href="/assets/graph-labels-ChinGFwI.js">
15
+ <link rel="modulepreload" crossorigin href="/assets/page-OVrxtgOZ.js">
16
+ <link rel="modulepreload" crossorigin href="/assets/page-DqPf65sS.js">
17
+ <link rel="stylesheet" crossorigin href="/assets/jsx-runtime-CVA1ZrPS.css">
18
18
  <link rel="stylesheet" crossorigin href="/assets/admin-CWMpccrR.css">
19
19
  <link rel="stylesheet" href="/brand-defaults.css">
20
20
  </head>
@@ -5,13 +5,13 @@
5
5
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
6
6
  <title>Real Agent</title>
7
7
  <link rel="icon" href="/favicon.ico">
8
- <script type="module" crossorigin src="/assets/public-DRrf63wm.js"></script>
8
+ <script type="module" crossorigin src="/assets/public-CJN5KAiK.js"></script>
9
9
  <link rel="modulepreload" crossorigin href="/assets/chunk-DD-I1_y5.js">
10
- <link rel="modulepreload" crossorigin href="/assets/jsx-runtime-DnY0498s.js">
10
+ <link rel="modulepreload" crossorigin href="/assets/jsx-runtime-nxP_2eNo.js">
11
11
  <link rel="modulepreload" crossorigin href="/assets/preload-helper-qlgyTAkD.js">
12
- <link rel="modulepreload" crossorigin href="/assets/Checkbox-BySsatDO.js">
13
- <link rel="modulepreload" crossorigin href="/assets/useVoiceRecorder-CR8gcELb.js">
14
- <link rel="stylesheet" crossorigin href="/assets/jsx-runtime-O5ef8xK8.css">
12
+ <link rel="modulepreload" crossorigin href="/assets/Checkbox-B9hff9s8.js">
13
+ <link rel="modulepreload" crossorigin href="/assets/useVoiceRecorder-DyVx7e7a.js">
14
+ <link rel="stylesheet" crossorigin href="/assets/jsx-runtime-CVA1ZrPS.css">
15
15
  <link rel="stylesheet" href="/brand-defaults.css">
16
16
  </head>
17
17
  <body>
@@ -83,7 +83,7 @@ import {
83
83
  vncLog,
84
84
  waitForExit,
85
85
  writeChromiumWrapper
86
- } from "./chunk-ND23BDBM.js";
86
+ } from "./chunk-DHSBEMWW.js";
87
87
  import {
88
88
  agentLogStream,
89
89
  clearSessionHistory,
@@ -10437,7 +10437,16 @@ async function hybrid(session, embed2, params) {
10437
10437
  bm25Score: normalised[i] ?? 0,
10438
10438
  related: []
10439
10439
  }));
10440
- return { mode: "bm25", results: results2, embedError: msg, expandMs: 0 };
10440
+ return {
10441
+ mode: "bm25",
10442
+ results: results2,
10443
+ embedError: msg,
10444
+ expandMs: 0,
10445
+ rawMerged: results2.length,
10446
+ suppressed: 0,
10447
+ bm25Bypass: 0,
10448
+ threshold: null
10449
+ };
10441
10450
  }
10442
10451
  const labelToIndex = await discoverIndexes(session);
10443
10452
  const keywordFilter = buildKeywordFilter(keywords, keywordMatch);
@@ -10448,7 +10457,15 @@ async function hybrid(session, embed2, params) {
10448
10457
  if (labels && labels.length > 0) {
10449
10458
  indexesToQuery = labels.map((l) => labelToIndex.get(l)).filter((idx) => idx !== void 0);
10450
10459
  if (indexesToQuery.length === 0) {
10451
- return { mode: "hybrid", results: [], expandMs: 0 };
10460
+ return {
10461
+ mode: "hybrid",
10462
+ results: [],
10463
+ expandMs: 0,
10464
+ rawMerged: 0,
10465
+ suppressed: 0,
10466
+ bm25Bypass: 0,
10467
+ threshold: params.vectorThreshold ?? null
10468
+ };
10452
10469
  }
10453
10470
  } else {
10454
10471
  indexesToQuery = [...new Set(labelToIndex.values())];
@@ -10493,7 +10510,8 @@ async function hybrid(session, embed2, params) {
10493
10510
  labels: record.get("nodeLabels"),
10494
10511
  properties: plainProperties(node.properties),
10495
10512
  vectorScore: score,
10496
- bm25Score: 0
10513
+ bm25Score: 0,
10514
+ bm25Hit: false
10497
10515
  });
10498
10516
  }
10499
10517
  }
@@ -10543,6 +10561,7 @@ async function hybrid(session, embed2, params) {
10543
10561
  const existing = scoreMap.get(nodeId);
10544
10562
  if (existing) {
10545
10563
  existing.bm25Score = Math.max(existing.bm25Score, 1);
10564
+ existing.bm25Hit = true;
10546
10565
  } else {
10547
10566
  const node = record.get("node");
10548
10567
  scoreMap.set(nodeId, {
@@ -10550,15 +10569,33 @@ async function hybrid(session, embed2, params) {
10550
10569
  labels: record.get("nodeLabels"),
10551
10570
  properties: plainProperties(node.properties),
10552
10571
  vectorScore: 0,
10553
- bm25Score: 1
10572
+ bm25Score: 1,
10573
+ bm25Hit: true
10554
10574
  });
10555
10575
  }
10556
10576
  }
10557
10577
  }
10558
- const merged = [...scoreMap.values()].map((node) => ({
10578
+ const allMerged = [...scoreMap.values()].map((node) => ({
10559
10579
  ...node,
10560
10580
  combinedScore: VECTOR_WEIGHT * node.vectorScore + BM25_WEIGHT * node.bm25Score
10561
- })).sort((a, b) => b.combinedScore - a.combinedScore).slice(0, limit);
10581
+ }));
10582
+ const rawMerged = allMerged.length;
10583
+ const threshold = params.vectorThreshold;
10584
+ let kept = allMerged;
10585
+ let bm25Bypass = 0;
10586
+ if (threshold !== void 0) {
10587
+ kept = [];
10588
+ for (const node of allMerged) {
10589
+ if (node.vectorScore >= threshold) {
10590
+ kept.push(node);
10591
+ } else if (node.bm25Hit) {
10592
+ kept.push(node);
10593
+ bm25Bypass++;
10594
+ }
10595
+ }
10596
+ }
10597
+ const suppressed = rawMerged - kept.length;
10598
+ const merged = kept.sort((a, b) => b.combinedScore - a.combinedScore).slice(0, limit);
10562
10599
  const results = merged.map((node) => ({
10563
10600
  nodeId: node.nodeId,
10564
10601
  labels: node.labels,
@@ -10614,19 +10651,29 @@ async function hybrid(session, embed2, params) {
10614
10651
  }
10615
10652
  }
10616
10653
  }
10617
- return { mode: "hybrid", results, expandMs };
10654
+ return {
10655
+ mode: "hybrid",
10656
+ results,
10657
+ expandMs,
10658
+ rawMerged,
10659
+ suppressed,
10660
+ bm25Bypass,
10661
+ threshold: threshold ?? null
10662
+ };
10618
10663
  }
10619
10664
  function mergeBm25Hit(map, hit, normalisedScore) {
10620
10665
  const existing = map.get(hit.nodeId);
10621
10666
  if (existing) {
10622
10667
  existing.bm25Score = Math.max(existing.bm25Score, normalisedScore);
10668
+ existing.bm25Hit = true;
10623
10669
  } else {
10624
10670
  map.set(hit.nodeId, {
10625
10671
  nodeId: hit.nodeId,
10626
10672
  labels: hit.labels,
10627
10673
  properties: hit.properties,
10628
10674
  vectorScore: 0,
10629
- bm25Score: normalisedScore
10675
+ bm25Score: normalisedScore,
10676
+ bm25Hit: true
10630
10677
  });
10631
10678
  }
10632
10679
  }
@@ -10646,6 +10693,7 @@ function plainProperties(properties) {
10646
10693
  // server/routes/admin/graph-search.ts
10647
10694
  var DEFAULT_LIMIT = 20;
10648
10695
  var MAX_LIMIT = 2e3;
10696
+ var DEFAULT_VECTOR_THRESHOLD = 0.72;
10649
10697
  var MESSAGE_FAMILY_LABELS = ["Message", "UserMessage", "AssistantMessage", "WhatsAppMessage"];
10650
10698
  var CONVERSATION_PARENT_LABELS = /* @__PURE__ */ new Set(["AdminConversation", "PublicConversation"]);
10651
10699
  var app24 = new Hono();
@@ -10654,6 +10702,7 @@ app24.get("/", requireAdminSession, async (c) => {
10654
10702
  const q = (c.req.query("q") ?? "").trim();
10655
10703
  const rawLimit = c.req.query("limit");
10656
10704
  const rawLabels = c.req.query("labels");
10705
+ const rawThreshold = c.req.query("threshold");
10657
10706
  const accountId = getAccountIdForSession(sessionKey);
10658
10707
  if (!accountId) {
10659
10708
  console.error(`[graph-search] auth-rejected endpoint="/api/admin/graph-search" reason="no account for session"`);
@@ -10667,6 +10716,13 @@ app24.get("/", requireAdminSession, async (c) => {
10667
10716
  }
10668
10717
  const parsedLimit = rawLimit ? parseInt(rawLimit, 10) : DEFAULT_LIMIT;
10669
10718
  const limit = Number.isFinite(parsedLimit) && parsedLimit > 0 ? Math.min(parsedLimit, MAX_LIMIT) : DEFAULT_LIMIT;
10719
+ let vectorThreshold = DEFAULT_VECTOR_THRESHOLD;
10720
+ if (rawThreshold !== void 0) {
10721
+ const parsed = Number(rawThreshold);
10722
+ if (Number.isFinite(parsed) && parsed >= 0 && parsed <= 1) {
10723
+ vectorThreshold = parsed;
10724
+ }
10725
+ }
10670
10726
  const wantsBodyFulltext = !wildcard && labels.some((l) => CONVERSATION_PARENT_LABELS.has(l));
10671
10727
  const forwardedLabels = wildcard ? void 0 : wantsBodyFulltext ? Array.from(/* @__PURE__ */ new Set([...labels, ...MESSAGE_FAMILY_LABELS])) : labels;
10672
10728
  const started = Date.now();
@@ -10677,7 +10733,8 @@ app24.get("/", requireAdminSession, async (c) => {
10677
10733
  accountId,
10678
10734
  labels: forwardedLabels,
10679
10735
  limit,
10680
- degradeOnEmbedFailure: true
10736
+ degradeOnEmbedFailure: true,
10737
+ vectorThreshold
10681
10738
  });
10682
10739
  const total = Date.now() - started;
10683
10740
  if (res.embedError) {
@@ -10744,11 +10801,23 @@ app24.get("/", requireAdminSession, async (c) => {
10744
10801
  console.error(
10745
10802
  `[graph-search] query="${q}" labels=${labelsToken} expanded=${expandedFlag} limit=${limit} mode=${res.mode} raw-results=${res.results.length} resolved-results=${resolvedResults.length} expand-ms=${res.expandMs} total-ms=${total}`
10746
10803
  );
10804
+ console.error(
10805
+ `[graph-search] threshold=${res.threshold ?? "off"} raw-merged=${res.rawMerged} suppressed=${res.suppressed} rendered=${res.results.length} bm25-bypass=${res.bm25Bypass}`
10806
+ );
10747
10807
  return c.json({
10748
10808
  results: resolvedResults,
10749
10809
  mode: res.mode,
10750
10810
  embedError: res.embedError,
10751
- elapsedMs: total
10811
+ elapsedMs: total,
10812
+ // Task 967 — UI surfaces ("N suppressed — show all" affordance on
10813
+ // /graph and /data). `suppressed` is the count of rows the lib
10814
+ // dropped via the vector-cosine floor; `threshold` echoes the value
10815
+ // applied (null when no filter ran). `bm25Bypass` is operator-only
10816
+ // diagnostic — kept on the wire so the /data score panel can flag
10817
+ // "kept by literal-match carve-out" rows in a future iteration.
10818
+ suppressed: res.suppressed,
10819
+ bm25Bypass: res.bm25Bypass,
10820
+ threshold: res.threshold
10752
10821
  });
10753
10822
  } catch (err) {
10754
10823
  const elapsed = Date.now() - started;
@@ -1 +0,0 @@
1
- import{i as e,t}from"./jsx-runtime-DnY0498s.js";import{t as n}from"./page-Ddc_nKh8.js";import"./useAdminFetch-B3MO55eB.js";var r=e(),i=t();(0,r.createRoot)(document.getElementById(`root`)).render((0,i.jsx)(n,{}));
@@ -1 +0,0 @@
1
- import{i as e,t}from"./jsx-runtime-DnY0498s.js";import{n}from"./page-IQBQoOdT.js";import"./useAdminFetch-B3MO55eB.js";import"./Checkbox-BySsatDO.js";var r=e(),i=t();(0,r.createRoot)(document.getElementById(`root`)).render((0,i.jsx)(n,{}));
@@ -1 +0,0 @@
1
- import{o as e}from"./chunk-DD-I1_y5.js";import{n as t,o as n,r,t as i}from"./jsx-runtime-DnY0498s.js";import{a,c as o,i as s,l as c,o as l,r as u,s as d,t as f}from"./useAdminFetch-B3MO55eB.js";var p=r(`circle-arrow-up`,[[`circle`,{cx:`12`,cy:`12`,r:`10`,key:`1mglay`}],[`path`,{d:`m16 12-4-4-4 4`,key:`177agl`}],[`path`,{d:`M12 16V8`,key:`1sbj14`}]]),m=r(`download`,[[`path`,{d:`M12 15V3`,key:`m9g1x1`}],[`path`,{d:`M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4`,key:`ih7n3h`}],[`path`,{d:`m7 10 5 5 5-5`,key:`brsn70`}]]),h=r(`file`,[[`path`,{d:`M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z`,key:`1oefj6`}],[`path`,{d:`M14 2v5a1 1 0 0 0 1 1h5`,key:`wfsgrz`}]]),g=r(`folder`,[[`path`,{d:`M20 20a2 2 0 0 0 2-2V8a2 2 0 0 0-2-2h-7.9a2 2 0 0 1-1.69-.9L9.6 3.9A2 2 0 0 0 7.93 3H4a2 2 0 0 0-2 2v13a2 2 0 0 0 2 2Z`,key:`1kt360`}]]),_=r(`upload`,[[`path`,{d:`M12 3v12`,key:`1x0j5s`}],[`path`,{d:`m17 8-5-5-5 5`,key:`7q97r8`}],[`path`,{d:`M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4`,key:`ih7n3h`}]]);function v(e){let t=new Set,n=[];for(let r of e){let e=r.labels[0];e&&(t.has(e)||(t.add(e),n.push(e)))}return n}var y=80;function b(e){let t=x(e.properties)??e.labels[0]??``,n=t.length>y?t.slice(0,y):t;return`/graph?${new URLSearchParams({nodeId:e.nodeId,label:n}).toString()}`}function x(e){for(let t of[`title`,`name`,`summary`,`headline`]){let n=e[t];if(typeof n==`string`&&n.length>0)return n}return null}var S=e(n(),1),C=i();function w(){let[e,t]=(0,S.useState)(null),[n,r]=(0,S.useState)(!1);return(0,S.useEffect)(()=>{let e=!1,n=null;try{n=sessionStorage.getItem(`maxy-admin-session-key`)}catch{}if(!n){t(null),r(!0);return}return fetch(`/api/admin/session?session_key=${encodeURIComponent(n)}`).then(i=>{if(!e){if(i.status===401){try{sessionStorage.removeItem(`maxy-admin-session-key`)}catch{}window.location.href=`/`;return}t(n),r(!0)}}).catch(()=>{e||(t(n),r(!0))}),()=>{e=!0}},[]),n?e?(0,C.jsxs)(`div`,{className:`data-page`,children:[(0,C.jsxs)(`header`,{className:`data-header`,children:[(0,C.jsxs)(`h1`,{className:`data-title`,children:[(0,C.jsx)(c,{size:18}),` Data`]}),(0,C.jsx)(E,{})]}),(0,C.jsx)(T,{sessionKey:e})]}):(0,C.jsxs)(`div`,{className:`data-page`,children:[(0,C.jsx)(`header`,{className:`data-header`,children:(0,C.jsxs)(`h1`,{className:`data-title`,children:[(0,C.jsx)(c,{size:18}),` Data`]})}),(0,C.jsxs)(`div`,{className:`data-empty`,children:[(0,C.jsx)(`p`,{children:`You are not signed in.`}),(0,C.jsxs)(`p`,{children:[`Open the `,(0,C.jsx)(`a`,{href:`/`,className:`data-link`,children:`main admin page`}),` and log in, then return here.`]})]})]}):(0,C.jsx)(`div`,{className:`data-page`,children:(0,C.jsxs)(`div`,{className:`data-loading`,children:[(0,C.jsx)(d,{size:18,className:`spin`}),` Loading…`]})})}function T({sessionKey:e,onOpenInGraph:t}){let{adminFetch:n,sessionKey:r,sessionRefetchNonce:i,banner:a,reloadToLogin:o}=f({initialSessionKey:e,surface:`data`});return(0,C.jsxs)(`main`,{className:`data-main`,children:[a&&(0,C.jsx)(`button`,{type:`button`,className:`data-session-banner`,onClick:o,"data-reason":a.reason,children:a.message}),(0,C.jsx)(D,{adminFetch:n,sessionRefetchNonce:i,onOpenInGraph:t}),(0,C.jsx)(M,{adminFetch:n,sessionKey:r,sessionRefetchNonce:i})]})}function E(){let[e,t]=(0,S.useState)(!1),n=(0,S.useRef)(null);(0,S.useEffect)(()=>{if(!e)return;let r=e=>{n.current&&(n.current.contains(e.target)||t(!1))},i=e=>{e.key===`Escape`&&t(!1)};return document.addEventListener(`mousedown`,r),document.addEventListener(`keydown`,i),()=>{document.removeEventListener(`mousedown`,r),document.removeEventListener(`keydown`,i)}},[e]);let r=(0,S.useCallback)(()=>{try{sessionStorage.removeItem(`maxy-admin-session-key`)}catch{}window.location.href=`/`},[]);return(0,C.jsxs)(`div`,{className:`chat-burger-wrap`,ref:n,children:[(0,C.jsx)(`button`,{type:`button`,className:`chat-burger`,onClick:()=>t(e=>!e),"aria-label":`Menu`,"aria-haspopup":`true`,"aria-expanded":e,children:(0,C.jsx)(a,{size:20})}),e&&(0,C.jsxs)(`div`,{className:`chat-menu`,children:[(0,C.jsxs)(`button`,{type:`button`,className:`chat-menu-item`,onClick:()=>{t(!1),window.location.href=`/`},children:[(0,C.jsx)(o,{size:14}),` Chat`]}),(0,C.jsxs)(`button`,{type:`button`,className:`chat-menu-item`,onClick:()=>{t(!1),window.location.href=`/graph`},children:[(0,C.jsx)(u,{size:14}),` Graph`]}),(0,C.jsx)(`div`,{className:`chat-menu-divider`}),(0,C.jsxs)(`button`,{type:`button`,className:`chat-menu-item`,onClick:()=>{t(!1),r()},children:[(0,C.jsx)(l,{size:14}),` Log out`]})]})]})}function D({adminFetch:e,sessionRefetchNonce:t,onOpenInGraph:n}){let[r,i]=(0,S.useState)(``),[a,o]=(0,S.useState)(``),[c,l]=(0,S.useState)(null),[u,f]=(0,S.useState)(`hybrid`),[p,m]=(0,S.useState)(!1),[h,g]=(0,S.useState)(null),[_,y]=(0,S.useState)(null);(0,S.useEffect)(()=>{let e=r.trim();if(!e){o(``),l(null);return}let t=setTimeout(()=>o(e),300);return()=>clearTimeout(t)},[r]),(0,S.useEffect)(()=>{if(!a)return;let t=!1;return m(!0),g(null),e(`/api/admin/graph-search?q=${encodeURIComponent(a)}&labels=*&limit=20`).then(async e=>{let t=await e.json().catch(()=>({error:`HTTP ${e.status}`}));if(!e.ok)throw Error(t.error??`HTTP ${e.status}`);return t}).then(e=>{t||(l(e.results),f(e.mode??`hybrid`))}).catch(e=>{t||g(e instanceof Error?e.message:String(e))}).finally(()=>{t||m(!1)}),()=>{t=!0}},[a,e,t]);let x=(0,S.useMemo)(()=>c?v(c):[],[c]),w=_&&x.includes(_)?_:null,T=(0,S.useMemo)(()=>c?w?c.filter(e=>e.labels[0]===w):c:null,[c,w]),E=(0,S.useCallback)(e=>{if(n){n(e);return}window.location.href=b(e)},[n]);return(0,C.jsxs)(`section`,{className:`data-panel`,children:[(0,C.jsx)(`h2`,{className:`data-panel-title`,children:`Search`}),(0,C.jsx)(`p`,{className:`data-panel-subtitle`,children:`Find anything in your knowledge base by keyword.`}),(0,C.jsxs)(`div`,{className:`data-search-input`,children:[(0,C.jsx)(s,{size:14}),(0,C.jsx)(`input`,{type:`text`,placeholder:`Search your knowledge…`,value:r,onChange:e=>i(e.target.value),autoFocus:!0,autoComplete:`off`,spellCheck:!1}),p&&(0,C.jsx)(d,{size:14,className:`spin`})]}),h&&(0,C.jsxs)(`div`,{className:`data-error`,children:[`Search failed: `,h]}),x.length>0&&(0,C.jsx)(`div`,{className:`data-chip-row`,role:`group`,"aria-label":`Filter by type`,children:x.map(e=>{let t=e===w;return(0,C.jsx)(`button`,{type:`button`,className:`data-chip`,"data-active":t,onClick:()=>y(t?null:e),children:k([e])??e},e)})}),c&&!p&&(0,C.jsxs)(`div`,{className:`data-results-meta`,children:[T.length,` of `,c.length,` result`,c.length===1?``:`s`]}),c&&c.length===0&&!p&&(0,C.jsx)(`div`,{className:`data-empty-results`,children:`No matches. Try a different keyword.`}),T&&T.length>0&&(0,C.jsx)(`ul`,{className:`data-results`,children:T.map(e=>(0,C.jsx)(O,{hit:e,mode:u,onOpenInGraph:E},e.nodeId))})]})}function O({hit:e,mode:t,onOpenInGraph:n}){let r=A(e.properties),i=j(e.properties),a=k(e.labels),[o,s]=(0,S.useState)(!1),c=i&&i.length>280,l=o||!c?i:i?.slice(0,280)+`…`,u=t===`bm25`?`bm25 ${e.bm25Score.toFixed(2)}`:`vector ${e.vectorScore.toFixed(2)} · bm25 ${e.bm25Score.toFixed(2)} · combined ${e.score.toFixed(2)}`;return(0,C.jsxs)(`li`,{className:`data-result`,children:[(0,C.jsxs)(`div`,{role:`button`,tabIndex:0,className:`data-result-open`,onClick:()=>n(e),onKeyDown:t=>{(t.key===`Enter`||t.key===` `)&&(t.preventDefault(),n(e))},"aria-label":`Open ${r??a??`node`} in graph`,children:[a&&(0,C.jsx)(`div`,{className:`data-result-header`,children:(0,C.jsx)(`span`,{className:`data-result-labels`,children:a})}),r&&(0,C.jsx)(`div`,{className:`data-result-title`,children:r}),i&&(0,C.jsx)(`pre`,{className:`data-result-body`,children:l}),(0,C.jsx)(`div`,{className:`data-result-scores`,children:u})]}),c&&(0,C.jsx)(`button`,{type:`button`,className:`data-result-toggle`,onClick:()=>s(e=>!e),children:o?`Show less`:`Show more`})]})}function k(e){if(e.length===0)return null;let t=e[0].replace(/([a-z])([A-Z])/g,`$1 $2`);return t.charAt(0).toUpperCase()+t.slice(1).toLowerCase()}function A(e){for(let t of[`title`,`name`,`summary`,`headline`]){let n=e[t];if(typeof n==`string`&&n.length>0)return n.length>140?n.slice(0,140)+`…`:n}return null}function j(e){for(let t of[`content`,`summary`,`body`,`description`,`text`]){let n=e[t];if(typeof n==`string`&&n.length>0)return n}return null}function M({adminFetch:e,sessionKey:n,sessionRefetchNonce:r}){let[i,a]=(0,S.useState)(`.`),[o,s]=(0,S.useState)(null),[c,l]=(0,S.useState)([]),[u,f]=(0,S.useState)(!1),[v,y]=(0,S.useState)(null),[b,x]=(0,S.useState)(!1),[w,T]=(0,S.useState)(null),[E,D]=(0,S.useState)(null),[O,k]=(0,S.useState)(null),[A,j]=(0,S.useState)(null),M=(0,S.useRef)(null),[F,I]=(0,S.useState)(0);(0,S.useEffect)(()=>{let t=!1;return f(!0),y(null),e(`/api/admin/files?path=${encodeURIComponent(i===`.`?``:i)}`).then(async e=>{let t=await e.json().catch(()=>({error:`HTTP ${e.status}`}));if(!e.ok)throw Error(t.error??`HTTP ${e.status}`);return t}).then(e=>{t||(s(e.entries),l(e.displayPath??[]))}).catch(e=>{t||(s([]),l([]),y(e instanceof Error?e.message:String(e)))}).finally(()=>{t||f(!1)}),()=>{t=!0}},[e,i,F,r]);let L=(0,S.useCallback)(e=>{a(e)},[]),R=(0,S.useCallback)(()=>{if(i===`.`||i===``)return;let e=i.split(`/`).filter(Boolean);e.pop(),a(e.length===0?`.`:e.join(`/`))},[i]),z=(0,S.useCallback)(e=>{let t=i===`.`?e.name:`${i}/${e.name}`,r=`/api/admin/files/download?session_key=${encodeURIComponent(n)}&path=${encodeURIComponent(t)}`,a=document.createElement(`a`);a.href=r,a.rel=`noopener noreferrer`,document.body.appendChild(a),a.click(),a.remove()},[i,n]),B=(0,S.useCallback)(()=>{M.current?.click()},[]),V=(0,S.useCallback)(e=>{D(null),j(e.name)},[]),H=(0,S.useCallback)(()=>{j(null)},[]),U=(0,S.useCallback)(async t=>{let n=i===`.`?t.name:`${i}/${t.name}`;D(null),j(null),k(t.name);try{let t=await e(`/api/admin/files?path=${encodeURIComponent(n)}`,{method:`DELETE`}),r=await t.json().catch(()=>({error:`HTTP ${t.status}`}));if(!t.ok)throw Error(r.error??`HTTP ${t.status}`);I(e=>e+1)}catch(e){D(e instanceof Error?e.message:String(e))}finally{k(null)}},[i,e]);(0,S.useEffect)(()=>{if(!A)return;let e=e=>{e.key===`Escape`&&j(null)};return document.addEventListener(`keydown`,e),()=>document.removeEventListener(`keydown`,e)},[A]);let W=(0,S.useCallback)(async e=>{T(null),x(!0);try{let t=new FormData;t.append(`session_key`,n),t.append(`file`,e);let r=await fetch(`/api/admin/files/upload`,{method:`POST`,body:t}),i=await r.json().catch(()=>({error:`HTTP ${r.status}`}));if(!r.ok)throw Error(i.error??`HTTP ${r.status}`);a(i.path.split(`/`).slice(0,-1).join(`/`)||`.`),I(e=>e+1)}catch(e){T(e instanceof Error?e.message:String(e))}finally{x(!1),M.current&&(M.current.value=``)}},[n]);return(0,C.jsxs)(`section`,{className:`data-panel`,children:[(0,C.jsxs)(`div`,{className:`data-panel-heading`,children:[(0,C.jsx)(`h2`,{className:`data-panel-title`,children:`Files`}),(0,C.jsxs)(`div`,{className:`data-file-actions`,children:[(0,C.jsxs)(`button`,{type:`button`,className:`data-btn`,onClick:B,disabled:b,children:[b?(0,C.jsx)(d,{size:14,className:`spin`}):(0,C.jsx)(_,{size:14}),` Upload`]}),(0,C.jsx)(`input`,{type:`file`,ref:M,hidden:!0,onChange:e=>{let t=e.target.files?.[0];t&&W(t)}})]})]}),(0,C.jsxs)(`p`,{className:`data-panel-subtitle`,children:[`Everything under your install's `,(0,C.jsx)(`code`,{children:`data/`}),` directory. Click a folder to open it, a file to download it. Uploads go to `,(0,C.jsx)(`code`,{children:`data/uploads/`}),`.`]}),(0,C.jsxs)(`div`,{className:`data-breadcrumbs`,children:[(0,C.jsx)(`button`,{type:`button`,className:`data-crumb`,onClick:()=>L(`.`),children:`data`}),c.map((e,t)=>{let n=c.slice(0,t+1).map(e=>e.name).join(`/`);return(0,C.jsxs)(`span`,{className:`data-crumb-wrap`,children:[(0,C.jsx)(`span`,{className:`data-crumb-sep`,children:`/`}),(0,C.jsx)(`button`,{type:`button`,className:`data-crumb`,onClick:()=>L(n),title:e.name,children:e.displayName??e.name})]},n)}),i!==`.`&&(0,C.jsxs)(`button`,{type:`button`,className:`data-btn data-btn-ghost data-up-btn`,onClick:R,children:[(0,C.jsx)(p,{size:14}),` Up`]})]}),w&&(0,C.jsxs)(`div`,{className:`data-error`,children:[`Upload failed: `,w]}),E&&(0,C.jsxs)(`div`,{className:`data-error`,children:[`Delete failed: `,E]}),v&&(0,C.jsx)(`div`,{className:`data-error`,children:v}),u&&(0,C.jsxs)(`div`,{className:`data-loading`,children:[(0,C.jsx)(d,{size:14,className:`spin`}),` Loading…`]}),!u&&o&&o.length===0&&!v&&(0,C.jsx)(`div`,{className:`data-empty-results`,children:`Empty directory.`}),!u&&o&&o.length>0&&(0,C.jsx)(`ul`,{className:`data-entries`,children:o.map(e=>{let n=e.displayName??e.name,r=O===e.name,a=A===e.name;return(0,C.jsx)(`li`,{className:`data-entry`,children:e.kind===`directory`?(0,C.jsxs)(`button`,{type:`button`,className:`data-entry-btn`,onClick:()=>L(i===`.`?e.name:`${i}/${e.name}`),children:[(0,C.jsx)(g,{size:14}),(0,C.jsx)(`span`,{className:`data-entry-name`,title:e.name,children:n})]}):e.kind===`file`?(0,C.jsxs)(C.Fragment,{children:[(0,C.jsxs)(`button`,{type:`button`,className:`data-entry-btn`,onClick:()=>z(e),title:`Download ${n}`,disabled:a,children:[(0,C.jsx)(h,{size:14}),(0,C.jsx)(`span`,{className:`data-entry-name`,title:e.name,children:n}),(0,C.jsxs)(`span`,{className:`data-entry-meta`,title:`Modified ${P(e.modifiedAt)}`,children:[N(e.sizeBytes),` · `,P(e.modifiedAt)]}),(0,C.jsx)(m,{size:12,className:`data-entry-download-icon`})]}),a?(0,C.jsxs)(`div`,{className:`data-entry-confirm`,role:`group`,"aria-label":`Confirm delete ${n}`,children:[(0,C.jsx)(`span`,{className:`data-entry-confirm-label`,children:`Delete?`}),(0,C.jsx)(`button`,{type:`button`,className:`data-entry-confirm-yes`,onClick:()=>U(e),autoFocus:!0,children:`Yes`}),(0,C.jsx)(`button`,{type:`button`,className:`data-entry-confirm-no`,onClick:H,children:`No`})]}):(0,C.jsx)(`button`,{type:`button`,className:`data-entry-delete`,onClick:()=>V(e),disabled:r,title:`Delete ${n}`,"aria-label":`Delete ${n}`,children:r?(0,C.jsx)(d,{size:14,className:`spin`}):(0,C.jsx)(t,{size:14})})]}):(0,C.jsxs)(`div`,{className:`data-entry-btn data-entry-disabled`,children:[(0,C.jsx)(h,{size:14}),(0,C.jsx)(`span`,{className:`data-entry-name`,children:n}),(0,C.jsx)(`span`,{className:`data-entry-meta`,children:`special`})]})},e.name)})})]})}function N(e){return e==null?``:e<1024?`${e} B`:e<1024*1024?`${(e/1024).toFixed(1)} KB`:e<1024*1024*1024?`${(e/1024/1024).toFixed(1)} MB`:`${(e/1024/1024/1024).toFixed(1)} GB`}function P(e){try{let t=new Date(e);if(isNaN(t.getTime()))return`—`;let n=typeof navigator<`u`?navigator.languages&&navigator.languages.length>0?[...navigator.languages]:[navigator.language]:void 0;return t.toLocaleString(n,{dateStyle:`medium`,timeStyle:`short`})}catch{return`—`}}export{m as a,h as i,T as n,x as r,w as t};