@waniwani/sdk 0.10.10-beta.1 → 0.10.10

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.
@@ -292,11 +292,6 @@ type ResourceConfig = {
292
292
  baseUrl: string;
293
293
  /** Path to the HTML file (relative to baseUrl) */
294
294
  htmlPath: string;
295
- /**
296
- * Cache key appended to widget template filenames as `-<cacheKey>`.
297
- * Defaults to `VERCEL_DEPLOYMENT_ID` when available.
298
- */
299
- cacheKey?: string;
300
295
  /** Domain for OpenAI security context */
301
296
  widgetDomain: string;
302
297
  /** Whether widget prefers border (defaults to true) */
@@ -310,9 +305,9 @@ type RegisteredResource = {
310
305
  readonly id: string;
311
306
  readonly title: string;
312
307
  readonly description: string | undefined;
313
- /** OpenAI URI: ui://widgets/apps-sdk/{id}.html, with -{cacheKey} before .html when configured */
308
+ /** OpenAI URI: ui://widgets/apps-sdk/{id}.html */
314
309
  readonly openaiUri: string;
315
- /** MCP URI: ui://widgets/ext-apps/{id}.html, with -{cacheKey} before .html when configured */
310
+ /** MCP URI: ui://widgets/ext-apps/{id}.html */
316
311
  readonly mcpUri: string;
317
312
  readonly autoHeight: boolean | undefined;
318
313
  /** Register this resource on an McpServer */
@@ -1223,17 +1218,18 @@ type WithWaniwaniOptions = {
1223
1218
  */
1224
1219
  injectWidgetToken?: boolean;
1225
1220
  /**
1226
- * Strip `latitude`/`longitude` from known location `_meta` entries
1221
+ * List of field names to strip from known location `_meta` entries
1227
1222
  * (`openai/userLocation`, `waniwani/geoLocation`, `waniwani/userLocation`)
1228
1223
  * before events are sent to the WaniWani API. Applied to both the
1229
1224
  * request-level `_meta` and any `_meta` on the tool response.
1230
1225
  *
1231
- * Enable this when precise geo coordinates should not reach the tracking
1232
- * backend; city/region/country fields are preserved.
1226
+ * Pass e.g. `["latitude", "longitude"]` to drop coordinates only, or
1227
+ * `["latitude", "longitude", "city", "region"]` to keep just `country`.
1228
+ * Empty/omitted = no redaction.
1233
1229
  *
1234
- * @default false
1230
+ * @default []
1235
1231
  */
1236
- stripGeoCoordinates?: boolean;
1232
+ stripLocationFields?: readonly string[];
1237
1233
  /**
1238
1234
  * Replace `input.stateUpdates[field]` with `"REDACTED"` for any field
1239
1235
  * marked via `redacted()` on a flow state schema. When `false` (default),
package/dist/mcp/index.js CHANGED
@@ -1,7 +1,7 @@
1
- function ne(){return typeof window<"u"&&"openai"in window?"openai":"mcp-apps"}function mt(){return ne()==="openai"}function ht(){return ne()==="mcp-apps"}var b="__start__",_="__end__",ke=Symbol.for("waniwani.flow.interrupt"),Se=Symbol.for("waniwani.flow.widget");function ve(e,t){let n=t?.context,o=[];for(let[r,i]of Object.entries(e))if(typeof i=="object"&&i!==null&&"question"in i){let s=i;o.push({question:s.question,field:r,suggestions:s.suggestions,context:s.context,validate:s.validate})}return{__type:ke,questions:o,context:n}}function Re(e,t){return{__type:Se,tool:typeof e=="string"?e:e.id,...t}}function xe(e){return typeof e=="object"&&e!==null&&"__type"in e&&e.__type===ke}function Ee(e){return typeof e=="object"&&e!==null&&"__type"in e&&e.__type===Se}import{z as M}from"zod";var oe="waniwani/client";function B(e){if(typeof e=="object"&&e!==null)return e[oe]}function Ie(e,t,n){return{track(o){return e.track({...o,meta:{...t,...o.meta}})},identify(o,r){return e.identify(o,r,t)},kb:e.kb,_config:n}}function D(e,t){for(let n of t){let o=e[n];if(typeof o=="string"&&o.length>0)return o}}var yt=["waniwani/sessionId","openai/sessionId","openai/session","sessionId","conversationId","mcp-session-id"],Tt=["waniwani/requestId","openai/requestId","requestId","mcp/requestId"],kt=["waniwani/traceId","openai/traceId","traceId","mcp/traceId","openai/requestId","requestId"],St=["waniwani/userId","openai/userId","externalUserId","userId","actorId"],vt=["correlationId","openai/requestId"];function P(e){return e?D(e,yt):void 0}function Ce(e){return e?D(e,Tt):void 0}function _e(e){return e?D(e,kt):void 0}function be(e){return e?D(e,St):void 0}function Pe(e){return e?D(e,vt):void 0}var Rt=[{key:"waniwani/sessionId",source:"chatbar"},{key:"openai/sessionId",source:"chatgpt"},{key:"openai/session",source:"chatgpt"}];function O(e){if(!e)return;let t=e["waniwani/source"];if(typeof t=="string"&&t.length>0)return t;for(let{key:n,source:o}of Rt){let r=e[n];if(typeof r=="string"&&r.length>0)return o}}function Fe(e){let t=e._zod?.def;return t?.type==="object"&&t.shape?t.shape:null}function K(e,t){let n=t.split("."),o=e;for(let r of n){if(o==null||typeof o!="object")return;o=o[r]}return o}function xt(e,t,n){let o=t.split("."),r=o.pop();if(!r)return;let i=e;for(let s of o)(i[s]==null||typeof i[s]!="object"||Array.isArray(i[s]))&&(i[s]={}),i=i[s];i[r]=n}function We(e,t){let n=t.split("."),o=n.pop();if(!o)return;let r=e;for(let i of n){if(r==null||typeof r!="object")return;r=r[i]}r!=null&&typeof r=="object"&&delete r[o]}function re(e){let t={};for(let[n,o]of Object.entries(e))n.includes(".")?xt(t,n,o):t[n]=o;return t}function F(e,t){let n={...e};for(let[o,r]of Object.entries(t))r!==null&&typeof r=="object"&&!Array.isArray(r)&&n[o]!==null&&typeof n[o]=="object"&&!Array.isArray(n[o])?n[o]=F(n[o],r):n[o]=r;return n}function ie(e){return e!=null&&e!==""}async function W(e,t){return e.type==="direct"?e.to:e.condition(t)}function Ae(e,t,n,o){if(e.every(p=>ie(K(o,p.field))))return null;let r=e.filter(p=>!ie(K(o,p.field))),i=r.length===1,s=r[0];return{content:i&&s?{status:"interrupt",question:s.question,field:s.field,...s.suggestions?{suggestions:s.suggestions}:{},...s.context||t?{context:s.context??t}:{}}:{status:"interrupt",questions:r,...t?{context:t}:{}},flowTokenContent:{step:n,state:o,...i&&s?{field:s.field}:{}}}}async function q(e,t,n,o,r,i,s,a,p){let c=e,d={...t},u=50,f=0;for(;f++<u;){if(c===_)return{content:{status:"complete"},flowTokenContent:{state:d}};let h=n.get(c);if(!h)return{content:{status:"error",error:`Unknown node: "${c}"`}};a?.get(c)?.trackFunnel&&s&&s.track({event:"flow.node_reached",properties:{flowId:p??"unknown",nodeId:c}}).catch(()=>{});try{let l=await h({state:d,meta:i,interrupt:ve,showWidget:Re,waniwani:s});if(xe(l)){for(let k of l.questions)k.validate&&r.set(`${c}:${k.field}`,k.validate);let y=Ae(l.questions,l.context,c,d);if(y)return y;for(let k of l.questions){let E=r.get(`${c}:${k.field}`);if(E)try{let v=K(d,k.field),I=await E(v);I&&typeof I=="object"&&(d=F(d,I))}catch(v){let I=v instanceof Error?v.message:String(v);We(d,k.field);let ye=l.questions.map(U=>U.field===k.field?{...U,context:U.context?`ERROR: ${I}
1
+ function ne(){return typeof window<"u"&&"openai"in window?"openai":"mcp-apps"}function mt(){return ne()==="openai"}function ht(){return ne()==="mcp-apps"}var b="__start__",C="__end__",ke=Symbol.for("waniwani.flow.interrupt"),Se=Symbol.for("waniwani.flow.widget");function ve(e,t){let n=t?.context,r=[];for(let[o,i]of Object.entries(e))if(typeof i=="object"&&i!==null&&"question"in i){let s=i;r.push({question:s.question,field:o,suggestions:s.suggestions,context:s.context,validate:s.validate})}return{__type:ke,questions:r,context:n}}function Re(e,t){return{__type:Se,tool:typeof e=="string"?e:e.id,...t}}function xe(e){return typeof e=="object"&&e!==null&&"__type"in e&&e.__type===ke}function Ee(e){return typeof e=="object"&&e!==null&&"__type"in e&&e.__type===Se}import{z as M}from"zod";var re="waniwani/client";function B(e){if(typeof e=="object"&&e!==null)return e[re]}function Ie(e,t,n){return{track(r){return e.track({...r,meta:{...t,...r.meta}})},identify(r,o){return e.identify(r,o,t)},kb:e.kb,_config:n}}function D(e,t){for(let n of t){let r=e[n];if(typeof r=="string"&&r.length>0)return r}}var yt=["waniwani/sessionId","openai/sessionId","openai/session","sessionId","conversationId","mcp-session-id"],Tt=["waniwani/requestId","openai/requestId","requestId","mcp/requestId"],kt=["waniwani/traceId","openai/traceId","traceId","mcp/traceId","openai/requestId","requestId"],St=["waniwani/userId","openai/userId","externalUserId","userId","actorId"],vt=["correlationId","openai/requestId"];function P(e){return e?D(e,yt):void 0}function _e(e){return e?D(e,Tt):void 0}function Ce(e){return e?D(e,kt):void 0}function be(e){return e?D(e,St):void 0}function Pe(e){return e?D(e,vt):void 0}var Rt=[{key:"waniwani/sessionId",source:"chatbar"},{key:"openai/sessionId",source:"chatgpt"},{key:"openai/session",source:"chatgpt"}];function O(e){if(!e)return;let t=e["waniwani/source"];if(typeof t=="string"&&t.length>0)return t;for(let{key:n,source:r}of Rt){let o=e[n];if(typeof o=="string"&&o.length>0)return r}}function Fe(e){let t=e._zod?.def;return t?.type==="object"&&t.shape?t.shape:null}function K(e,t){let n=t.split("."),r=e;for(let o of n){if(r==null||typeof r!="object")return;r=r[o]}return r}function xt(e,t,n){let r=t.split("."),o=r.pop();if(!o)return;let i=e;for(let s of r)(i[s]==null||typeof i[s]!="object"||Array.isArray(i[s]))&&(i[s]={}),i=i[s];i[o]=n}function We(e,t){let n=t.split("."),r=n.pop();if(!r)return;let o=e;for(let i of n){if(o==null||typeof o!="object")return;o=o[i]}o!=null&&typeof o=="object"&&delete o[r]}function oe(e){let t={};for(let[n,r]of Object.entries(e))n.includes(".")?xt(t,n,r):t[n]=r;return t}function F(e,t){let n={...e};for(let[r,o]of Object.entries(t))o!==null&&typeof o=="object"&&!Array.isArray(o)&&n[r]!==null&&typeof n[r]=="object"&&!Array.isArray(n[r])?n[r]=F(n[r],o):n[r]=o;return n}function ie(e){return e!=null&&e!==""}async function W(e,t){return e.type==="direct"?e.to:e.condition(t)}function Ae(e,t,n,r){if(e.every(p=>ie(K(r,p.field))))return null;let o=e.filter(p=>!ie(K(r,p.field))),i=o.length===1,s=o[0];return{content:i&&s?{status:"interrupt",question:s.question,field:s.field,...s.suggestions?{suggestions:s.suggestions}:{},...s.context||t?{context:s.context??t}:{}}:{status:"interrupt",questions:o,...t?{context:t}:{}},flowTokenContent:{step:n,state:r,...i&&s?{field:s.field}:{}}}}async function q(e,t,n,r,o,i,s,a,p){let c=e,d={...t},u=50,l=0;for(;l++<u;){if(c===C)return{content:{status:"complete"},flowTokenContent:{state:d}};let h=n.get(c);if(!h)return{content:{status:"error",error:`Unknown node: "${c}"`}};a?.get(c)?.trackFunnel&&s&&s.track({event:"flow.node_reached",properties:{flowId:p??"unknown",nodeId:c}}).catch(()=>{});try{let f=await h({state:d,meta:i,interrupt:ve,showWidget:Re,waniwani:s});if(xe(f)){for(let k of f.questions)k.validate&&o.set(`${c}:${k.field}`,k.validate);let T=Ae(f.questions,f.context,c,d);if(T)return T;for(let k of f.questions){let E=o.get(`${c}:${k.field}`);if(E)try{let v=K(d,k.field),I=await E(v);I&&typeof I=="object"&&(d=F(d,I))}catch(v){let I=v instanceof Error?v.message:String(v);We(d,k.field);let ye=f.questions.map(U=>U.field===k.field?{...U,context:U.context?`ERROR: ${I}
2
2
 
3
- ${U.context}`:`ERROR: ${I}`}:U),Te=Ae(ye,l.context,c,d);if(Te)return Te;break}}let T=o.get(c);if(!T)return{content:{status:"error",error:`No outgoing edge from node "${c}"`}};c=await W(T,d);continue}if(Ee(l)){let y=l.field;if(y&&ie(K(d,y))){let T=o.get(c);if(!T)return{content:{status:"error",error:`No outgoing edge from node "${c}"`}};c=await W(T,d);continue}return{content:{status:"widget",tool:l.tool,data:l.data,description:l.description,interactive:l.interactive!==!1},flowTokenContent:{step:c,state:d,field:y,widgetId:l.tool}}}d=F(d,l);let x=o.get(c);if(!x)return{content:{status:"error",error:`No outgoing edge from node "${c}"`}};c=await W(x,d)}catch(R){return{content:{status:"error",error:R instanceof Error?R.message:String(R)},flowTokenContent:{step:c,state:d}}}}return{content:{status:"error",error:"Flow exceeded maximum iterations (possible infinite loop)"}}}function se(e){return typeof e=="object"&&e!==null&&e.__ww_enc===1&&typeof e.ct=="string"&&typeof e.iv=="string"}var Me=new Map;async function Ne(e){let t=Me.get(e);if(t)return t;let n=Buffer.from(e,"base64");if(n.length!==32)throw new Error("[WaniWani KV] WANIWANI_ENCRYPTION_KEY must be a base64-encoded 32-byte (256-bit) key.");let o=await globalThis.crypto.subtle.importKey("raw",n,{name:"AES-GCM"},!1,["encrypt","decrypt"]);return Me.set(e,o),o}async function Ue(e,t){let n=await Ne(t),o=globalThis.crypto.getRandomValues(new Uint8Array(12)),r=new TextEncoder().encode(JSON.stringify(e)),i=await globalThis.crypto.subtle.encrypt({name:"AES-GCM",iv:o},n,r);return{__ww_enc:1,ct:Buffer.from(i).toString("base64"),iv:Buffer.from(o).toString("base64")}}async function De(e,t){let n=await Ne(t),o=Buffer.from(e.ct,"base64"),r=Buffer.from(e.iv,"base64"),i;try{i=await globalThis.crypto.subtle.decrypt({name:"AES-GCM",iv:r},n,o)}catch{throw new Error("[WaniWani KV] Decryption failed. The encryption key may be incorrect or the data may be corrupted.")}return JSON.parse(new TextDecoder().decode(i))}var Et="@waniwani/sdk",It="https://app.waniwani.ai";function Ct(...e){process.env.WANIWANI_DEBUG&&console.log("[waniwani:kv]",...e)}var A=class{get baseUrl(){return(process.env.WANIWANI_API_URL??It).replace(/\/$/,"")}get apiKey(){return process.env.WANIWANI_API_KEY}get encryptionKey(){return process.env.WANIWANI_ENCRYPTION_KEY}async get(t){if(!this.apiKey)throw new Error("[WaniWani KV] No API key configured. Set WANIWANI_API_KEY env var.");let n=await this.request("/api/mcp/redis/get",{key:t});if(n==null)return null;if(se(n)){if(!this.encryptionKey)throw new Error("[WaniWani KV] Encrypted data found but WANIWANI_ENCRYPTION_KEY is not set.");return De(n,this.encryptionKey)}return n}async set(t,n){if(!this.apiKey)throw new Error("[WaniWani KV] No API key configured. Set WANIWANI_API_KEY env var.");let o=this.encryptionKey;Ct(`set "${t}" \u2014 encryption ${o?"enabled":"disabled (no WANIWANI_ENCRYPTION_KEY)"}`);let r=o?await Ue(n,o):n;await this.request("/api/mcp/redis/set",{key:t,value:r})}async delete(t){if(this.apiKey)try{await this.request("/api/mcp/redis/delete",{key:t})}catch(n){console.error("[WaniWani KV] delete failed for key:",t,n)}}async request(t,n){let o=`${this.baseUrl}${t}`,r=await fetch(o,{method:"POST",headers:{Authorization:`Bearer ${this.apiKey}`,"Content-Type":"application/json","X-WaniWani-SDK":Et},body:JSON.stringify(n)});if(!r.ok){let s=await r.text().catch(()=>"");throw new Error(s||`KV store API error: HTTP ${r.status}`)}return(await r.json()).data}};var j=class{store=new A;get(t){return this.store.get(t)}set(t,n){return this.store.set(t,n)}delete(t){return this.store.delete(t)}};function _t(e){let t=e.toString();return t.includes("showWidget")?"widget":t.includes("interrupt")?"interrupt":"action"}function bt(e,t){let n=e.toString(),o=[];for(let r of t){let i=`"${r}"`,s=`'${r}'`,a=`\`${r}\``;(n.includes(i)||n.includes(s)||n.includes(a))&&o.push(r)}return o}function Oe(e,t,n,o){let r=new Set([...t.keys(),_]),i=[];for(let[a,p]of t){let c=o.get(a);i.push({id:a,type:_t(p),trackFunnel:c?.trackFunnel??!1})}let s=[];for(let[a,p]of n)if(p.type==="direct")s.push({from:a,to:p.to,type:"direct"});else{let c=bt(p.condition,r);s.push({from:a,to:c,type:"conditional"})}return{flowId:e.id,title:e.title,nodes:i,edges:s}}function Ke(e){let t=e.description??"",n=e._zod?.def;if(n?.type==="enum"&&n.entries){let o=Object.keys(n.entries).map(r=>`"${r}"`).join(" | ");return t?`${o} \u2014 ${t}`:o}return t}function je(e){let t=["","## FLOW EXECUTION PROTOCOL","","This tool implements a multi-step conversational flow. Follow this protocol exactly:","",'1. Call with `action: "start"` to begin and include `intent`.'," `intent` must be a brief summary of the user's goal for this flow."," Do NOT invent missing intent."," Optionally include `context` \u2014 the situation or environment that led the user to start"," this flow (e.g. what page they are on, what they were doing, or what triggered the request)."," Only provide `context` when there is genuinely relevant situational information. Do NOT invent missing context."," If the user's message already contains answers to likely questions,"," extract them into `stateUpdates` as `{ field: value }` pairs."," The engine will auto-skip steps whose fields are already filled."," Only extract values the user explicitly stated \u2014 do NOT guess or invent values."];if(e.omitIntentPII&&t.push(" Do NOT include PII in `intent` or `context` \u2014 no names, emails, phones, addresses, IDs, ages, or birthdates.",' Summarize the goal abstractly (e.g. "user wants a quote", not "Jane Doe wants a quote").'),e.state){let n=[];for(let[o,r]of Object.entries(e.state)){let i=Fe(r);if(i){let s=r.description??"",a=Object.entries(i).map(([p,c])=>{let d=Ke(c);return d?`\`${o}.${p}\` (${d})`:`\`${o}.${p}\``}).join(", ");n.push(s?`\`${o}\` (${s}): ${a}`:`\`${o}\`: ${a}`)}else{let s=Ke(r);n.push(s?`\`${o}\` (${s})`:`\`${o}\``)}}t.push(` Known fields: ${n.join(", ")}.`)}return t.push(" For grouped fields (shown as `group.subfield`), use dot-notation keys in `stateUpdates`:",' e.g. `{ "driver.name": "John", "driver.license": "ABC123" }`.',"2. The response JSON `status` field tells you what to do next:",' - `"interrupt"`: Pause and ask the user. Two forms:'," a. Single question: `{ question, field, context? }` \u2014 ask `question`, store answer in `field`."," b. Multi-question: `{ questions: [{question, field}, ...], context? }` \u2014 ask ALL questions"," in one conversational message, collect all answers."," `context` (if present) is hidden AI instructions \u2014 use to shape your response, do NOT show verbatim."," Then call again with:",' `action: "continue"`,'," `stateUpdates` = answers keyed by their `field` names, plus any other fields the user mentioned.",' - `"widget"`: The flow wants to show a UI widget. Call the tool named in the `tool`'," field, passing the `data` object as the tool's input."," Check the `interactive` field in the response:"," \u2022 `interactive: true` \u2014 The widget requires user interaction. After calling the display tool,"," STOP and WAIT for the user to interact with the widget. Do NOT call this flow tool again"," until the user has responded. When they do, call with:",' `action: "continue"`,'," `stateUpdates` = `{ [field]: <user's selection> }` plus any other fields the user mentioned."," \u2022 `interactive: false` \u2014 The widget is display-only. Call the display tool, then immediately",' call THIS flow tool again with `action: "continue"`. Do NOT wait for user interaction.',' - `"complete"`: The flow is done. Present the result to the user.',' - `"error"`: Something went wrong. Show the `error` message.',"","3. Do NOT invent state values. Only use `stateUpdates` for information the user explicitly provided.","4. Include only the fields the user actually answered in `stateUpdates` \u2014 do NOT guess missing ones."," If the user did not answer all pending questions, the engine will re-prompt for the remaining ones."," If the user mentioned values for other known fields, include those too \u2014"," they will be applied immediately and those steps will be auto-skipped."),t.join(`
4
- `)}function ae(e){return!!e&&typeof e=="object"&&!Array.isArray(e)}function $e(e){let t=Be(e),n=ae(t.waniwani)?t.waniwani:{};return e.meta({...t,waniwani:{...n,redacted:!0}})}function Pt(e){let n=Be(e).waniwani;return ae(n)&&n.redacted===!0}function Be(e){let t=e.meta;if(typeof t!="function")return{};let n=t.call(e);return ae(n)?n:{}}function qe(e){if(!e)return[];let t=[];for(let[n,o]of Object.entries(e))Pt(o)&&t.push(n);return t}var L="waniwani/redactedStateUpdateFields";function Ft(e){let t=e.omitIntentPII?" Do not include PII (names, emails, phones, addresses, IDs, ages, birthdates) \u2014 summarize abstractly.":"";return{action:M.enum(["start","continue"]).describe('"start" to begin the flow, "continue" to resume after a pause (interrupt or widget)'),intent:M.string().optional().describe(`Required when action is "start". Provide a brief summary of the user's goal for this flow. Do not invent missing intent.${t}`),context:M.string().optional().describe(`Optional when action is "start". Describe the situation or environment that led the user to start this flow \u2014 e.g. what page they are on, what they were doing, or what triggered the request. Do not invent missing context.${t}`),stateUpdates:M.record(M.string(),M.unknown()).optional().describe("State field values to set before processing the next node. Use this to pass the user's answer (keyed by the field name from the response) and any other values the user mentioned.")}}function Le(e){let{config:t,nodes:n,edges:o}=e,r=Ft(t),i=Oe(t,n,o,e.nodeOptions),s=je(t),a=`${t.description}
5
- ${s}`,p=e.store??new j,c=new Map;async function d(g,R,l,x){if(g.action==="start"){let y=typeof g.intent=="string"?g.intent.trim():void 0;if(!y)return{content:{status:"error",error:`Missing required "intent" for action "start". Include a brief summary of the user's goal for this flow and any relevant prior context that led to triggering it, if available.`}};if(g.intent=y,typeof g.context=="string"){let v=g.context.trim();g.context=v||void 0}let T=o.get(b);if(!T)return{content:{status:"error",error:"No start edge"}};let k=re(g.stateUpdates??{}),E=await W(T,k);return q(E,k,n,o,c,l,x,e.nodeOptions,t.id)}if(g.action==="continue"){if(!R)return{content:{status:"error",error:"No session ID available for continue action."}};let y;try{y=await p.get(R)}catch(v){let I=v instanceof Error?v.message:String(v);return{content:{status:"error",error:`Failed to load flow state (session "${R}"): ${I}`}}}if(!y)return{content:{status:"error",error:`Flow state not found for session "${R}". The flow may have expired.`}};let T=y.state,k=y.step;if(!k)return{content:{status:"error",error:'This flow has already completed. Use action "start" to begin a new flow.'}};let E=F(T,re(g.stateUpdates??{}));if(y.widgetId){let v=o.get(k);if(!v)return{content:{status:"error",error:`No edge from step "${k}"`}};let I=await W(v,E);return q(I,E,n,o,c,l,x,e.nodeOptions,t.id)}return q(k,E,n,o,c,l,x,e.nodeOptions,t.id)}return{content:{status:"error",error:`Unknown action: "${g.action}"`}}}let u=qe(t.state),f={title:t.title,description:a,inputSchema:r,annotations:t.annotations,...u.length>0&&{_meta:{[L]:u}}},h=(async(g,R)=>{let l=R,x=l._meta??{},y=P(x),T=B(l),k=await d(g,y,x,T);if(y)if(k.flowTokenContent&&k.content.status!=="complete")try{await p.set(y,k.flowTokenContent)}catch(v){let I=v instanceof Error?v.message:String(v);return{content:[{type:"text",text:JSON.stringify({status:"error",error:`Flow state failed to persist (session "${y}"): ${I}`},null,2)}],_meta:x,isError:!0}}else k.content.status==="complete"&&await p.delete(y);return{content:[{type:"text",text:JSON.stringify(k.content,null,2)}],_meta:x,...k.content.status==="error"?{isError:!0}:{}}});return{name:t.id,config:f,handler:h,async register(g){let R={...f,_meta:{...f._meta,_flowGraph:i}};g.registerTool(t.id,R,h)},graph:e.graph,flowGraph:i}}function He(e,t){let n=["flowchart TD"];n.push(` ${b}((Start))`);for(let[o]of e)n.push(` ${o}[${o}]`);n.push(` ${_}((End))`);for(let[o,r]of t)r.type==="direct"?n.push(` ${o} --> ${r.to}`):n.push(` ${o} -.-> ${o}_branch([?])`);return n.join(`
6
- `)}var N=class{nodes=new Map;edges=new Map;nodeOptions=new Map;config;constructor(t){this.config=t}addNode(t,n,o){if(t===b||t===_)throw new Error(`"${t}" is a reserved name and cannot be used as a node name`);if(this.nodes.has(t))throw new Error(`Node "${t}" already exists`);return this.nodes.set(t,n),o&&this.nodeOptions.set(t,o),this}addEdge(t,n){if(this.edges.has(t))throw new Error(`Node "${t}" already has an outgoing edge. Use addConditionalEdge for branching.`);return this.edges.set(t,{type:"direct",to:n}),this}addConditionalEdge(t,n){if(this.edges.has(t))throw new Error(`Node "${t}" already has an outgoing edge.`);return this.edges.set(t,{type:"conditional",condition:n}),this}graph(){return He(this.nodes,this.edges)}compile(t){this.validate();let n=new Map(this.nodes),o=new Map(this.edges);return Le({config:this.config,nodes:n,edges:o,store:t?.store,graph:()=>He(n,o),nodeOptions:new Map(this.nodeOptions)})}validate(){if(!this.edges.has(b))throw new Error('Flow must have an entry point. Add an edge from START: .addEdge(START, "first_node")');let t=this.edges.get(b);if(t?.type==="direct"&&t.to!==_&&!this.nodes.has(t.to))throw new Error(`START edge references non-existent node: "${t.to}"`);for(let[n,o]of this.edges){if(n!==b&&!this.nodes.has(n))throw new Error(`Edge from non-existent node: "${n}"`);if(o.type==="direct"&&o.to!==_&&!this.nodes.has(o.to))throw new Error(`Edge from "${n}" references non-existent node: "${o.to}"`)}for(let[n]of this.nodes)if(!this.edges.has(n))throw new Error(`Node "${n}" has no outgoing edge. Add one with .addEdge("${n}", ...) or .addConditionalEdge("${n}", ...)`)}};function Ve(e){return new N(e)}function ze(e){let t=e.content;return JSON.parse(t[0]?.text??"")}async function Ge(e,t){let n=t?.stateStore,o=[],r=`test-session-${Math.random().toString(36).slice(2,10)}`,i={registerTool:(...c)=>{o.push(c)}};await e.register(i);let s=o[0]?.[2];if(!s)throw new Error(`Flow "${e.name}" did not register a handler`);let a={_meta:{sessionId:r}};async function p(c){return{...c,decodedState:n?await n.get(r):null}}return{async start(c,d,u){let f=await s({action:"start",intent:c,...u?{context:u}:{},...d?{stateUpdates:d}:{}},a);return p(ze(f))},async continueWith(c){let d=await s({action:"continue",...c?{stateUpdates:c}:{}},a);return p(ze(d))},async lastState(){return n?n.get(r):null}}}var H="text/html+skybridge",V="text/html;profile=mcp-app",Ye=async(e,t)=>{let n=e.endsWith("/")?e.slice(0,-1):e;return await(await fetch(`${n}${t}`)).text()};function Ze(e){return{"openai/widgetDescription":e.description,"openai/widgetPrefersBorder":e.prefersBorder,"openai/widgetDomain":e.widgetDomain,...e.widgetCSP&&{"openai/widgetCSP":e.widgetCSP}}}function Je(e){let t=e.widgetCSP?{connectDomains:e.widgetCSP.connect_domains,resourceDomains:e.widgetCSP.resource_domains,frameDomains:e.widgetCSP.frame_domains,redirectDomains:e.widgetCSP.redirect_domains}:void 0;return{ui:{...t&&{csp:t},...e.widgetDomain&&{domain:e.widgetDomain},...e.prefersBorder!==void 0&&{prefersBorder:e.prefersBorder}}}}function ce(e){return{...e.openaiTemplateUri&&{"openai/outputTemplate":e.openaiTemplateUri},"openai/toolInvocation/invoking":e.invoking,"openai/toolInvocation/invoked":e.invoked,"openai/widgetAccessible":!0,"openai/resultCanProduceWidget":!0,...e.mcpTemplateUri&&{ui:{resourceUri:e.mcpTemplateUri,...e.autoHeight&&{autoHeight:!0}}},...e.mcpTemplateUri&&{"ui/resourceUri":e.mcpTemplateUri}}}function Xe(e){let{id:t,title:n,description:o,baseUrl:r,htmlPath:i,cacheKey:s=process.env.VERCEL_DEPLOYMENT_ID,widgetDomain:a,prefersBorder:p=!0,autoHeight:c=!0}=e,d=e.widgetCSP??{connect_domains:[r],resource_domains:[r]};if(process.env.NODE_ENV==="development")try{let{hostname:y}=new URL(r);(y==="localhost"||y==="127.0.0.1")&&(d={...d,connect_domains:[...d.connect_domains||[],`ws://${y}:*`,`wss://${y}:*`],resource_domains:[...d.resource_domains||[],`http://${y}:*`]})}catch{}let u=typeof s=="string"&&s.trim()?`-${encodeURIComponent(s.trim())}`:"",f=`ui://widgets/apps-sdk/${t}${u}.html`,h=`ui://widgets/ext-apps/${t}${u}.html`,g=null,R=()=>(g||(g=Ye(r,i)),g),l=o;async function x(y){let T=await R();y.registerResource(`${t}-openai-widget`,f,{title:n,description:l,mimeType:H,_meta:{"openai/widgetDescription":l,"openai/widgetPrefersBorder":p}},async k=>({contents:[{uri:k.href,mimeType:H,text:T,_meta:Ze({description:l,prefersBorder:p,widgetDomain:a,widgetCSP:d})}]})),y.registerResource(`${t}-mcp-widget`,h,{title:n,description:l,mimeType:V,_meta:{ui:{prefersBorder:p}}},async k=>({contents:[{uri:k.href,mimeType:V,text:T,_meta:Je({description:l,prefersBorder:p,widgetDomain:a,widgetCSP:d})}]}))}return{id:t,title:n,description:o,openaiUri:f,mcpUri:h,autoHeight:c,register:x}}function Wt(e,t){let{resource:n,description:o,inputSchema:r,annotations:i,autoInjectResultText:s=!0}=e,a=e.id??n?.id,p=e.title??n?.title;if(!a)throw new Error("createTool: `id` is required when no resource is provided");if(!p)throw new Error("createTool: `title` is required when no resource is provided");let c=n?ce({openaiTemplateUri:n.openaiUri,mcpTemplateUri:n.mcpUri,invoking:e.invoking??"Loading...",invoked:e.invoked??"Loaded",autoHeight:n.autoHeight}):void 0;return{id:a,title:p,description:o,async register(d){d.registerTool(a,{title:p,description:o,inputSchema:r,annotations:i,...c&&{_meta:c}},(async(u,f)=>{let h=f,g=h._meta??{},R=B(h),l=await t(u,{extra:{_meta:g},waniwani:R});return n&&l.data?{content:[{type:"text",text:l.text}],structuredContent:l.data,_meta:{...c,...g,...s===!1?{"waniwani/autoInjectResultText":!1}:{}}}:{content:[{type:"text",text:l.text}],...l.data?{structuredContent:l.data}:{},...s===!1?{_meta:{"waniwani/autoInjectResultText":!1}}:{}}}))}}}async function At(e,t){await Promise.all(t.map(n=>n.register(e)))}var z=class extends Error{constructor(n,o){super(n);this.status=o;this.name="WaniWaniError"}};var Mt="@waniwani/sdk";function Qe(e){let{apiUrl:t,apiKey:n}=e;function o(){if(!n)throw new Error("WANIWANI_API_KEY is not set");return n}async function r(i,s,a){let p=o(),c=`${t.replace(/\/$/,"")}${s}`,d={Authorization:`Bearer ${p}`,"X-WaniWani-SDK":Mt},u={method:i,headers:d};a!==void 0&&(d["Content-Type"]="application/json",u.body=JSON.stringify(a));let f=await fetch(c,u);if(!f.ok){let g=await f.text().catch(()=>"");throw new z(g||`KB API error: HTTP ${f.status}`,f.status)}return(await f.json()).data}return{async ingest(i){return r("POST","/api/mcp/kb/ingest",{files:i})},async search(i,s){return r("POST","/api/mcp/kb/search",{query:i,...s})},async sources(){return r("GET","/api/mcp/kb/sources")}}}var Nt="@waniwani/sdk";function Y(e,t={}){let n=t.now??(()=>new Date),o=t.generateId??et,r=Ot(e),i=G(e.meta),s=G(e.metadata),a=Kt(e,i),p=C(e.eventId)??o(),c=jt(e.timestamp,n),d=C(e.source)??O(i)??t.source??Nt,u=de(e)?{...e}:void 0,f={...s};return Object.keys(i).length>0&&(f.meta=i),u&&(f.rawLegacy=u),{id:p,type:"mcp.event",name:r,source:d,timestamp:c,correlation:a,properties:Ut(e,r),metadata:f,rawLegacy:u}}function et(){return typeof crypto<"u"&&typeof crypto.randomUUID=="function"?`evt_${crypto.randomUUID()}`:`evt_${Math.random().toString(36).slice(2,10)}_${Date.now().toString(36)}`}function Ut(e,t){if(!de(e))return G(e.properties);let n=Dt(e,t),o=G(e.properties);return{...n,...o}}function Dt(e,t){switch(t){case"tool.called":{let n={};return C(e.toolName)&&(n.name=e.toolName),C(e.toolType)&&(n.type=e.toolType),n}case"quote.succeeded":{let n={};return typeof e.quoteAmount=="number"&&(n.amount=e.quoteAmount),C(e.quoteCurrency)&&(n.currency=e.quoteCurrency),n}case"link.clicked":{let n={};return C(e.linkUrl)&&(n.url=e.linkUrl),n}case"purchase.completed":{let n={};return typeof e.purchaseAmount=="number"&&(n.amount=e.purchaseAmount),C(e.purchaseCurrency)&&(n.currency=e.purchaseCurrency),n}default:return{}}}function Ot(e){return de(e)?e.eventType:e.event}function Kt(e,t){let n=C(e.requestId)??Ce(t),o=C(e.sessionId)??P(t),r=C(e.traceId)??_e(t),i=C(e.externalUserId)??be(t),s=C(e.correlationId)??Pe(t)??n,a={};return o&&(a.sessionId=o),r&&(a.traceId=r),n&&(a.requestId=n),s&&(a.correlationId=s),i&&(a.externalUserId=i),a}function jt(e,t){if(e instanceof Date)return e.toISOString();if(typeof e=="string"){let n=new Date(e);if(!Number.isNaN(n.getTime()))return n.toISOString()}return t().toISOString()}function G(e){return!e||typeof e!="object"||Array.isArray(e)?{}:e}function C(e){if(typeof e=="string"&&e.trim().length!==0)return e}function de(e){return"eventType"in e}var $t="/api/mcp/events/v2/batch";var tt="@waniwani/sdk",Bt=new Set([401,403]),qt=new Set([408,425,429,500,502,503,504]);function nt(e){return new pe(e)}var pe=class{endpointUrl;flushIntervalMs;maxBatchSize;maxBufferSize;maxRetries;retryBaseDelayMs;retryMaxDelayMs;shutdownTimeoutMs;sdkVersion;fetchFn;logger;now;sleep;apiKey;buffer=[];flushTimer;flushScheduled=!1;flushScheduledTimer;flushInFlight;inFlightCount=0;isStopped=!1;isShuttingDown=!1;constructor(t){this.endpointUrl=Vt(t.apiUrl,t.endpointPath??$t),this.flushIntervalMs=t.flushIntervalMs??1e3,this.maxBatchSize=t.maxBatchSize??20,this.maxBufferSize=t.maxBufferSize??1e3,this.maxRetries=t.maxRetries??3,this.retryBaseDelayMs=t.retryBaseDelayMs??200,this.retryMaxDelayMs=t.retryMaxDelayMs??2e3,this.shutdownTimeoutMs=t.shutdownTimeoutMs??2e3,this.fetchFn=t.fetchFn??fetch,this.logger=t.logger??console,this.now=t.now??(()=>new Date),this.sleep=t.sleep??(n=>new Promise(o=>setTimeout(o,n))),this.apiKey=t.apiKey,this.sdkVersion=t.sdkVersion,this.flushIntervalMs>0&&(this.flushTimer=setInterval(()=>{this.flush()},this.flushIntervalMs))}enqueue(t){if(this.isStopped||this.isShuttingDown){this.logger.warn("[WaniWani] Tracking transport is stopped, dropping event %s",t.id);return}if(this.buffer.length>=this.maxBufferSize){let n=this.buffer.length-this.maxBufferSize+1;this.buffer.splice(0,n),this.logger.warn("[WaniWani] Tracking buffer overflow, dropped %d oldest event(s)",n)}if(this.buffer.push(t),this.buffer.length>=this.maxBatchSize){this.flush();return}this.scheduleMicroFlush()}pendingEvents(){return this.buffer.length+this.inFlightCount}async flush(){return this.flushInFlight?this.flushInFlight:(this.flushInFlight=this.flushLoop().finally(()=>{this.flushInFlight=void 0}),this.flushInFlight)}async shutdown(t){this.isShuttingDown=!0,this.flushTimer&&(clearInterval(this.flushTimer),this.flushTimer=void 0),this.flushScheduledTimer&&(clearTimeout(this.flushScheduledTimer),this.flushScheduledTimer=void 0,this.flushScheduled=!1);let n=t?.timeoutMs??this.shutdownTimeoutMs,o=this.flush();if(!Number.isFinite(n)||n<=0)return await o,this.isStopped=!0,{timedOut:!1,pendingEvents:this.pendingEvents()};let r=Symbol("shutdown-timeout");return await Promise.race([o.then(()=>"flushed"),this.sleep(n).then(()=>r)])===r?(this.isStopped=!0,{timedOut:!0,pendingEvents:this.pendingEvents()}):(this.isStopped=!0,{timedOut:!1,pendingEvents:this.pendingEvents()})}scheduleMicroFlush(){this.flushScheduled||(this.flushScheduled=!0,this.flushScheduledTimer=setTimeout(()=>{this.flushScheduledTimer=void 0,this.flushScheduled=!1,this.flush()},0))}async flushLoop(){for(;this.buffer.length>0&&!this.isStopped;){let t=this.buffer.splice(0,this.maxBatchSize);await this.sendBatchWithRetry(t)}}async sendBatchWithRetry(t){let n=0,o=t;for(;o.length>0&&!this.isStopped;){this.inFlightCount=o.length;let r=await this.sendBatchOnce(o);switch(this.inFlightCount=0,r.kind){case"success":return;case"auth":this.stopTransportForAuthFailure(r.status,o.length);return;case"permanent":this.logger.error("[WaniWani] Dropping %d event(s) after permanent failure: %s",o.length,r.reason);return;case"retryable":if(n>=this.maxRetries){this.logger.error("[WaniWani] Dropping %d event(s) after retry exhaustion: %s",o.length,r.reason);return}await this.sleep(this.backoffDelayMs(n)),n+=1;continue;case"partial":if(r.permanent.length>0&&this.logger.error("[WaniWani] Dropping %d event(s) rejected as permanent",r.permanent.length),r.retryable.length===0)return;if(n>=this.maxRetries){this.logger.error("[WaniWani] Dropping %d retryable event(s) after retry exhaustion",r.retryable.length);return}o=r.retryable,await this.sleep(this.backoffDelayMs(n)),n+=1;continue}}}async sendBatchOnce(t){let n;try{n=await this.fetchFn(this.endpointUrl,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${this.apiKey}`,"X-WaniWani-SDK":tt},body:JSON.stringify(this.makeBatchRequest(t))})}catch(i){return{kind:"retryable",reason:zt(i)}}if(Bt.has(n.status))return{kind:"auth",status:n.status};if(qt.has(n.status))return{kind:"retryable",reason:`HTTP ${n.status}`};if(!n.ok)return{kind:"permanent",reason:`HTTP ${n.status}`};let o=await Ht(n);if(!o?.rejected||o.rejected.length===0)return{kind:"success"};let r=this.classifyRejectedEvents(t,o.rejected);return r.retryable.length===0&&r.permanent.length===0?{kind:"success"}:{kind:"partial",retryable:r.retryable,permanent:r.permanent}}makeBatchRequest(t){return{sentAt:this.now().toISOString(),source:{sdk:tt,version:this.sdkVersion??"0.0.0"},events:t}}classifyRejectedEvents(t,n){let o=new Map(t.map(s=>[s.id,s])),r=[],i=[];for(let s of n){let a=o.get(s.eventId);if(a){if(Lt(s)){r.push(a);continue}i.push(a)}}return{retryable:r,permanent:i}}backoffDelayMs(t){let n=this.retryBaseDelayMs*2**t;return Math.min(n,this.retryMaxDelayMs)}stopTransportForAuthFailure(t,n){this.isStopped=!0;let o=this.buffer.length;this.buffer.splice(0,o),this.logger.error("[WaniWani] Auth failure (HTTP %d). Stopping tracking transport and dropping %d queued event(s)",t,n+o)}};function Lt(e){if(e.retryable===!0)return!0;let t=e.code.toLowerCase();return t.includes("timeout")||t.includes("temporary")||t.includes("unavailable")||t.includes("rate_limit")||t.includes("transient")||t.includes("server")}async function Ht(e){let t=await e.text();if(t)try{return JSON.parse(t)}catch{return}}function Vt(e,t){let n=e.endsWith("/")?e:`${e}/`,o=t.startsWith("/")?t.slice(1):t;return`${n}${o}`}function zt(e){return e instanceof Error?e.message:String(e)}function ot(e){let{apiUrl:t,apiKey:n,tracking:o}=e;function r(){if(!n)throw new Error("WANIWANI_API_KEY is not set");return n}let i=n?nt({apiUrl:t,apiKey:n,endpointPath:o.endpointPath,flushIntervalMs:o.flushIntervalMs,maxBatchSize:o.maxBatchSize,maxBufferSize:o.maxBufferSize,maxRetries:o.maxRetries,retryBaseDelayMs:o.retryBaseDelayMs,retryMaxDelayMs:o.retryMaxDelayMs,shutdownTimeoutMs:o.shutdownTimeoutMs}):void 0,s={async identify(a,p,c){r();let d=Y({event:"user.identified",externalUserId:a,properties:p,meta:c});return i?.enqueue(d),{eventId:d.id}},async track(a){r();let p=Y(a);return i?.enqueue(p),{eventId:p.id}},async flush(){r(),await i?.flush()},async shutdown(a){return r(),await i?.shutdown({timeoutMs:a?.timeoutMs??o.shutdownTimeoutMs})??{timedOut:!1,pendingEvents:0}}};return i&&Gt(s,o.shutdownTimeoutMs),s}function Gt(e,t){if(typeof process>"u"||typeof process.once!="function"||typeof process.on!="function")return;let n=()=>{e.shutdown({timeoutMs:t})};process.once("beforeExit",n),process.once("SIGINT",n),process.once("SIGTERM",n)}function Z(e){let t=e,n=t?.apiUrl??"https://app.waniwani.ai",o=t?.apiKey??process.env.WANIWANI_API_KEY,r={endpointPath:t?.tracking?.endpointPath??"/api/mcp/events/v2/batch",flushIntervalMs:t?.tracking?.flushIntervalMs??1e3,maxBatchSize:t?.tracking?.maxBatchSize??20,maxBufferSize:t?.tracking?.maxBufferSize??1e3,maxRetries:t?.tracking?.maxRetries??3,retryBaseDelayMs:t?.tracking?.retryBaseDelayMs??200,retryMaxDelayMs:t?.tracking?.retryMaxDelayMs??2e3,shutdownTimeoutMs:t?.tracking?.shutdownTimeoutMs??2e3},i={apiUrl:n,apiKey:o,tracking:r},s=ot(i),a=Qe(i);return{...s,kb:a,_config:i}}function Yt(e){let t=e.event_type??"widget_click",o=t.startsWith("widget_")?t:`widget_${t}`,r={...e.metadata??{}};return e.event_name&&(r.event_name=e.event_name),{event:o,properties:r,sessionId:e.session_id,traceId:e.trace_id,externalUserId:e.user_id,eventId:e.event_id,timestamp:e.timestamp,source:e.source??"widget"}}function Zt(e){let t={apiKey:e?.apiKey,apiUrl:e?.apiUrl},n;function o(){return n||(n=Z(t)),n}return async function(i){let s;try{s=await i.json()}catch{return new Response(JSON.stringify({error:"Invalid JSON"}),{status:400,headers:{"Content-Type":"application/json"}})}if(!Array.isArray(s.events)||s.events.length===0)return new Response(JSON.stringify({error:"Missing or empty events array"}),{status:400,headers:{"Content-Type":"application/json"}});try{let a=o(),p=[];for(let c of s.events){let d=Yt(c),u=await a.track(d);p.push(u.eventId)}return await a.flush(),new Response(JSON.stringify({ok:!0,accepted:p.length}),{status:200,headers:{"Content-Type":"application/json"}})}catch(a){let p=a instanceof Error?a.message:"Unknown error";return new Response(JSON.stringify({error:p}),{status:500,headers:{"Content-Type":"application/json"}})}}}function J(e,t){return t?(...n)=>console.log(`[waniwani:${e}]`,...n):()=>{}}var ue=J("widget-token",!!process.env.WANIWANI_DEBUG),Jt=120*1e3,X=class{cached=null;pending=null;config;constructor(t){this.config=t}async getToken(t,n){return this.cached&&Date.now()<this.cached.expiresAt-Jt?this.cached.token:this.pending?this.pending:(this.pending=this.mint(t,n).finally(()=>{this.pending=null}),this.pending)}async mint(t,n){let o=Xt(this.config.apiUrl,"/api/mcp/widget-tokens");ue("minting token from",o);let r={};t&&(r.sessionId=t),n&&(r.traceId=n);try{let i=await fetch(o,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${this.config.apiKey}`},body:JSON.stringify(r),signal:AbortSignal.timeout(5e3)});if(ue("mint response:",i.status),!i.ok)return null;let s=await i.json(),a=s.data&&typeof s.data=="object"?s.data:s,p=new Date(a.expiresAt).getTime();return!a.token||Number.isNaN(p)?null:(this.cached={token:a.token,expiresAt:p},a.token)}catch(i){return ue("mint failed:",i),null}}};function Xt(e,t){return`${e.endsWith("/")?e.slice(0,-1):e}${t}`}async function Qt(e){let t=JSON.stringify({nodes:e.nodes,edges:e.edges});if(typeof globalThis.crypto?.subtle?.digest=="function"){let o=await crypto.subtle.digest("SHA-256",new TextEncoder().encode(t));return Array.from(new Uint8Array(o)).map(r=>r.toString(16).padStart(2,"0")).join("")}let n=0;for(let o=0;o<t.length;o++)n=(n<<5)-n+t.charCodeAt(o)|0;return`simple-${Math.abs(n).toString(36)}`}async function rt(e,t,n){if(e.length===0)return;let o=await Promise.all(e.map(async i=>({...i,configHash:await Qt(i)}))),r=JSON.stringify({flows:o});try{let i=`${t}/api/mcp/funnel/sync`;fetch(i,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${n}`},body:r}).catch(()=>{})}catch{}}var it="waniwani/sessionId",Q="waniwani/geoLocation",ee="waniwani/userLocation",en="openai/userLocation",tn=[en,Q,ee],nn=["latitude","longitude"];function st(e){let t;for(let n of tn){let o=e[n];if(!S(o))continue;let r;for(let i of nn)i in o&&(r||(r={...o}),delete r[i]);r&&(t||(t={...e}),t[n]=r)}return t??e}function S(e){return!!e&&typeof e=="object"&&!Array.isArray(e)}function $(e){if(!S(e))return;let t=e._meta;return S(t)?t:void 0}function le(e){if(!S(e))return;let t=e.content;return Array.isArray(t)?t.find(o=>S(o)&&o.type==="text"&&typeof o.text=="string")?.text:void 0}function on(e,t){return typeof t=="function"?t(e)??"other":t??"other"}function fe(e,t,n,o,r,i){let s=on(e,n.toolType),a=$(t),p=a&&n.stripGeoCoordinates?st(a):a,c=i?.input!==void 0&&n.redactInput?n.redactInput(i.input):i?.input,d=n.stripGeoCoordinates&&S(i?.output)&&S(i.output._meta)?{...i.output,_meta:st(i.output._meta)}:i?.output;return{event:"tool.called",properties:{name:e,type:s,...o??{},...c!==void 0&&{input:c},...d!==void 0&&{output:d}},meta:p,source:O(p),metadata:{...n.metadata??{},...r&&{clientInfo:r}}}}async function ge(e,t,n){try{await e.track(t)}catch(o){n?.(me(o))}}async function we(e,t){try{await e.flush()}catch(n){t?.(me(n))}}async function ct(e,t,n,o,r){if(!S(e))return;S(e._meta)||(e._meta={});let i=e._meta,s=S(i.waniwani)?i.waniwani:void 0,a={...s??{},endpoint:s?.endpoint??`${n.replace(/\/$/,"")}/api/mcp/events/v2/batch`};if(t)try{let u=await t.getToken();u&&(a.token=u)}catch(u){r?.(me(u))}let p=P(i);p&&(a.sessionId||(a.sessionId=p));let c=ut(i);c!==void 0&&(a.geoLocation||(a.geoLocation=c));let d=O($(o));d&&!a.source&&(a.source=d),i.waniwani=a}var at=["openai/outputTemplate","openai/widgetAccessible","openai/resultCanProduceWidget","openai/toolInvocation/invoking","openai/toolInvocation/invoked","ui/resourceUri","ui"];function dt(e,t){if(!t||!S(e))return;let n=!1;for(let r of at)if(r in t){n=!0;break}if(!n)return;S(e._meta)||(e._meta={});let o=e._meta;for(let r of at)r in t&&(r in o||(o[r]=t[r]))}function pt(e,t){let n=$(t);if(!n||!S(e))return;S(e._meta)||(e._meta={});let o=e._meta,r=P(n);r&&!o[it]&&(o[it]=r);let i=ut(n);i&&(o[Q]||(o[Q]=i),o[ee]||(o[ee]=i))}function ut(e){if(!e)return;let t=e[Q]??e[ee];if(S(t)||typeof t=="string")return t}function me(e){return e instanceof Error?e:new Error(String(e))}function lt(e){return typeof e=="object"&&e!==null&&!Array.isArray(e)}function ft(e){if(typeof e.sessionId=="string"&&e.sessionId)return e.sessionId;if(lt(e.requestInfo)){let t=e.requestInfo.headers;if(lt(t)){let n={};for(let r of Object.keys(t))n[r.toLowerCase()]=t[r];let o=n["mcp-session-id"];if(typeof o=="string"&&o)return o}}}var wt=Symbol.for("waniwani.wrappedHandler"),te=J("mcp",!!process.env.WANIWANI_DEBUG),he="https://app.waniwani.ai",rn="REDACTED";function sn(e){if(!e)return;let t=e[L];if(!Array.isArray(t)||t.length===0)return;let n=new Set(t.filter(o=>typeof o=="string"));if(n.size!==0)return o=>{if(!S(o))return o;let r=o.stateUpdates;if(!S(r))return o;let i=!1,s={...r};for(let a of n)a in s&&(s[a]=rn,i=!0);return i?{...o,stateUpdates:s}:o}}function gt(e,t,n,o){let{server:r,tracker:i,opts:s,tokenCache:a,injectToken:p}=n,c=s.applyFieldRedactions===!0?sn(o):void 0,d=c?{...s,redactInput:c}:s,u=async(f,h)=>{let g=$(h)??{};if(!P(g)&&S(h)){let T=ft(h);T&&(g["waniwani/sessionId"]=T,h._meta=g)}let l=Ie(i,g,{apiUrl:i._config.apiUrl,apiKey:i._config.apiKey});S(h)&&(h[oe]=l);let x=performance.now(),y=r.server?.getClientVersion?.();try{let T=await t(f,h),k=Math.round(performance.now()-x);te(`tool "${e}" handler returned in ${k}ms, running post-processing...`);let E=S(T)&&T.isError===!0;if(E){let v=le(T);console.error(`[waniwani] Tool "${e}" returned error${v?`: ${v}`:""}`)}return await ge(i,fe(e,h,d,{durationMs:k,status:E?"error":"ok",...E&&{errorMessage:le(T)??"Unknown tool error"}},y,{input:f,output:T}),s.onError),te(`tool "${e}" tracking done`),s.flushAfterToolCall&&await we(i,s.onError),pt(T,h),dt(T,o),p&&(await ct(T,a,i._config.apiUrl??he,h,s.onError),te(`tool "${e}" widget config injected`)),te(`tool "${e}" post-processing complete, returning result`),T}catch(T){let k=Math.round(performance.now()-x);throw await ge(i,fe(e,h,d,{durationMs:k,status:"error",errorMessage:T instanceof Error?T.message:String(T)},y,{input:f}),s.onError),s.flushAfterToolCall&&await we(i,s.onError),T}};return u[wt]=!0,u}function an(e,t){let n=e;if(n.__waniwaniWrapped)return n;n.__waniwaniWrapped=!0;let o=t??{},r=o.client??Z(),i=o.injectWidgetToken!==!1,s=r._config.apiKey?new X({apiUrl:r._config.apiUrl??he,apiKey:r._config.apiKey}):null,a={server:e,tracker:r,opts:o,tokenCache:s,injectToken:i},p=e.registerTool.bind(e);n.registerTool=((...d)=>{let[u,f,h]=d;if(typeof h!="function")return p(...d);let g=typeof u=="string"&&u.trim().length>0?u:"unknown",R=S(f)&&S(f._meta)?f._meta:void 0,l=gt(g,h,a,R);return p(u,f,l)});let c=e._registeredTools;if(S(c))for(let[d,u]of Object.entries(c)){if(!S(u))continue;let f=u.handler;if(typeof f!="function"||f[wt])continue;let h=S(u._meta)?u._meta:void 0;u.handler=gt(d,f,a,h)}if(r._config.apiKey){let d=e._registeredTools,u=[];if(d&&typeof d=="object"){for(let f of Object.values(d))if(f&&typeof f=="object"){let h=f._meta,g=h&&typeof h=="object"?h._flowGraph:void 0;g?.nodes?.length&&u.push(g)}}u.length>0&&rt(u,r._config.apiUrl??he,r._config.apiKey)}return n}export{_ as END,b as START,N as StateGraph,A as WaniwaniKvStore,Ve as createFlow,Ge as createFlowTestHarness,Xe as createResource,Wt as createTool,Zt as createTrackingRoute,ne as detectPlatform,ht as isMCPApps,mt as isOpenAI,$e as redacted,At as registerTools,an as withWaniwani};
3
+ ${U.context}`:`ERROR: ${I}`}:U),Te=Ae(ye,f.context,c,d);if(Te)return Te;break}}let y=r.get(c);if(!y)return{content:{status:"error",error:`No outgoing edge from node "${c}"`}};c=await W(y,d);continue}if(Ee(f)){let T=f.field;if(T&&ie(K(d,T))){let y=r.get(c);if(!y)return{content:{status:"error",error:`No outgoing edge from node "${c}"`}};c=await W(y,d);continue}return{content:{status:"widget",tool:f.tool,data:f.data,description:f.description,interactive:f.interactive!==!1},flowTokenContent:{step:c,state:d,field:T,widgetId:f.tool}}}d=F(d,f);let x=r.get(c);if(!x)return{content:{status:"error",error:`No outgoing edge from node "${c}"`}};c=await W(x,d)}catch(R){return{content:{status:"error",error:R instanceof Error?R.message:String(R)},flowTokenContent:{step:c,state:d}}}}return{content:{status:"error",error:"Flow exceeded maximum iterations (possible infinite loop)"}}}function se(e){return typeof e=="object"&&e!==null&&e.__ww_enc===1&&typeof e.ct=="string"&&typeof e.iv=="string"}var Me=new Map;async function Ne(e){let t=Me.get(e);if(t)return t;let n=Buffer.from(e,"base64");if(n.length!==32)throw new Error("[WaniWani KV] WANIWANI_ENCRYPTION_KEY must be a base64-encoded 32-byte (256-bit) key.");let r=await globalThis.crypto.subtle.importKey("raw",n,{name:"AES-GCM"},!1,["encrypt","decrypt"]);return Me.set(e,r),r}async function Ue(e,t){let n=await Ne(t),r=globalThis.crypto.getRandomValues(new Uint8Array(12)),o=new TextEncoder().encode(JSON.stringify(e)),i=await globalThis.crypto.subtle.encrypt({name:"AES-GCM",iv:r},n,o);return{__ww_enc:1,ct:Buffer.from(i).toString("base64"),iv:Buffer.from(r).toString("base64")}}async function De(e,t){let n=await Ne(t),r=Buffer.from(e.ct,"base64"),o=Buffer.from(e.iv,"base64"),i;try{i=await globalThis.crypto.subtle.decrypt({name:"AES-GCM",iv:o},n,r)}catch{throw new Error("[WaniWani KV] Decryption failed. The encryption key may be incorrect or the data may be corrupted.")}return JSON.parse(new TextDecoder().decode(i))}var Et="@waniwani/sdk",It="https://app.waniwani.ai";function _t(...e){process.env.WANIWANI_DEBUG&&console.log("[waniwani:kv]",...e)}var A=class{get baseUrl(){return(process.env.WANIWANI_API_URL??It).replace(/\/$/,"")}get apiKey(){return process.env.WANIWANI_API_KEY}get encryptionKey(){return process.env.WANIWANI_ENCRYPTION_KEY}async get(t){if(!this.apiKey)throw new Error("[WaniWani KV] No API key configured. Set WANIWANI_API_KEY env var.");let n=await this.request("/api/mcp/redis/get",{key:t});if(n==null)return null;if(se(n)){if(!this.encryptionKey)throw new Error("[WaniWani KV] Encrypted data found but WANIWANI_ENCRYPTION_KEY is not set.");return De(n,this.encryptionKey)}return n}async set(t,n){if(!this.apiKey)throw new Error("[WaniWani KV] No API key configured. Set WANIWANI_API_KEY env var.");let r=this.encryptionKey;_t(`set "${t}" \u2014 encryption ${r?"enabled":"disabled (no WANIWANI_ENCRYPTION_KEY)"}`);let o=r?await Ue(n,r):n;await this.request("/api/mcp/redis/set",{key:t,value:o})}async delete(t){if(this.apiKey)try{await this.request("/api/mcp/redis/delete",{key:t})}catch(n){console.error("[WaniWani KV] delete failed for key:",t,n)}}async request(t,n){let r=`${this.baseUrl}${t}`,o=await fetch(r,{method:"POST",headers:{Authorization:`Bearer ${this.apiKey}`,"Content-Type":"application/json","X-WaniWani-SDK":Et},body:JSON.stringify(n)});if(!o.ok){let s=await o.text().catch(()=>"");throw new Error(s||`KV store API error: HTTP ${o.status}`)}return(await o.json()).data}};var j=class{store=new A;get(t){return this.store.get(t)}set(t,n){return this.store.set(t,n)}delete(t){return this.store.delete(t)}};function Ct(e){let t=e.toString();return t.includes("showWidget")?"widget":t.includes("interrupt")?"interrupt":"action"}function bt(e,t){let n=e.toString(),r=[];for(let o of t){let i=`"${o}"`,s=`'${o}'`,a=`\`${o}\``;(n.includes(i)||n.includes(s)||n.includes(a))&&r.push(o)}return r}function Oe(e,t,n,r){let o=new Set([...t.keys(),C]),i=[];for(let[a,p]of t){let c=r.get(a);i.push({id:a,type:Ct(p),trackFunnel:c?.trackFunnel??!1})}let s=[];for(let[a,p]of n)if(p.type==="direct")s.push({from:a,to:p.to,type:"direct"});else{let c=bt(p.condition,o);s.push({from:a,to:c,type:"conditional"})}return{flowId:e.id,title:e.title,nodes:i,edges:s}}function Ke(e){let t=e.description??"",n=e._zod?.def;if(n?.type==="enum"&&n.entries){let r=Object.keys(n.entries).map(o=>`"${o}"`).join(" | ");return t?`${r} \u2014 ${t}`:r}return t}function je(e){let t=["","## FLOW EXECUTION PROTOCOL","","This tool implements a multi-step conversational flow. Follow this protocol exactly:","",'1. Call with `action: "start"` to begin and include `intent`.'," `intent` must be a brief summary of the user's goal for this flow."," Do NOT invent missing intent."," Optionally include `context` \u2014 the situation or environment that led the user to start"," this flow (e.g. what page they are on, what they were doing, or what triggered the request)."," Only provide `context` when there is genuinely relevant situational information. Do NOT invent missing context."," If the user's message already contains answers to likely questions,"," extract them into `stateUpdates` as `{ field: value }` pairs."," The engine will auto-skip steps whose fields are already filled."," Only extract values the user explicitly stated \u2014 do NOT guess or invent values."];if(e.omitIntentPII&&t.push(" Do NOT include PII in `intent` or `context` \u2014 no names, emails, phones, addresses, IDs, ages, or birthdates.",' Summarize the goal abstractly (e.g. "user wants a quote", not "Jane Doe wants a quote").'),e.state){let n=[];for(let[r,o]of Object.entries(e.state)){let i=Fe(o);if(i){let s=o.description??"",a=Object.entries(i).map(([p,c])=>{let d=Ke(c);return d?`\`${r}.${p}\` (${d})`:`\`${r}.${p}\``}).join(", ");n.push(s?`\`${r}\` (${s}): ${a}`:`\`${r}\`: ${a}`)}else{let s=Ke(o);n.push(s?`\`${r}\` (${s})`:`\`${r}\``)}}t.push(` Known fields: ${n.join(", ")}.`)}return t.push(" For grouped fields (shown as `group.subfield`), use dot-notation keys in `stateUpdates`:",' e.g. `{ "driver.name": "John", "driver.license": "ABC123" }`.',"2. The response JSON `status` field tells you what to do next:",' - `"interrupt"`: Pause and ask the user. Two forms:'," a. Single question: `{ question, field, context? }` \u2014 ask `question`, store answer in `field`."," b. Multi-question: `{ questions: [{question, field}, ...], context? }` \u2014 ask ALL questions"," in one conversational message, collect all answers."," `context` (if present) is hidden AI instructions \u2014 use to shape your response, do NOT show verbatim."," Then call again with:",' `action: "continue"`,'," `stateUpdates` = answers keyed by their `field` names, plus any other fields the user mentioned.",' - `"widget"`: The flow wants to show a UI widget. Call the tool named in the `tool`'," field, passing the `data` object as the tool's input."," Check the `interactive` field in the response:"," \u2022 `interactive: true` \u2014 The widget requires user interaction. After calling the display tool,"," STOP and WAIT for the user to interact with the widget. Do NOT call this flow tool again"," until the user has responded. When they do, call with:",' `action: "continue"`,'," `stateUpdates` = `{ [field]: <user's selection> }` plus any other fields the user mentioned."," \u2022 `interactive: false` \u2014 The widget is display-only. Call the display tool, then immediately",' call THIS flow tool again with `action: "continue"`. Do NOT wait for user interaction.',' - `"complete"`: The flow is done. Present the result to the user.',' - `"error"`: Something went wrong. Show the `error` message.',"","3. Do NOT invent state values. Only use `stateUpdates` for information the user explicitly provided.","4. Include only the fields the user actually answered in `stateUpdates` \u2014 do NOT guess missing ones."," If the user did not answer all pending questions, the engine will re-prompt for the remaining ones."," If the user mentioned values for other known fields, include those too \u2014"," they will be applied immediately and those steps will be auto-skipped."),t.join(`
4
+ `)}function ae(e){return!!e&&typeof e=="object"&&!Array.isArray(e)}function $e(e){let t=Be(e),n=ae(t.waniwani)?t.waniwani:{};return e.meta({...t,waniwani:{...n,redacted:!0}})}function Pt(e){let n=Be(e).waniwani;return ae(n)&&n.redacted===!0}function Be(e){let t=e.meta;if(typeof t!="function")return{};let n=t.call(e);return ae(n)?n:{}}function qe(e){if(!e)return[];let t=[];for(let[n,r]of Object.entries(e))Pt(r)&&t.push(n);return t}var L="waniwani/redactedStateUpdateFields";function Ft(e){let t=e.omitIntentPII?" Do not include PII (names, emails, phones, addresses, IDs, ages, birthdates) \u2014 summarize abstractly.":"";return{action:M.enum(["start","continue"]).describe('"start" to begin the flow, "continue" to resume after a pause (interrupt or widget)'),intent:M.string().optional().describe(`Required when action is "start". Provide a brief summary of the user's goal for this flow. Do not invent missing intent.${t}`),context:M.string().optional().describe(`Optional when action is "start". Describe the situation or environment that led the user to start this flow \u2014 e.g. what page they are on, what they were doing, or what triggered the request. Do not invent missing context.${t}`),stateUpdates:M.record(M.string(),M.unknown()).optional().describe("State field values to set before processing the next node. Use this to pass the user's answer (keyed by the field name from the response) and any other values the user mentioned.")}}function Le(e){let{config:t,nodes:n,edges:r}=e,o=Ft(t),i=Oe(t,n,r,e.nodeOptions),s=je(t),a=`${t.description}
5
+ ${s}`,p=e.store??new j,c=new Map;async function d(g,R,f,x){if(g.action==="start"){let T=typeof g.intent=="string"?g.intent.trim():void 0;if(!T)return{content:{status:"error",error:`Missing required "intent" for action "start". Include a brief summary of the user's goal for this flow and any relevant prior context that led to triggering it, if available.`}};if(g.intent=T,typeof g.context=="string"){let v=g.context.trim();g.context=v||void 0}let y=r.get(b);if(!y)return{content:{status:"error",error:"No start edge"}};let k=oe(g.stateUpdates??{}),E=await W(y,k);return q(E,k,n,r,c,f,x,e.nodeOptions,t.id)}if(g.action==="continue"){if(!R)return{content:{status:"error",error:"No session ID available for continue action."}};let T;try{T=await p.get(R)}catch(v){let I=v instanceof Error?v.message:String(v);return{content:{status:"error",error:`Failed to load flow state (session "${R}"): ${I}`}}}if(!T)return{content:{status:"error",error:`Flow state not found for session "${R}". The flow may have expired.`}};let y=T.state,k=T.step;if(!k)return{content:{status:"error",error:'This flow has already completed. Use action "start" to begin a new flow.'}};let E=F(y,oe(g.stateUpdates??{}));if(T.widgetId){let v=r.get(k);if(!v)return{content:{status:"error",error:`No edge from step "${k}"`}};let I=await W(v,E);return q(I,E,n,r,c,f,x,e.nodeOptions,t.id)}return q(k,E,n,r,c,f,x,e.nodeOptions,t.id)}return{content:{status:"error",error:`Unknown action: "${g.action}"`}}}let u=qe(t.state),l={title:t.title,description:a,inputSchema:o,annotations:t.annotations,...u.length>0&&{_meta:{[L]:u}}},h=(async(g,R)=>{let f=R,x=f._meta??{},T=P(x),y=B(f),k=await d(g,T,x,y);if(T)if(k.flowTokenContent&&k.content.status!=="complete")try{await p.set(T,k.flowTokenContent)}catch(v){let I=v instanceof Error?v.message:String(v);return{content:[{type:"text",text:JSON.stringify({status:"error",error:`Flow state failed to persist (session "${T}"): ${I}`},null,2)}],_meta:x,isError:!0}}else k.content.status==="complete"&&await p.delete(T);return{content:[{type:"text",text:JSON.stringify(k.content,null,2)}],_meta:x,...k.content.status==="error"?{isError:!0}:{}}});return{name:t.id,config:l,handler:h,async register(g){let R={...l,_meta:{...l._meta,_flowGraph:i}};g.registerTool(t.id,R,h)},graph:e.graph,flowGraph:i}}function He(e,t){let n=["flowchart TD"];n.push(` ${b}((Start))`);for(let[r]of e)n.push(` ${r}[${r}]`);n.push(` ${C}((End))`);for(let[r,o]of t)o.type==="direct"?n.push(` ${r} --> ${o.to}`):n.push(` ${r} -.-> ${r}_branch([?])`);return n.join(`
6
+ `)}var N=class{nodes=new Map;edges=new Map;nodeOptions=new Map;config;constructor(t){this.config=t}addNode(t,n,r){if(t===b||t===C)throw new Error(`"${t}" is a reserved name and cannot be used as a node name`);if(this.nodes.has(t))throw new Error(`Node "${t}" already exists`);return this.nodes.set(t,n),r&&this.nodeOptions.set(t,r),this}addEdge(t,n){if(this.edges.has(t))throw new Error(`Node "${t}" already has an outgoing edge. Use addConditionalEdge for branching.`);return this.edges.set(t,{type:"direct",to:n}),this}addConditionalEdge(t,n){if(this.edges.has(t))throw new Error(`Node "${t}" already has an outgoing edge.`);return this.edges.set(t,{type:"conditional",condition:n}),this}graph(){return He(this.nodes,this.edges)}compile(t){this.validate();let n=new Map(this.nodes),r=new Map(this.edges);return Le({config:this.config,nodes:n,edges:r,store:t?.store,graph:()=>He(n,r),nodeOptions:new Map(this.nodeOptions)})}validate(){if(!this.edges.has(b))throw new Error('Flow must have an entry point. Add an edge from START: .addEdge(START, "first_node")');let t=this.edges.get(b);if(t?.type==="direct"&&t.to!==C&&!this.nodes.has(t.to))throw new Error(`START edge references non-existent node: "${t.to}"`);for(let[n,r]of this.edges){if(n!==b&&!this.nodes.has(n))throw new Error(`Edge from non-existent node: "${n}"`);if(r.type==="direct"&&r.to!==C&&!this.nodes.has(r.to))throw new Error(`Edge from "${n}" references non-existent node: "${r.to}"`)}for(let[n]of this.nodes)if(!this.edges.has(n))throw new Error(`Node "${n}" has no outgoing edge. Add one with .addEdge("${n}", ...) or .addConditionalEdge("${n}", ...)`)}};function Ve(e){return new N(e)}function ze(e){let t=e.content;return JSON.parse(t[0]?.text??"")}async function Ge(e,t){let n=t?.stateStore,r=[],o=`test-session-${Math.random().toString(36).slice(2,10)}`,i={registerTool:(...c)=>{r.push(c)}};await e.register(i);let s=r[0]?.[2];if(!s)throw new Error(`Flow "${e.name}" did not register a handler`);let a={_meta:{sessionId:o}};async function p(c){return{...c,decodedState:n?await n.get(o):null}}return{async start(c,d,u){let l=await s({action:"start",intent:c,...u?{context:u}:{},...d?{stateUpdates:d}:{}},a);return p(ze(l))},async continueWith(c){let d=await s({action:"continue",...c?{stateUpdates:c}:{}},a);return p(ze(d))},async lastState(){return n?n.get(o):null}}}var H="text/html+skybridge",V="text/html;profile=mcp-app",Ye=async(e,t)=>{let n=e.endsWith("/")?e.slice(0,-1):e;return await(await fetch(`${n}${t}`)).text()};function Ze(e){return{"openai/widgetDescription":e.description,"openai/widgetPrefersBorder":e.prefersBorder,"openai/widgetDomain":e.widgetDomain,...e.widgetCSP&&{"openai/widgetCSP":e.widgetCSP}}}function Je(e){let t=e.widgetCSP?{connectDomains:e.widgetCSP.connect_domains,resourceDomains:e.widgetCSP.resource_domains,frameDomains:e.widgetCSP.frame_domains,redirectDomains:e.widgetCSP.redirect_domains}:void 0;return{ui:{...t&&{csp:t},...e.widgetDomain&&{domain:e.widgetDomain},...e.prefersBorder!==void 0&&{prefersBorder:e.prefersBorder}}}}function ce(e){return{...e.openaiTemplateUri&&{"openai/outputTemplate":e.openaiTemplateUri},"openai/toolInvocation/invoking":e.invoking,"openai/toolInvocation/invoked":e.invoked,"openai/widgetAccessible":!0,"openai/resultCanProduceWidget":!0,...e.mcpTemplateUri&&{ui:{resourceUri:e.mcpTemplateUri,...e.autoHeight&&{autoHeight:!0}}},...e.mcpTemplateUri&&{"ui/resourceUri":e.mcpTemplateUri}}}function Xe(e){let{id:t,title:n,description:r,baseUrl:o,htmlPath:i,widgetDomain:s,prefersBorder:a=!0,autoHeight:p=!0}=e,c=e.widgetCSP??{connect_domains:[o],resource_domains:[o]};if(process.env.NODE_ENV==="development")try{let{hostname:f}=new URL(o);(f==="localhost"||f==="127.0.0.1")&&(c={...c,connect_domains:[...c.connect_domains||[],`ws://${f}:*`,`wss://${f}:*`],resource_domains:[...c.resource_domains||[],`http://${f}:*`]})}catch{}let d=`ui://widgets/apps-sdk/${t}.html`,u=`ui://widgets/ext-apps/${t}.html`,l=null,h=()=>(l||(l=Ye(o,i)),l),g=r;async function R(f){let x=await h();f.registerResource(`${t}-openai-widget`,d,{title:n,description:g,mimeType:H,_meta:{"openai/widgetDescription":g,"openai/widgetPrefersBorder":a}},async T=>({contents:[{uri:T.href,mimeType:H,text:x,_meta:Ze({description:g,prefersBorder:a,widgetDomain:s,widgetCSP:c})}]})),f.registerResource(`${t}-mcp-widget`,u,{title:n,description:g,mimeType:V,_meta:{ui:{prefersBorder:a}}},async T=>({contents:[{uri:T.href,mimeType:V,text:x,_meta:Je({description:g,prefersBorder:a,widgetDomain:s,widgetCSP:c})}]}))}return{id:t,title:n,description:r,openaiUri:d,mcpUri:u,autoHeight:p,register:R}}function Wt(e,t){let{resource:n,description:r,inputSchema:o,annotations:i,autoInjectResultText:s=!0}=e,a=e.id??n?.id,p=e.title??n?.title;if(!a)throw new Error("createTool: `id` is required when no resource is provided");if(!p)throw new Error("createTool: `title` is required when no resource is provided");let c=n?ce({openaiTemplateUri:n.openaiUri,mcpTemplateUri:n.mcpUri,invoking:e.invoking??"Loading...",invoked:e.invoked??"Loaded",autoHeight:n.autoHeight}):void 0;return{id:a,title:p,description:r,async register(d){d.registerTool(a,{title:p,description:r,inputSchema:o,annotations:i,...c&&{_meta:c}},(async(u,l)=>{let h=l,g=h._meta??{},R=B(h),f=await t(u,{extra:{_meta:g},waniwani:R});return n&&f.data?{content:[{type:"text",text:f.text}],structuredContent:f.data,_meta:{...c,...g,...s===!1?{"waniwani/autoInjectResultText":!1}:{}}}:{content:[{type:"text",text:f.text}],...f.data?{structuredContent:f.data}:{},...s===!1?{_meta:{"waniwani/autoInjectResultText":!1}}:{}}}))}}}async function At(e,t){await Promise.all(t.map(n=>n.register(e)))}var z=class extends Error{constructor(n,r){super(n);this.status=r;this.name="WaniWaniError"}};var Mt="@waniwani/sdk";function Qe(e){let{apiUrl:t,apiKey:n}=e;function r(){if(!n)throw new Error("WANIWANI_API_KEY is not set");return n}async function o(i,s,a){let p=r(),c=`${t.replace(/\/$/,"")}${s}`,d={Authorization:`Bearer ${p}`,"X-WaniWani-SDK":Mt},u={method:i,headers:d};a!==void 0&&(d["Content-Type"]="application/json",u.body=JSON.stringify(a));let l=await fetch(c,u);if(!l.ok){let g=await l.text().catch(()=>"");throw new z(g||`KB API error: HTTP ${l.status}`,l.status)}return(await l.json()).data}return{async ingest(i){return o("POST","/api/mcp/kb/ingest",{files:i})},async search(i,s){return o("POST","/api/mcp/kb/search",{query:i,...s})},async sources(){return o("GET","/api/mcp/kb/sources")}}}var Nt="@waniwani/sdk";function Y(e,t={}){let n=t.now??(()=>new Date),r=t.generateId??et,o=Ot(e),i=G(e.meta),s=G(e.metadata),a=Kt(e,i),p=_(e.eventId)??r(),c=jt(e.timestamp,n),d=_(e.source)??O(i)??t.source??Nt,u=de(e)?{...e}:void 0,l={...s};return Object.keys(i).length>0&&(l.meta=i),u&&(l.rawLegacy=u),{id:p,type:"mcp.event",name:o,source:d,timestamp:c,correlation:a,properties:Ut(e,o),metadata:l,rawLegacy:u}}function et(){return typeof crypto<"u"&&typeof crypto.randomUUID=="function"?`evt_${crypto.randomUUID()}`:`evt_${Math.random().toString(36).slice(2,10)}_${Date.now().toString(36)}`}function Ut(e,t){if(!de(e))return G(e.properties);let n=Dt(e,t),r=G(e.properties);return{...n,...r}}function Dt(e,t){switch(t){case"tool.called":{let n={};return _(e.toolName)&&(n.name=e.toolName),_(e.toolType)&&(n.type=e.toolType),n}case"quote.succeeded":{let n={};return typeof e.quoteAmount=="number"&&(n.amount=e.quoteAmount),_(e.quoteCurrency)&&(n.currency=e.quoteCurrency),n}case"link.clicked":{let n={};return _(e.linkUrl)&&(n.url=e.linkUrl),n}case"purchase.completed":{let n={};return typeof e.purchaseAmount=="number"&&(n.amount=e.purchaseAmount),_(e.purchaseCurrency)&&(n.currency=e.purchaseCurrency),n}default:return{}}}function Ot(e){return de(e)?e.eventType:e.event}function Kt(e,t){let n=_(e.requestId)??_e(t),r=_(e.sessionId)??P(t),o=_(e.traceId)??Ce(t),i=_(e.externalUserId)??be(t),s=_(e.correlationId)??Pe(t)??n,a={};return r&&(a.sessionId=r),o&&(a.traceId=o),n&&(a.requestId=n),s&&(a.correlationId=s),i&&(a.externalUserId=i),a}function jt(e,t){if(e instanceof Date)return e.toISOString();if(typeof e=="string"){let n=new Date(e);if(!Number.isNaN(n.getTime()))return n.toISOString()}return t().toISOString()}function G(e){return!e||typeof e!="object"||Array.isArray(e)?{}:e}function _(e){if(typeof e=="string"&&e.trim().length!==0)return e}function de(e){return"eventType"in e}var $t="/api/mcp/events/v2/batch";var tt="@waniwani/sdk",Bt=new Set([401,403]),qt=new Set([408,425,429,500,502,503,504]);function nt(e){return new pe(e)}var pe=class{endpointUrl;flushIntervalMs;maxBatchSize;maxBufferSize;maxRetries;retryBaseDelayMs;retryMaxDelayMs;shutdownTimeoutMs;sdkVersion;fetchFn;logger;now;sleep;apiKey;buffer=[];flushTimer;flushScheduled=!1;flushScheduledTimer;flushInFlight;inFlightCount=0;isStopped=!1;isShuttingDown=!1;constructor(t){this.endpointUrl=Vt(t.apiUrl,t.endpointPath??$t),this.flushIntervalMs=t.flushIntervalMs??1e3,this.maxBatchSize=t.maxBatchSize??20,this.maxBufferSize=t.maxBufferSize??1e3,this.maxRetries=t.maxRetries??3,this.retryBaseDelayMs=t.retryBaseDelayMs??200,this.retryMaxDelayMs=t.retryMaxDelayMs??2e3,this.shutdownTimeoutMs=t.shutdownTimeoutMs??2e3,this.fetchFn=t.fetchFn??fetch,this.logger=t.logger??console,this.now=t.now??(()=>new Date),this.sleep=t.sleep??(n=>new Promise(r=>setTimeout(r,n))),this.apiKey=t.apiKey,this.sdkVersion=t.sdkVersion,this.flushIntervalMs>0&&(this.flushTimer=setInterval(()=>{this.flush()},this.flushIntervalMs))}enqueue(t){if(this.isStopped||this.isShuttingDown){this.logger.warn("[WaniWani] Tracking transport is stopped, dropping event %s",t.id);return}if(this.buffer.length>=this.maxBufferSize){let n=this.buffer.length-this.maxBufferSize+1;this.buffer.splice(0,n),this.logger.warn("[WaniWani] Tracking buffer overflow, dropped %d oldest event(s)",n)}if(this.buffer.push(t),this.buffer.length>=this.maxBatchSize){this.flush();return}this.scheduleMicroFlush()}pendingEvents(){return this.buffer.length+this.inFlightCount}async flush(){return this.flushInFlight?this.flushInFlight:(this.flushInFlight=this.flushLoop().finally(()=>{this.flushInFlight=void 0}),this.flushInFlight)}async shutdown(t){this.isShuttingDown=!0,this.flushTimer&&(clearInterval(this.flushTimer),this.flushTimer=void 0),this.flushScheduledTimer&&(clearTimeout(this.flushScheduledTimer),this.flushScheduledTimer=void 0,this.flushScheduled=!1);let n=t?.timeoutMs??this.shutdownTimeoutMs,r=this.flush();if(!Number.isFinite(n)||n<=0)return await r,this.isStopped=!0,{timedOut:!1,pendingEvents:this.pendingEvents()};let o=Symbol("shutdown-timeout");return await Promise.race([r.then(()=>"flushed"),this.sleep(n).then(()=>o)])===o?(this.isStopped=!0,{timedOut:!0,pendingEvents:this.pendingEvents()}):(this.isStopped=!0,{timedOut:!1,pendingEvents:this.pendingEvents()})}scheduleMicroFlush(){this.flushScheduled||(this.flushScheduled=!0,this.flushScheduledTimer=setTimeout(()=>{this.flushScheduledTimer=void 0,this.flushScheduled=!1,this.flush()},0))}async flushLoop(){for(;this.buffer.length>0&&!this.isStopped;){let t=this.buffer.splice(0,this.maxBatchSize);await this.sendBatchWithRetry(t)}}async sendBatchWithRetry(t){let n=0,r=t;for(;r.length>0&&!this.isStopped;){this.inFlightCount=r.length;let o=await this.sendBatchOnce(r);switch(this.inFlightCount=0,o.kind){case"success":return;case"auth":this.stopTransportForAuthFailure(o.status,r.length);return;case"permanent":this.logger.error("[WaniWani] Dropping %d event(s) after permanent failure: %s",r.length,o.reason);return;case"retryable":if(n>=this.maxRetries){this.logger.error("[WaniWani] Dropping %d event(s) after retry exhaustion: %s",r.length,o.reason);return}await this.sleep(this.backoffDelayMs(n)),n+=1;continue;case"partial":if(o.permanent.length>0&&this.logger.error("[WaniWani] Dropping %d event(s) rejected as permanent",o.permanent.length),o.retryable.length===0)return;if(n>=this.maxRetries){this.logger.error("[WaniWani] Dropping %d retryable event(s) after retry exhaustion",o.retryable.length);return}r=o.retryable,await this.sleep(this.backoffDelayMs(n)),n+=1;continue}}}async sendBatchOnce(t){let n;try{n=await this.fetchFn(this.endpointUrl,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${this.apiKey}`,"X-WaniWani-SDK":tt},body:JSON.stringify(this.makeBatchRequest(t))})}catch(i){return{kind:"retryable",reason:zt(i)}}if(Bt.has(n.status))return{kind:"auth",status:n.status};if(qt.has(n.status))return{kind:"retryable",reason:`HTTP ${n.status}`};if(!n.ok)return{kind:"permanent",reason:`HTTP ${n.status}`};let r=await Ht(n);if(!r?.rejected||r.rejected.length===0)return{kind:"success"};let o=this.classifyRejectedEvents(t,r.rejected);return o.retryable.length===0&&o.permanent.length===0?{kind:"success"}:{kind:"partial",retryable:o.retryable,permanent:o.permanent}}makeBatchRequest(t){return{sentAt:this.now().toISOString(),source:{sdk:tt,version:this.sdkVersion??"0.0.0"},events:t}}classifyRejectedEvents(t,n){let r=new Map(t.map(s=>[s.id,s])),o=[],i=[];for(let s of n){let a=r.get(s.eventId);if(a){if(Lt(s)){o.push(a);continue}i.push(a)}}return{retryable:o,permanent:i}}backoffDelayMs(t){let n=this.retryBaseDelayMs*2**t;return Math.min(n,this.retryMaxDelayMs)}stopTransportForAuthFailure(t,n){this.isStopped=!0;let r=this.buffer.length;this.buffer.splice(0,r),this.logger.error("[WaniWani] Auth failure (HTTP %d). Stopping tracking transport and dropping %d queued event(s)",t,n+r)}};function Lt(e){if(e.retryable===!0)return!0;let t=e.code.toLowerCase();return t.includes("timeout")||t.includes("temporary")||t.includes("unavailable")||t.includes("rate_limit")||t.includes("transient")||t.includes("server")}async function Ht(e){let t=await e.text();if(t)try{return JSON.parse(t)}catch{return}}function Vt(e,t){let n=e.endsWith("/")?e:`${e}/`,r=t.startsWith("/")?t.slice(1):t;return`${n}${r}`}function zt(e){return e instanceof Error?e.message:String(e)}function rt(e){let{apiUrl:t,apiKey:n,tracking:r}=e;function o(){if(!n)throw new Error("WANIWANI_API_KEY is not set");return n}let i=n?nt({apiUrl:t,apiKey:n,endpointPath:r.endpointPath,flushIntervalMs:r.flushIntervalMs,maxBatchSize:r.maxBatchSize,maxBufferSize:r.maxBufferSize,maxRetries:r.maxRetries,retryBaseDelayMs:r.retryBaseDelayMs,retryMaxDelayMs:r.retryMaxDelayMs,shutdownTimeoutMs:r.shutdownTimeoutMs}):void 0,s={async identify(a,p,c){o();let d=Y({event:"user.identified",externalUserId:a,properties:p,meta:c});return i?.enqueue(d),{eventId:d.id}},async track(a){o();let p=Y(a);return i?.enqueue(p),{eventId:p.id}},async flush(){o(),await i?.flush()},async shutdown(a){return o(),await i?.shutdown({timeoutMs:a?.timeoutMs??r.shutdownTimeoutMs})??{timedOut:!1,pendingEvents:0}}};return i&&Gt(s,r.shutdownTimeoutMs),s}function Gt(e,t){if(typeof process>"u"||typeof process.once!="function"||typeof process.on!="function")return;let n=()=>{e.shutdown({timeoutMs:t})};process.once("beforeExit",n),process.once("SIGINT",n),process.once("SIGTERM",n)}function Z(e){let t=e,n=t?.apiUrl??"https://app.waniwani.ai",r=t?.apiKey??process.env.WANIWANI_API_KEY,o={endpointPath:t?.tracking?.endpointPath??"/api/mcp/events/v2/batch",flushIntervalMs:t?.tracking?.flushIntervalMs??1e3,maxBatchSize:t?.tracking?.maxBatchSize??20,maxBufferSize:t?.tracking?.maxBufferSize??1e3,maxRetries:t?.tracking?.maxRetries??3,retryBaseDelayMs:t?.tracking?.retryBaseDelayMs??200,retryMaxDelayMs:t?.tracking?.retryMaxDelayMs??2e3,shutdownTimeoutMs:t?.tracking?.shutdownTimeoutMs??2e3},i={apiUrl:n,apiKey:r,tracking:o},s=rt(i),a=Qe(i);return{...s,kb:a,_config:i}}function Yt(e){let t=e.event_type??"widget_click",r=t.startsWith("widget_")?t:`widget_${t}`,o={...e.metadata??{}};return e.event_name&&(o.event_name=e.event_name),{event:r,properties:o,sessionId:e.session_id,traceId:e.trace_id,externalUserId:e.user_id,eventId:e.event_id,timestamp:e.timestamp,source:e.source??"widget"}}function Zt(e){let t={apiKey:e?.apiKey,apiUrl:e?.apiUrl},n;function r(){return n||(n=Z(t)),n}return async function(i){let s;try{s=await i.json()}catch{return new Response(JSON.stringify({error:"Invalid JSON"}),{status:400,headers:{"Content-Type":"application/json"}})}if(!Array.isArray(s.events)||s.events.length===0)return new Response(JSON.stringify({error:"Missing or empty events array"}),{status:400,headers:{"Content-Type":"application/json"}});try{let a=r(),p=[];for(let c of s.events){let d=Yt(c),u=await a.track(d);p.push(u.eventId)}return await a.flush(),new Response(JSON.stringify({ok:!0,accepted:p.length}),{status:200,headers:{"Content-Type":"application/json"}})}catch(a){let p=a instanceof Error?a.message:"Unknown error";return new Response(JSON.stringify({error:p}),{status:500,headers:{"Content-Type":"application/json"}})}}}function J(e,t){return t?(...n)=>console.log(`[waniwani:${e}]`,...n):()=>{}}var ue=J("widget-token",!!process.env.WANIWANI_DEBUG),Jt=120*1e3,X=class{cached=null;pending=null;config;constructor(t){this.config=t}async getToken(t,n){return this.cached&&Date.now()<this.cached.expiresAt-Jt?this.cached.token:this.pending?this.pending:(this.pending=this.mint(t,n).finally(()=>{this.pending=null}),this.pending)}async mint(t,n){let r=Xt(this.config.apiUrl,"/api/mcp/widget-tokens");ue("minting token from",r);let o={};t&&(o.sessionId=t),n&&(o.traceId=n);try{let i=await fetch(r,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${this.config.apiKey}`},body:JSON.stringify(o),signal:AbortSignal.timeout(5e3)});if(ue("mint response:",i.status),!i.ok)return null;let s=await i.json(),a=s.data&&typeof s.data=="object"?s.data:s,p=new Date(a.expiresAt).getTime();return!a.token||Number.isNaN(p)?null:(this.cached={token:a.token,expiresAt:p},a.token)}catch(i){return ue("mint failed:",i),null}}};function Xt(e,t){return`${e.endsWith("/")?e.slice(0,-1):e}${t}`}async function Qt(e){let t=JSON.stringify({nodes:e.nodes,edges:e.edges});if(typeof globalThis.crypto?.subtle?.digest=="function"){let r=await crypto.subtle.digest("SHA-256",new TextEncoder().encode(t));return Array.from(new Uint8Array(r)).map(o=>o.toString(16).padStart(2,"0")).join("")}let n=0;for(let r=0;r<t.length;r++)n=(n<<5)-n+t.charCodeAt(r)|0;return`simple-${Math.abs(n).toString(36)}`}async function ot(e,t,n){if(e.length===0)return;let r=await Promise.all(e.map(async i=>({...i,configHash:await Qt(i)}))),o=JSON.stringify({flows:r});try{let i=`${t}/api/mcp/funnel/sync`;fetch(i,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${n}`},body:o}).catch(()=>{})}catch{}}var it="waniwani/sessionId",Q="waniwani/geoLocation",ee="waniwani/userLocation",en="openai/userLocation",tn=[en,Q,ee];function st(e,t){if(t.length===0)return e;let n;for(let r of tn){let o=e[r];if(!S(o))continue;let i;for(let s of t)s in o&&(i||(i={...o}),delete i[s]);i&&(n||(n={...e}),n[r]=i)}return n??e}function S(e){return!!e&&typeof e=="object"&&!Array.isArray(e)}function $(e){if(!S(e))return;let t=e._meta;return S(t)?t:void 0}function le(e){if(!S(e))return;let t=e.content;return Array.isArray(t)?t.find(r=>S(r)&&r.type==="text"&&typeof r.text=="string")?.text:void 0}function nn(e,t){return typeof t=="function"?t(e)??"other":t??"other"}function fe(e,t,n,r,o,i){let s=nn(e,n.toolType),a=n.stripLocationFields,p=a&&a.length>0,c=$(t),d=c&&p?st(c,a):c,u=i?.input!==void 0&&n.redactInput?n.redactInput(i.input):i?.input,l=p&&S(i?.output)&&S(i.output._meta)?{...i.output,_meta:st(i.output._meta,a)}:i?.output;return{event:"tool.called",properties:{name:e,type:s,...r??{},...u!==void 0&&{input:u},...l!==void 0&&{output:l}},meta:d,source:O(d),metadata:{...n.metadata??{},...o&&{clientInfo:o}}}}async function ge(e,t,n){try{await e.track(t)}catch(r){n?.(me(r))}}async function we(e,t){try{await e.flush()}catch(n){t?.(me(n))}}async function ct(e,t,n,r,o){if(!S(e))return;S(e._meta)||(e._meta={});let i=e._meta,s=S(i.waniwani)?i.waniwani:void 0,a={...s??{},endpoint:s?.endpoint??`${n.replace(/\/$/,"")}/api/mcp/events/v2/batch`};if(t)try{let u=await t.getToken();u&&(a.token=u)}catch(u){o?.(me(u))}let p=P(i);p&&(a.sessionId||(a.sessionId=p));let c=ut(i);c!==void 0&&(a.geoLocation||(a.geoLocation=c));let d=O($(r));d&&!a.source&&(a.source=d),i.waniwani=a}var at=["openai/outputTemplate","openai/widgetAccessible","openai/resultCanProduceWidget","openai/toolInvocation/invoking","openai/toolInvocation/invoked","ui/resourceUri","ui"];function dt(e,t){if(!t||!S(e))return;let n=!1;for(let o of at)if(o in t){n=!0;break}if(!n)return;S(e._meta)||(e._meta={});let r=e._meta;for(let o of at)o in t&&(o in r||(r[o]=t[o]))}function pt(e,t){let n=$(t);if(!n||!S(e))return;S(e._meta)||(e._meta={});let r=e._meta,o=P(n);o&&!r[it]&&(r[it]=o);let i=ut(n);i&&(r[Q]||(r[Q]=i),r[ee]||(r[ee]=i))}function ut(e){if(!e)return;let t=e[Q]??e[ee];if(S(t)||typeof t=="string")return t}function me(e){return e instanceof Error?e:new Error(String(e))}function lt(e){return typeof e=="object"&&e!==null&&!Array.isArray(e)}function ft(e){if(typeof e.sessionId=="string"&&e.sessionId)return e.sessionId;if(lt(e.requestInfo)){let t=e.requestInfo.headers;if(lt(t)){let n={};for(let o of Object.keys(t))n[o.toLowerCase()]=t[o];let r=n["mcp-session-id"];if(typeof r=="string"&&r)return r}}}var wt=Symbol.for("waniwani.wrappedHandler"),te=J("mcp",!!process.env.WANIWANI_DEBUG),he="https://app.waniwani.ai",rn="REDACTED";function on(e){if(!e)return;let t=e[L];if(!Array.isArray(t)||t.length===0)return;let n=new Set(t.filter(r=>typeof r=="string"));if(n.size!==0)return r=>{if(!S(r))return r;let o=r.stateUpdates;if(!S(o))return r;let i=!1,s={...o};for(let a of n)a in s&&(s[a]=rn,i=!0);return i?{...r,stateUpdates:s}:r}}function gt(e,t,n,r){let{server:o,tracker:i,opts:s,tokenCache:a,injectToken:p}=n,c=s.applyFieldRedactions===!0?on(r):void 0,d=c?{...s,redactInput:c}:s,u=async(l,h)=>{let g=$(h)??{};if(!P(g)&&S(h)){let y=ft(h);y&&(g["waniwani/sessionId"]=y,h._meta=g)}let f=Ie(i,g,{apiUrl:i._config.apiUrl,apiKey:i._config.apiKey});S(h)&&(h[re]=f);let x=performance.now(),T=o.server?.getClientVersion?.();try{let y=await t(l,h),k=Math.round(performance.now()-x);te(`tool "${e}" handler returned in ${k}ms, running post-processing...`);let E=S(y)&&y.isError===!0;if(E){let v=le(y);console.error(`[waniwani] Tool "${e}" returned error${v?`: ${v}`:""}`)}return await ge(i,fe(e,h,d,{durationMs:k,status:E?"error":"ok",...E&&{errorMessage:le(y)??"Unknown tool error"}},T,{input:l,output:y}),s.onError),te(`tool "${e}" tracking done`),s.flushAfterToolCall&&await we(i,s.onError),pt(y,h),dt(y,r),p&&(await ct(y,a,i._config.apiUrl??he,h,s.onError),te(`tool "${e}" widget config injected`)),te(`tool "${e}" post-processing complete, returning result`),y}catch(y){let k=Math.round(performance.now()-x);throw await ge(i,fe(e,h,d,{durationMs:k,status:"error",errorMessage:y instanceof Error?y.message:String(y)},T,{input:l}),s.onError),s.flushAfterToolCall&&await we(i,s.onError),y}};return u[wt]=!0,u}function sn(e,t){let n=e;if(n.__waniwaniWrapped)return n;n.__waniwaniWrapped=!0;let r=t??{},o=r.client??Z(),i=r.injectWidgetToken!==!1,s=o._config.apiKey?new X({apiUrl:o._config.apiUrl??he,apiKey:o._config.apiKey}):null,a={server:e,tracker:o,opts:r,tokenCache:s,injectToken:i},p=e.registerTool.bind(e);n.registerTool=((...d)=>{let[u,l,h]=d;if(typeof h!="function")return p(...d);let g=typeof u=="string"&&u.trim().length>0?u:"unknown",R=S(l)&&S(l._meta)?l._meta:void 0,f=gt(g,h,a,R);return p(u,l,f)});let c=e._registeredTools;if(S(c))for(let[d,u]of Object.entries(c)){if(!S(u))continue;let l=u.handler;if(typeof l!="function"||l[wt])continue;let h=S(u._meta)?u._meta:void 0;u.handler=gt(d,l,a,h)}if(o._config.apiKey){let d=e._registeredTools,u=[];if(d&&typeof d=="object"){for(let l of Object.values(d))if(l&&typeof l=="object"){let h=l._meta,g=h&&typeof h=="object"?h._flowGraph:void 0;g?.nodes?.length&&u.push(g)}}u.length>0&&ot(u,o._config.apiUrl??he,o._config.apiKey)}return n}export{C as END,b as START,N as StateGraph,A as WaniwaniKvStore,Ve as createFlow,Ge as createFlowTestHarness,Xe as createResource,Wt as createTool,Zt as createTrackingRoute,ne as detectPlatform,ht as isMCPApps,mt as isOpenAI,$e as redacted,At as registerTools,sn as withWaniwani};
7
7
  //# sourceMappingURL=index.js.map