@waniwani/sdk 0.4.8-beta.4 → 0.4.9-beta.1

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.
@@ -0,0 +1,166 @@
1
+ import { UIMessage } from 'ai';
2
+
3
+ interface ToolCallTrace {
4
+ name: string;
5
+ input: Record<string, unknown>;
6
+ output: unknown;
7
+ }
8
+ interface ChatResult {
9
+ output: string;
10
+ toolsCalled: string[];
11
+ toolCallTraces: ToolCallTrace[];
12
+ }
13
+ /** A single turn in a multi-turn conversation. */
14
+ interface ConversationTurn {
15
+ input: string;
16
+ }
17
+ /** Assertion result for a single tool call check within a turn. */
18
+ interface TurnAssertion {
19
+ /** Whether the actual tool calls matched the expected ones. */
20
+ passed: boolean;
21
+ /** Expected tool names recorded in the session. */
22
+ expected: string[];
23
+ /** Actual tool names called during replay. */
24
+ actual: string[];
25
+ }
26
+ /** Result of a single turn. */
27
+ interface ConversationTurnResult {
28
+ input: string;
29
+ response: ChatResult;
30
+ /** Auto-derived assertion results comparing actual vs. recorded tool calls. */
31
+ assertions: TurnAssertion[];
32
+ }
33
+ /** Full result of a multi-turn conversation. */
34
+ interface ConversationResult {
35
+ turns: ConversationTurnResult[];
36
+ }
37
+ /**
38
+ * A recorded conversation session that can be replayed as a test.
39
+ * Messages use the AI SDK's UIMessage format — same as what the
40
+ * Export button in the chatbar produces.
41
+ *
42
+ * Two replay modes:
43
+ * - **"regenerate"** (default): Only user messages are sent.
44
+ * The LLM generates fresh responses at each turn. Assertions are
45
+ * auto-derived by comparing actual tool calls to the recorded ones.
46
+ *
47
+ * - **"inject"**: Prior turns are injected as-is, only the
48
+ * final user message gets a fresh LLM response.
49
+ */
50
+ interface SessionReplay {
51
+ name: string;
52
+ messages: UIMessage[];
53
+ mode?: "regenerate" | "inject";
54
+ /**
55
+ * Optional session-level outcome assertion.
56
+ * Checks that all listed tools were called at least once across the session.
57
+ */
58
+ outcome?: {
59
+ toolsCalled: string[];
60
+ };
61
+ }
62
+
63
+ /**
64
+ * Send a single user message to a WaniWani MCP chat endpoint.
65
+ */
66
+ declare function chat(url: string, message: string): Promise<ChatResult>;
67
+ /**
68
+ * Run a multi-turn conversation. Returns the result of each turn.
69
+ */
70
+ declare function conversation(url: string, turns: ConversationTurn[]): Promise<ConversationResult>;
71
+ /**
72
+ * Replay a recorded conversation session (exported from the chatbar debug button).
73
+ * Uses UIMessage[] directly — same format as useChat's messages array.
74
+ *
75
+ * **"regenerate" mode** (default):
76
+ * Sends only user messages. The LLM generates fresh responses.
77
+ * Per-turn assertions are auto-derived by comparing actual tool calls
78
+ * to the tool calls recorded in the session.
79
+ *
80
+ * **"inject" mode**:
81
+ * Injects the recorded conversation as-is, only generates a fresh
82
+ * response for the final user message.
83
+ */
84
+ declare function replaySession(url: string, session: SessionReplay): Promise<ConversationResult>;
85
+
86
+ /**
87
+ * Create a local Braintrust reporter that writes JSON results to a directory and
88
+ * prints a summary to console.
89
+ *
90
+ * Requires the `braintrust` package: bun add -d braintrust
91
+ *
92
+ * @param outputDir - Directory to write JSON result files (default: "evals/runs")
93
+ */
94
+ declare function createLocalReporter(outputDir?: string): unknown;
95
+
96
+ /**
97
+ * Parse the JSON-stringified ChatResult from a Braintrust task output.
98
+ */
99
+ declare function parseTaskOutput(output: unknown): ChatResult;
100
+ /**
101
+ * Checks whether the expected tool was called.
102
+ * Looks for the tool name in `metadata.expectedTool` first (for cases where `expected` is a
103
+ * reference answer), then falls back to `expected` directly.
104
+ */
105
+ declare function calledExpectedTool({ output, expected, metadata, }: {
106
+ output: unknown;
107
+ expected?: unknown;
108
+ metadata?: Record<string, unknown>;
109
+ }): {
110
+ name: string;
111
+ score: number;
112
+ metadata: {
113
+ expected: string;
114
+ actual: string[];
115
+ };
116
+ };
117
+ /**
118
+ * Checks whether the assistant produced any text output.
119
+ */
120
+ declare function hasOutput({ output }: {
121
+ output: unknown;
122
+ }): {
123
+ name: string;
124
+ score: number;
125
+ };
126
+ /**
127
+ * Checks specific fields in the first tool call's `stateUpdates` against expected values.
128
+ * Supports nested fields via dot notation (e.g. "mixedBreed.knowsBreeds").
129
+ * Returns partial credit (fraction of matching fields).
130
+ */
131
+ declare function toolInputFieldsMatch({ output, metadata, }: {
132
+ output: unknown;
133
+ metadata?: Record<string, unknown>;
134
+ }): {
135
+ name: string;
136
+ score: number;
137
+ metadata?: undefined;
138
+ } | {
139
+ name: string;
140
+ score: number;
141
+ metadata: Record<string, {
142
+ expected: unknown;
143
+ actual: unknown;
144
+ match: boolean;
145
+ }>;
146
+ };
147
+ /** ClosedQA — checks if the answer correctly addresses the question given a reference answer. */
148
+ declare const FaqAccuracy: (args: {
149
+ input: unknown;
150
+ output: unknown;
151
+ expected?: unknown;
152
+ }) => Promise<unknown>;
153
+ /** Factuality — checks if the output is factually consistent with the expected output. */
154
+ declare const OutputFactuality: (args: {
155
+ input: unknown;
156
+ output: unknown;
157
+ expected?: unknown;
158
+ }) => Promise<unknown>;
159
+ /** Moderation — flags unsafe or inappropriate content. */
160
+ declare const SafetyCheck: (args: {
161
+ input: unknown;
162
+ output: unknown;
163
+ expected?: unknown;
164
+ }) => Promise<unknown>;
165
+
166
+ export { type ChatResult, type ConversationResult, type ConversationTurn, type ConversationTurnResult, FaqAccuracy, OutputFactuality, SafetyCheck, type SessionReplay, type ToolCallTrace, type TurnAssertion, calledExpectedTool, chat, conversation, createLocalReporter, hasOutput, parseTaskOutput, replaySession, toolInputFieldsMatch };
@@ -0,0 +1,8 @@
1
+ var S=(t=>typeof require<"u"?require:typeof Proxy<"u"?new Proxy(t,{get:(e,n)=>(typeof require<"u"?require:e)[n]}):t)(function(t){if(typeof require<"u")return require.apply(this,arguments);throw Error('Dynamic require of "'+t+'" is not supported')});import{parseJsonEventStream as v,readUIMessageStream as U,uiMessageChunkSchema as I}from"ai";function T(t){let e=t.parts.filter(s=>s.type==="text").map(s=>s.text).join(""),n=t.parts.filter(s=>s.type.startsWith("tool-")||s.type==="dynamic-tool").map(s=>s),o=n.map(s=>s.toolName),r=n.map(s=>({name:s.toolName,input:s.input??{},output:s.output}));return{output:e,toolsCalled:o,toolCallTraces:r}}function C(t){return t.parts.filter(e=>e.type==="text").map(e=>e.text).join("")}function b(t){return t.parts.filter(e=>e.type==="dynamic-tool"||e.type.startsWith("tool-")).map(e=>e.toolName).filter(Boolean)}async function y(t,e){let n=await fetch(`${t}/api/waniwani`,{method:"POST",headers:{"Content-Type":"application/json"},signal:AbortSignal.timeout(6e4),body:JSON.stringify({messages:e})});if(!n.ok)throw new Error(`Chat returned ${n.status}: ${await n.text()}`);if(!n.body)throw new Error("Chat response has no body");let o=v({stream:n.body,schema:I}).pipeThrough(new TransformStream({transform(s,a){s.success&&a.enqueue(s.value)}})),r;for await(let s of U({stream:o}))r=s;if(!r)throw new Error("No message received from stream");return{result:T(r),message:r}}async function O(t,e){let n={id:crypto.randomUUID(),role:"user",parts:[{type:"text",text:e}]},{result:o}=await y(t,[n]);return o}async function $(t,e){let n=[],o=[];for(let r of e){n.push({id:crypto.randomUUID(),role:"user",parts:[{type:"text",text:r.input}]});let{result:s,message:a}=await y(t,n);n.push(a),o.push({input:r.input,response:s,assertions:[]})}return{turns:o}}async function N(t,e){let n=e.mode??"regenerate",o=[],r=[],s=[];for(let a=0;a<e.messages.length;a++){let l=e.messages[a];if(l.role==="user"){let u=e.messages[a+1];s.push({userMsg:l,assistantMsg:u?.role==="assistant"?u:void 0})}}for(let a=0;a<s.length;a++){let{userMsg:l,assistantMsg:u}=s[a],c=a===s.length-1,p=u?b(u):[];if(o.push(l),n==="inject"&&!c&&u){o.push(u);let f=T(u),M=R(p,f.toolsCalled);r.push({input:C(l),response:f,assertions:M});continue}let{result:i,message:g}=await y(t,o);o.push(g);let m=R(p,i.toolsCalled);r.push({input:C(l),response:i,assertions:m})}return{turns:r}}function R(t,e){if(t.length===0)return[];let n=new Set(e);return[...new Set(t)].map(r=>({passed:n.has(r),expected:[r],actual:e}))}import{existsSync as j,mkdirSync as J,writeFileSync as L}from"fs";function d(t){try{return JSON.parse(t)}catch{return{output:"",toolsCalled:[],toolCallTraces:[]}}}function F({output:t,expected:e,metadata:n}){let o=d(t),r=n?.expectedTool??e;return{name:"called_expected_tool",score:o.toolsCalled.includes(r)?1:0,metadata:{expected:r,actual:o.toolsCalled}}}function A({output:t}){return{name:"has_output",score:d(t).output.length>0?1:0}}function E({output:t,metadata:e}){let n=d(t),o=e?.expectedFields??{},r=Object.keys(o);if(r.length===0)return{name:"field_extraction",score:1};let a=n.toolCallTraces[0]?.input?.stateUpdates??{},l=0,u={};for(let c of r){let p=o[c],i;if(c.includes(".")){let[m,f]=c.split(".");i=a[m]?.[f]}else i=a[c];let g=JSON.stringify(i)===JSON.stringify(p);g&&l++,u[c]={expected:p,actual:i,match:g}}return{name:"field_extraction",score:l/r.length,metadata:u}}function x(t){return async e=>{let n=d(e.output);return t({input:e.input,output:n.output,expected:e.expected})}}async function k(t){return(await import("autoevals").catch(()=>{throw new Error(`LLM scorer "${t}" requires the "autoevals" package: bun add -d autoevals`)}))[t]}var q=async t=>x(await k("ClosedQA"))(t),P=async t=>x(await k("Factuality"))(t),_=async t=>x(await k("Moderation"))(t);function W(t="evals/runs"){let e;try{e=S("braintrust").Reporter}catch{throw new Error('Local reporter requires the "braintrust" package: bun add -d braintrust')}return j(t)||J(t,{recursive:!0}),e("local",{reportEval(n,o){let r=n.experimentName??"unknown",s=new Date().toISOString().replace(/[:.]/g,"-"),a=o.results.map(c=>{let p=c.scores??{},i=d(c.output);return{input:c.input,output:i.output,toolsCalled:i.toolsCalled,toolCallTraces:i.toolCallTraces,scores:p}}),l=`${t}/${r}-${s}.json`;L(l,JSON.stringify(a,null,2)),console.log(`
2
+ \u{1F4CA} ${r} (${o.results.length} cases):
3
+ `);let u=0;for(let c of o.results){let p=c.scores??{},i=p.called_expected_tool===1;i||u++,console.log(` ${i?"\u2705":"\u274C"} ${c.input.slice(0,70)}`);for(let[g,m]of Object.entries(p))console.log(` ${g}: ${m}`)}return console.log(`
4
+ ${o.results.length-u}/${o.results.length} passed`),console.log(` \u2192 ${l}
5
+ `),u===0},reportRun(n){let o=n.every(r=>r===!0);return console.log(o?`
6
+ \u2705 All experiments passed`:`
7
+ \u274C Some experiments failed`),o}})}export{q as FaqAccuracy,P as OutputFactuality,_ as SafetyCheck,F as calledExpectedTool,O as chat,$ as conversation,W as createLocalReporter,A as hasOutput,d as parseTaskOutput,N as replaySession,E as toolInputFieldsMatch};
8
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/evals/chat.ts","../../src/evals/reporter.ts","../../src/evals/scorers.ts"],"sourcesContent":["import {\n\tparseJsonEventStream,\n\treadUIMessageStream,\n\ttype UIMessage,\n\tuiMessageChunkSchema,\n} from \"ai\";\nimport type {\n\tChatResult,\n\tConversationResult,\n\tConversationTurn,\n\tConversationTurnResult,\n\tSessionReplay,\n\tToolCallTrace,\n\tTurnAssertion,\n} from \"./types\";\n\n// --- Internal helpers ---\n\nfunction parseUIMessage(msg: UIMessage): ChatResult {\n\tconst output = msg.parts\n\t\t.filter((p): p is { type: \"text\"; text: string } => p.type === \"text\")\n\t\t.map((p) => p.text)\n\t\t.join(\"\");\n\n\tconst toolParts = msg.parts\n\t\t.filter((p) => p.type.startsWith(\"tool-\") || p.type === \"dynamic-tool\")\n\t\t.map(\n\t\t\t(p) =>\n\t\t\t\tp as unknown as {\n\t\t\t\t\ttoolName: string;\n\t\t\t\t\tinput?: Record<string, unknown>;\n\t\t\t\t\toutput?: unknown;\n\t\t\t\t},\n\t\t);\n\tconst toolsCalled = toolParts.map((p) => p.toolName);\n\tconst toolCallTraces: ToolCallTrace[] = toolParts.map((p) => ({\n\t\tname: p.toolName,\n\t\tinput: p.input ?? {},\n\t\toutput: p.output,\n\t}));\n\n\treturn { output, toolsCalled, toolCallTraces };\n}\n\nfunction textFromUIMessage(msg: UIMessage): string {\n\treturn msg.parts\n\t\t.filter((p): p is { type: \"text\"; text: string } => p.type === \"text\")\n\t\t.map((p) => p.text)\n\t\t.join(\"\");\n}\n\n/** Extract the tool names called in a recorded assistant UIMessage. */\nfunction extractRecordedTools(msg: UIMessage): string[] {\n\treturn msg.parts\n\t\t.filter((p) => p.type === \"dynamic-tool\" || p.type.startsWith(\"tool-\"))\n\t\t.map((p) => (p as unknown as { toolName: string }).toolName)\n\t\t.filter(Boolean);\n}\n\nasync function sendMessages(\n\turl: string,\n\tmessages: UIMessage[],\n): Promise<{ result: ChatResult; message: UIMessage }> {\n\tconst response = await fetch(`${url}/api/waniwani`, {\n\t\tmethod: \"POST\",\n\t\theaders: { \"Content-Type\": \"application/json\" },\n\t\tsignal: AbortSignal.timeout(60_000),\n\t\tbody: JSON.stringify({ messages }),\n\t});\n\n\tif (!response.ok) {\n\t\tthrow new Error(\n\t\t\t`Chat returned ${response.status}: ${await response.text()}`,\n\t\t);\n\t}\n\n\tif (!response.body) {\n\t\tthrow new Error(\"Chat response has no body\");\n\t}\n\n\tconst chunkStream = parseJsonEventStream({\n\t\tstream: response.body,\n\t\tschema: uiMessageChunkSchema,\n\t}).pipeThrough(\n\t\tnew TransformStream({\n\t\t\ttransform(chunk, controller) {\n\t\t\t\tif (chunk.success) {\n\t\t\t\t\tcontroller.enqueue(chunk.value);\n\t\t\t\t}\n\t\t\t},\n\t\t}),\n\t);\n\n\tlet finalMessage: UIMessage | undefined;\n\tfor await (const msg of readUIMessageStream({ stream: chunkStream })) {\n\t\tfinalMessage = msg;\n\t}\n\n\tif (!finalMessage) {\n\t\tthrow new Error(\"No message received from stream\");\n\t}\n\n\treturn { result: parseUIMessage(finalMessage), message: finalMessage };\n}\n\n// --- Public API ---\n\n/**\n * Send a single user message to a WaniWani MCP chat endpoint.\n */\nexport async function chat(url: string, message: string): Promise<ChatResult> {\n\tconst userMsg: UIMessage = {\n\t\tid: crypto.randomUUID(),\n\t\trole: \"user\",\n\t\tparts: [{ type: \"text\", text: message }],\n\t};\n\tconst { result } = await sendMessages(url, [userMsg]);\n\treturn result;\n}\n\n/**\n * Run a multi-turn conversation. Returns the result of each turn.\n */\nexport async function conversation(\n\turl: string,\n\tturns: ConversationTurn[],\n): Promise<ConversationResult> {\n\tconst history: UIMessage[] = [];\n\tconst turnResults: ConversationTurnResult[] = [];\n\n\tfor (const turn of turns) {\n\t\thistory.push({\n\t\t\tid: crypto.randomUUID(),\n\t\t\trole: \"user\",\n\t\t\tparts: [{ type: \"text\", text: turn.input }],\n\t\t});\n\n\t\tconst { result, message } = await sendMessages(url, history);\n\t\thistory.push(message);\n\n\t\tturnResults.push({ input: turn.input, response: result, assertions: [] });\n\t}\n\n\treturn { turns: turnResults };\n}\n\n/**\n * Replay a recorded conversation session (exported from the chatbar debug button).\n * Uses UIMessage[] directly — same format as useChat's messages array.\n *\n * **\"regenerate\" mode** (default):\n * Sends only user messages. The LLM generates fresh responses.\n * Per-turn assertions are auto-derived by comparing actual tool calls\n * to the tool calls recorded in the session.\n *\n * **\"inject\" mode**:\n * Injects the recorded conversation as-is, only generates a fresh\n * response for the final user message.\n */\nexport async function replaySession(\n\turl: string,\n\tsession: SessionReplay,\n): Promise<ConversationResult> {\n\tconst mode = session.mode ?? \"regenerate\";\n\tconst history: UIMessage[] = [];\n\tconst turnResults: ConversationTurnResult[] = [];\n\n\t// Pair user messages with their assistant responses\n\tconst userTurns: { userMsg: UIMessage; assistantMsg?: UIMessage }[] = [];\n\tfor (let i = 0; i < session.messages.length; i++) {\n\t\tconst msg = session.messages[i];\n\t\tif (msg.role === \"user\") {\n\t\t\tconst next = session.messages[i + 1];\n\t\t\tuserTurns.push({\n\t\t\t\tuserMsg: msg,\n\t\t\t\tassistantMsg: next?.role === \"assistant\" ? next : undefined,\n\t\t\t});\n\t\t}\n\t}\n\n\tfor (let turnIdx = 0; turnIdx < userTurns.length; turnIdx++) {\n\t\tconst { userMsg, assistantMsg } = userTurns[turnIdx];\n\t\tconst isLastTurn = turnIdx === userTurns.length - 1;\n\n\t\t// Extract expected tools from the recorded assistant message\n\t\tconst expectedTools = assistantMsg\n\t\t\t? extractRecordedTools(assistantMsg)\n\t\t\t: [];\n\n\t\thistory.push(userMsg);\n\n\t\tif (mode === \"inject\" && !isLastTurn && assistantMsg) {\n\t\t\thistory.push(assistantMsg);\n\t\t\tconst response = parseUIMessage(assistantMsg);\n\t\t\tconst assertions = buildAssertions(expectedTools, response.toolsCalled);\n\t\t\tturnResults.push({\n\t\t\t\tinput: textFromUIMessage(userMsg),\n\t\t\t\tresponse,\n\t\t\t\tassertions,\n\t\t\t});\n\t\t\tcontinue;\n\t\t}\n\n\t\tconst { result, message } = await sendMessages(url, history);\n\t\thistory.push(message);\n\n\t\tconst assertions = buildAssertions(expectedTools, result.toolsCalled);\n\t\tturnResults.push({\n\t\t\tinput: textFromUIMessage(userMsg),\n\t\t\tresponse: result,\n\t\t\tassertions,\n\t\t});\n\t}\n\n\treturn { turns: turnResults };\n}\n\n/** Compare expected vs. actual tool calls and return assertion results. */\nfunction buildAssertions(\n\texpected: string[],\n\tactual: string[],\n): TurnAssertion[] {\n\tif (expected.length === 0) {\n\t\treturn [];\n\t}\n\n\t// Group expected tools and check each against actual calls\n\tconst actualSet = new Set(actual);\n\tconst expectedUnique = [...new Set(expected)];\n\n\treturn expectedUnique.map((tool) => ({\n\t\tpassed: actualSet.has(tool),\n\t\texpected: [tool],\n\t\tactual,\n\t}));\n}\n","import { existsSync, mkdirSync, writeFileSync } from \"node:fs\";\nimport { parseTaskOutput } from \"./scorers\";\n\ntype ReporterFn = (\n\tname: string,\n\thandlers: {\n\t\treportEval(\n\t\t\tevaluator: unknown,\n\t\t\tresult: {\n\t\t\t\tresults: Array<{\n\t\t\t\t\tinput: unknown;\n\t\t\t\t\toutput: unknown;\n\t\t\t\t\tscores?: Record<string, number>;\n\t\t\t\t}>;\n\t\t\t},\n\t\t): boolean;\n\t\treportRun(results: boolean[]): boolean;\n\t},\n) => unknown;\n\n/**\n * Create a local Braintrust reporter that writes JSON results to a directory and\n * prints a summary to console.\n *\n * Requires the `braintrust` package: bun add -d braintrust\n *\n * @param outputDir - Directory to write JSON result files (default: \"evals/runs\")\n */\nexport function createLocalReporter(outputDir = \"evals/runs\") {\n\tlet Reporter: ReporterFn;\n\ttry {\n\t\tReporter = (require(\"braintrust\") as { Reporter: ReporterFn }).Reporter;\n\t} catch {\n\t\tthrow new Error(\n\t\t\t'Local reporter requires the \"braintrust\" package: bun add -d braintrust',\n\t\t);\n\t}\n\n\tif (!existsSync(outputDir)) {\n\t\tmkdirSync(outputDir, { recursive: true });\n\t}\n\n\treturn Reporter(\"local\", {\n\t\treportEval(\n\t\t\tevaluator: unknown,\n\t\t\tresult: {\n\t\t\t\tresults: Array<{\n\t\t\t\t\tinput: unknown;\n\t\t\t\t\toutput: unknown;\n\t\t\t\t\tscores?: Record<string, number>;\n\t\t\t\t}>;\n\t\t\t},\n\t\t) {\n\t\t\tconst name =\n\t\t\t\t(evaluator as { experimentName?: string }).experimentName ?? \"unknown\";\n\t\t\tconst timestamp = new Date().toISOString().replace(/[:.]/g, \"-\");\n\t\t\tconst rows = result.results.map((r) => {\n\t\t\t\tconst scores = r.scores ?? {};\n\t\t\t\tconst parsed = parseTaskOutput(r.output);\n\t\t\t\treturn {\n\t\t\t\t\tinput: r.input,\n\t\t\t\t\toutput: parsed.output,\n\t\t\t\t\ttoolsCalled: parsed.toolsCalled,\n\t\t\t\t\ttoolCallTraces: parsed.toolCallTraces,\n\t\t\t\t\tscores,\n\t\t\t\t};\n\t\t\t});\n\n\t\t\tconst outPath = `${outputDir}/${name}-${timestamp}.json`;\n\t\t\twriteFileSync(outPath, JSON.stringify(rows, null, 2));\n\n\t\t\tconsole.log(`\\n📊 ${name} (${result.results.length} cases):\\n`);\n\n\t\t\tlet failures = 0;\n\t\t\tfor (const r of result.results) {\n\t\t\t\tconst scores = r.scores ?? {};\n\t\t\t\tconst pass = scores.called_expected_tool === 1;\n\t\t\t\tif (!pass) {\n\t\t\t\t\tfailures++;\n\t\t\t\t}\n\n\t\t\t\tconsole.log(\n\t\t\t\t\t` ${pass ? \"✅\" : \"❌\"} ${(r.input as string).slice(0, 70)}`,\n\t\t\t\t);\n\t\t\t\tfor (const [scoreName, value] of Object.entries(scores)) {\n\t\t\t\t\tconsole.log(` ${scoreName}: ${value}`);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconsole.log(\n\t\t\t\t`\\n ${result.results.length - failures}/${result.results.length} passed`,\n\t\t\t);\n\t\t\tconsole.log(` → ${outPath}\\n`);\n\t\t\treturn failures === 0;\n\t\t},\n\n\t\treportRun(results: boolean[]) {\n\t\t\tconst allPassed = results.every((r) => r === true);\n\t\t\tconsole.log(\n\t\t\t\tallPassed\n\t\t\t\t\t? \"\\n✅ All experiments passed\"\n\t\t\t\t\t: \"\\n❌ Some experiments failed\",\n\t\t\t);\n\t\t\treturn allPassed;\n\t\t},\n\t});\n}\n","import type { ChatResult } from \"./types\";\n\n/**\n * Parse the JSON-stringified ChatResult from a Braintrust task output.\n */\nexport function parseTaskOutput(output: unknown): ChatResult {\n\ttry {\n\t\treturn JSON.parse(output as string);\n\t} catch {\n\t\treturn { output: \"\", toolsCalled: [], toolCallTraces: [] };\n\t}\n}\n\n/**\n * Checks whether the expected tool was called.\n * Looks for the tool name in `metadata.expectedTool` first (for cases where `expected` is a\n * reference answer), then falls back to `expected` directly.\n */\nexport function calledExpectedTool({\n\toutput,\n\texpected,\n\tmetadata,\n}: {\n\toutput: unknown;\n\texpected?: unknown;\n\tmetadata?: Record<string, unknown>;\n}) {\n\tconst parsed = parseTaskOutput(output);\n\tconst expectedTool =\n\t\t(metadata?.expectedTool as string) ?? (expected as string);\n\tconst found = parsed.toolsCalled.includes(expectedTool);\n\treturn {\n\t\tname: \"called_expected_tool\",\n\t\tscore: found ? 1 : 0,\n\t\tmetadata: { expected: expectedTool, actual: parsed.toolsCalled },\n\t};\n}\n\n/**\n * Checks whether the assistant produced any text output.\n */\nexport function hasOutput({ output }: { output: unknown }) {\n\tconst parsed = parseTaskOutput(output);\n\treturn {\n\t\tname: \"has_output\",\n\t\tscore: parsed.output.length > 0 ? 1 : 0,\n\t};\n}\n\n/**\n * Checks specific fields in the first tool call's `stateUpdates` against expected values.\n * Supports nested fields via dot notation (e.g. \"mixedBreed.knowsBreeds\").\n * Returns partial credit (fraction of matching fields).\n */\nexport function toolInputFieldsMatch({\n\toutput,\n\tmetadata,\n}: {\n\toutput: unknown;\n\tmetadata?: Record<string, unknown>;\n}) {\n\tconst parsed = parseTaskOutput(output);\n\tconst expectedFields = (metadata?.expectedFields ?? {}) as Record<\n\t\tstring,\n\t\tunknown\n\t>;\n\tconst fieldNames = Object.keys(expectedFields);\n\n\tif (fieldNames.length === 0) {\n\t\treturn { name: \"field_extraction\", score: 1 };\n\t}\n\n\tconst trace = parsed.toolCallTraces[0];\n\tconst stateUpdates = (trace?.input?.stateUpdates ?? {}) as Record<\n\t\tstring,\n\t\tunknown\n\t>;\n\n\tlet matches = 0;\n\tconst details: Record<\n\t\tstring,\n\t\t{ expected: unknown; actual: unknown; match: boolean }\n\t> = {};\n\n\tfor (const field of fieldNames) {\n\t\tconst expected = expectedFields[field];\n\t\tlet actual: unknown;\n\n\t\tif (field.includes(\".\")) {\n\t\t\tconst [parent, child] = field.split(\".\");\n\t\t\tactual = (stateUpdates[parent] as Record<string, unknown>)?.[child];\n\t\t} else {\n\t\t\tactual = stateUpdates[field];\n\t\t}\n\n\t\tconst match = JSON.stringify(actual) === JSON.stringify(expected);\n\t\tif (match) {\n\t\t\tmatches++;\n\t\t}\n\t\tdetails[field] = { expected, actual, match };\n\t}\n\n\treturn {\n\t\tname: \"field_extraction\",\n\t\tscore: matches / fieldNames.length,\n\t\tmetadata: details,\n\t};\n}\n\n/**\n * Wraps an autoevals scorer to extract the text output from the JSON-stringified ChatResult.\n * Requires the `autoevals` package: bun add -d autoevals\n */\nfunction wrapAutoeval(\n\tscorer: (args: {\n\t\tinput: unknown;\n\t\toutput: string;\n\t\texpected?: unknown;\n\t}) => unknown,\n) {\n\treturn async (args: {\n\t\tinput: unknown;\n\t\toutput: unknown;\n\t\texpected?: unknown;\n\t}) => {\n\t\tconst parsed = parseTaskOutput(args.output);\n\t\treturn scorer({\n\t\t\tinput: args.input,\n\t\t\toutput: parsed.output,\n\t\t\texpected: args.expected,\n\t\t});\n\t};\n}\n\n// LLM-based scorers — require `autoevals` as a dev dependency.\n// These are dynamically imported so the module loads even if autoevals is not installed.\n// Using LLM scorers without autoevals installed will throw at call time.\n\nasync function getAutoeval(name: string) {\n\tconst mod = await import(\"autoevals\").catch(() => {\n\t\tthrow new Error(\n\t\t\t`LLM scorer \"${name}\" requires the \"autoevals\" package: bun add -d autoevals`,\n\t\t);\n\t});\n\treturn (mod as Record<string, unknown>)[name] as (args: {\n\t\tinput: unknown;\n\t\toutput: string;\n\t\texpected?: unknown;\n\t}) => unknown;\n}\n\n/** ClosedQA — checks if the answer correctly addresses the question given a reference answer. */\nexport const FaqAccuracy = async (args: {\n\tinput: unknown;\n\toutput: unknown;\n\texpected?: unknown;\n}) => wrapAutoeval(await getAutoeval(\"ClosedQA\"))(args);\n\n/** Factuality — checks if the output is factually consistent with the expected output. */\nexport const OutputFactuality = async (args: {\n\tinput: unknown;\n\toutput: unknown;\n\texpected?: unknown;\n}) => wrapAutoeval(await getAutoeval(\"Factuality\"))(args);\n\n/** Moderation — flags unsafe or inappropriate content. */\nexport const SafetyCheck = async (args: {\n\tinput: unknown;\n\toutput: unknown;\n\texpected?: unknown;\n}) => wrapAutoeval(await getAutoeval(\"Moderation\"))(args);\n"],"mappings":"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","names":["parseJsonEventStream","readUIMessageStream","uiMessageChunkSchema","parseUIMessage","msg","output","p","toolParts","toolsCalled","toolCallTraces","textFromUIMessage","extractRecordedTools","sendMessages","url","messages","response","chunkStream","chunk","controller","finalMessage","chat","message","userMsg","result","conversation","turns","history","turnResults","turn","replaySession","session","mode","userTurns","i","next","turnIdx","assistantMsg","isLastTurn","expectedTools","assertions","buildAssertions","expected","actual","actualSet","tool","existsSync","mkdirSync","writeFileSync","parseTaskOutput","output","calledExpectedTool","expected","metadata","parsed","expectedTool","hasOutput","toolInputFieldsMatch","expectedFields","fieldNames","stateUpdates","matches","details","field","actual","parent","child","match","wrapAutoeval","scorer","args","getAutoeval","name","FaqAccuracy","OutputFactuality","SafetyCheck","createLocalReporter","outputDir","Reporter","existsSync","mkdirSync","evaluator","result","name","timestamp","rows","r","scores","parsed","parseTaskOutput","outPath","writeFileSync","failures","pass","scoreName","value","results","allPassed"]}
package/dist/mcp/index.js CHANGED
@@ -1,7 +1,7 @@
1
- function Y(){return typeof window<"u"&&"openai"in window?"openai":"mcp-apps"}function Ve(){return Y()==="openai"}function ze(){return Y()==="mcp-apps"}var R="__start__",C="__end__",de=Symbol.for("waniwani.flow.interrupt"),ue=Symbol.for("waniwani.flow.widget");function pe(e,t){let n=t?.context,r=[];for(let[o,s]of Object.entries(e))if(typeof s=="object"&&s!==null&&"question"in s){let i=s;r.push({question:i.question,field:o,suggestions:i.suggestions,context:i.context,validate:i.validate})}return{__type:de,questions:r,context:n}}function le(e,t){return{__type:ue,tool:e,...t}}function fe(e){return typeof e=="object"&&e!==null&&"__type"in e&&e.__type===de}function ge(e){return typeof e=="object"&&e!==null&&"__type"in e&&e.__type===ue}import{z as K}from"zod";var J="waniwani/client";function O(e){if(typeof e=="object"&&e!==null)return e[J]}function me(e,t){return{track(n){return e.track({...n,meta:{...t,...n.meta}})},identify(n,r){return e.identify(n,r,t)},kb:e.kb}}function A(e,t){for(let n of t){let r=e[n];if(typeof r=="string"&&r.length>0)return r}}var He=["waniwani/sessionId","openai/sessionId","sessionId","conversationId","anthropic/sessionId"],Ye=["waniwani/requestId","openai/requestId","requestId","mcp/requestId"],Je=["waniwani/traceId","openai/traceId","traceId","mcp/traceId","openai/requestId","requestId"],Xe=["waniwani/userId","openai/userId","externalUserId","userId","actorId"],Ge=["correlationId","openai/requestId"];function _(e){return e?A(e,He):void 0}function he(e){return e?A(e,Ye):void 0}function we(e){return e?A(e,Je):void 0}function ye(e){return e?A(e,Xe):void 0}function Te(e){return e?A(e,Ge):void 0}var Ze=[{key:"waniwani/sessionId",source:"chatbar"},{key:"openai/sessionId",source:"chatgpt"},{key:"anthropic/sessionId",source:"claude"}];function N(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 Ze){let o=e[n];if(typeof o=="string"&&o.length>0)return r}}function ke(e){let t=e._zod?.def;return t?.type==="object"&&t.shape?t.shape:null}function D(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 Qe(e,t,n){let r=t.split("."),o=r.pop();if(!o)return;let s=e;for(let i of r)(s[i]==null||typeof s[i]!="object"||Array.isArray(s[i]))&&(s[i]={}),s=s[i];s[o]=n}function Se(e,t){let n=t.split("."),r=n.pop();if(!r)return;let o=e;for(let s of n){if(o==null||typeof o!="object")return;o=o[s]}o!=null&&typeof o=="object"&&delete o[r]}function X(e){let t={};for(let[n,r]of Object.entries(e))n.includes(".")?Qe(t,n,r):t[n]=r;return t}function M(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]=M(n[r],o):n[r]=o;return n}function G(e){return e!=null&&e!==""}async function F(e,t){return e.type==="direct"?e.to:e.condition(t)}function ve(e,t,n,r){if(e.every(c=>G(D(r,c.field))))return null;let o=e.filter(c=>!G(D(r,c.field))),s=o.length===1,i=o[0];return{content:s&&i?{status:"interrupt",question:i.question,field:i.field,...i.suggestions?{suggestions:i.suggestions}:{},...i.context||t?{context:i.context??t}:{}}:{status:"interrupt",questions:o,...t?{context:t}:{}},flowTokenContent:{step:n,state:r,...s&&i?{field:i.field}:{}}}}async function j(e,t,n,r,o,s,i){let a=e,c={...t},d=50,f=0;for(;f++<d;){if(a===C)return{content:{status:"complete"},flowTokenContent:{state:c}};let h=n.get(a);if(!h)return{content:{status:"error",error:`Unknown node: "${a}"`}};try{let p=await h({state:c,meta:s,interrupt:pe,showWidget:le,waniwani:i});if(fe(p)){for(let T of p.questions)T.validate&&o.set(`${a}:${T.field}`,T.validate);let y=ve(p.questions,p.context,a,c);if(y)return y;for(let T of p.questions){let x=o.get(`${a}:${T.field}`);if(x)try{let b=D(c,T.field),E=await x(b);E&&typeof E=="object"&&(c=M(c,E))}catch(b){let E=b instanceof Error?b.message:String(b);Se(c,T.field);let k=p.questions.map(I=>I.field===T.field?{...I,context:I.context?`ERROR: ${E}
1
+ function Y(){return typeof window<"u"&&"openai"in window?"openai":"mcp-apps"}function Ve(){return Y()==="openai"}function ze(){return Y()==="mcp-apps"}var R="__start__",C="__end__",de=Symbol.for("waniwani.flow.interrupt"),ue=Symbol.for("waniwani.flow.widget");function pe(e,t){let n=t?.context,r=[];for(let[o,s]of Object.entries(e))if(typeof s=="object"&&s!==null&&"question"in s){let a=s;r.push({question:a.question,field:o,suggestions:a.suggestions,context:a.context,validate:a.validate})}return{__type:de,questions:r,context:n}}function le(e,t){return{__type:ue,tool:e,...t}}function fe(e){return typeof e=="object"&&e!==null&&"__type"in e&&e.__type===de}function ge(e){return typeof e=="object"&&e!==null&&"__type"in e&&e.__type===ue}import{z as $}from"zod";var J="waniwani/client";function j(e){if(typeof e=="object"&&e!==null)return e[J]}function me(e,t){return{track(n){return e.track({...n,meta:{...t,...n.meta}})},identify(n,r){return e.identify(n,r,t)},kb:e.kb}}function N(e,t){for(let n of t){let r=e[n];if(typeof r=="string"&&r.length>0)return r}}var He=["waniwani/sessionId","openai/sessionId","sessionId","conversationId","anthropic/sessionId"],Ye=["waniwani/requestId","openai/requestId","requestId","mcp/requestId"],Je=["waniwani/traceId","openai/traceId","traceId","mcp/traceId","openai/requestId","requestId"],Xe=["waniwani/userId","openai/userId","externalUserId","userId","actorId"],Ge=["correlationId","openai/requestId"];function _(e){return e?N(e,He):void 0}function he(e){return e?N(e,Ye):void 0}function we(e){return e?N(e,Je):void 0}function ye(e){return e?N(e,Xe):void 0}function Te(e){return e?N(e,Ge):void 0}var Ze=[{key:"waniwani/sessionId",source:"chatbar"},{key:"openai/sessionId",source:"chatgpt"},{key:"anthropic/sessionId",source:"claude"}];function M(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 Ze){let o=e[n];if(typeof o=="string"&&o.length>0)return r}}function ke(e){let t=e._zod?.def;return t?.type==="object"&&t.shape?t.shape:null}function D(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 Qe(e,t,n){let r=t.split("."),o=r.pop();if(!o)return;let s=e;for(let a of r)(s[a]==null||typeof s[a]!="object"||Array.isArray(s[a]))&&(s[a]={}),s=s[a];s[o]=n}function Se(e,t){let n=t.split("."),r=n.pop();if(!r)return;let o=e;for(let s of n){if(o==null||typeof o!="object")return;o=o[s]}o!=null&&typeof o=="object"&&delete o[r]}function X(e){let t={};for(let[n,r]of Object.entries(e))n.includes(".")?Qe(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 G(e){return e!=null&&e!==""}async function W(e,t){return e.type==="direct"?e.to:e.condition(t)}function ve(e,t,n,r){if(e.every(c=>G(D(r,c.field))))return null;let o=e.filter(c=>!G(D(r,c.field))),s=o.length===1,a=o[0];return{content:s&&a?{status:"interrupt",question:a.question,field:a.field,...a.suggestions?{suggestions:a.suggestions}:{},...a.context||t?{context:a.context??t}:{}}:{status:"interrupt",questions:o,...t?{context:t}:{}},flowTokenContent:{step:n,state:r,...s&&a?{field:a.field}:{}}}}async function K(e,t,n,r,o,s,a){let i=e,c={...t},d=50,l=0;for(;l++<d;){if(i===C)return{content:{status:"complete"},flowTokenContent:{state:c}};let g=n.get(i);if(!g)return{content:{status:"error",error:`Unknown node: "${i}"`}};try{let p=await g({state:c,meta:s,interrupt:pe,showWidget:le,waniwani:a});if(fe(p)){for(let T of p.questions)T.validate&&o.set(`${i}:${T.field}`,T.validate);let y=ve(p.questions,p.context,i,c);if(y)return y;for(let T of p.questions){let x=o.get(`${i}:${T.field}`);if(x)try{let b=D(c,T.field),E=await x(b);E&&typeof E=="object"&&(c=F(c,E))}catch(b){let E=b instanceof Error?b.message:String(b);Se(c,T.field);let k=p.questions.map(I=>I.field===T.field?{...I,context:I.context?`ERROR: ${E}
2
2
 
3
- ${I.context}`:`ERROR: ${E}`}:I),P=ve(k,p.context,a,c);if(P)return P;break}}let u=r.get(a);if(!u)return{content:{status:"error",error:`No outgoing edge from node "${a}"`}};a=await F(u,c);continue}if(ge(p)){let y=p.field;if(y&&G(D(c,y))){let u=r.get(a);if(!u)return{content:{status:"error",error:`No outgoing edge from node "${a}"`}};a=await F(u,c);continue}return{content:{status:"widget",tool:p.tool.id,data:p.data,description:p.description,interactive:p.interactive!==!1},flowTokenContent:{step:a,state:c,field:y,widgetId:p.tool.id}}}c=M(c,p);let w=r.get(a);if(!w)return{content:{status:"error",error:`No outgoing edge from node "${a}"`}};a=await F(w,c)}catch(l){return{content:{status:"error",error:l instanceof Error?l.message:String(l)},flowTokenContent:{step:a,state:c}}}}return{content:{status:"error",error:"Flow exceeded maximum iterations (possible infinite loop)"}}}var et="@waniwani/sdk",tt="https://app.waniwani.ai",U=class{baseUrl;apiKey;constructor(t){this.baseUrl=(t?.baseUrl??process.env.WANIWANI_BASE_URL??tt).replace(/\/$/,""),this.apiKey=t?.apiKey??process.env.WANIWANI_API_KEY}async get(t){if(!this.apiKey)return null;try{return await this.request("/api/mcp/redis/get",{key:t})??null}catch{return null}}async set(t,n){if(this.apiKey)try{await this.request("/api/mcp/redis/set",{key:t,value:n})}catch{}}async delete(t){if(this.apiKey)try{await this.request("/api/mcp/redis/delete",{key:t})}catch{}}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 i=await o.text().catch(()=>"");throw new Error(i||`Flow state API error: HTTP ${o.status}`)}return(await o.json()).data}};function xe(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 Re(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. 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.state){let n=[];for(let[r,o]of Object.entries(e.state)){let s=ke(o);if(s){let i=o.description??"",a=Object.entries(s).map(([c,d])=>{let f=xe(d);return f?`\`${r}.${c}\` (${f})`:`\`${r}.${c}\``}).join(", ");n.push(i?`\`${r}\` (${i}): ${a}`:`\`${r}\`: ${a}`)}else{let i=xe(o);n.push(i?`\`${r}\` (${i})`:`\`${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
- `)}var nt={action:K.enum(["start","continue"]).describe('"start" to begin the flow, "continue" to resume after a pause (interrupt or widget)'),stateUpdates:K.record(K.string(),K.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 Ee(e){let{config:t,nodes:n,edges:r}=e,o=Re(t),s=`${t.description}
5
- ${o}`,i=e.store??new U,a=new Map;async function c(d,f,h,l){if(d.action==="start"){let p=r.get(R);if(!p)return{content:{status:"error",error:"No start edge"}};let w=X(d.stateUpdates??{}),y=await F(p,w);return j(y,w,n,r,a,h,l)}if(d.action==="continue"){if(!f)return{content:{status:"error",error:"No session ID available for continue action."}};let p=await i.get(f);if(!p)return{content:{status:"error",error:"Flow state not found. The flow may have expired."}};let w=p.state,y=p.step;if(!y)return{content:{status:"error",error:"Flow state is missing the current step. The flow may have expired."}};let u=M(w,X(d.stateUpdates??{}));if(p.widgetId){let T=r.get(y);if(!T)return{content:{status:"error",error:`No edge from step "${y}"`}};let x=await F(T,u);return j(x,u,n,r,a,h,l)}return j(y,u,n,r,a,h,l)}return{content:{status:"error",error:`Unknown action: "${d.action}"`}}}return{id:t.id,title:t.title,description:s,graph:e.graph,async register(d){d.registerTool(t.id,{title:t.title,description:s,inputSchema:nt,annotations:t.annotations},(async(f,h)=>{let l=h,p=l._meta??{},w=_(p),y=O(l),u=await c(f,w,p,y);return u.flowTokenContent&&w&&await i.set(w,u.flowTokenContent),{content:[{type:"text",text:JSON.stringify(u.content,null,2)}],_meta:p,...u.content.status==="error"?{isError:!0}:{}}}))}}}function Ie(e,t){let n=["flowchart TD"];n.push(` ${R}((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 W=class{nodes=new Map;edges=new Map;config;constructor(t){this.config=t}addNode(t,n){if(t===R||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),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 Ie(this.nodes,this.edges)}compile(t){this.validate();let n=new Map(this.nodes),r=new Map(this.edges);return Ee({config:this.config,nodes:n,edges:r,store:t?.store,graph:()=>Ie(n,r)})}validate(){if(!this.edges.has(R))throw new Error('Flow must have an entry point. Add an edge from START: .addEdge(START, "first_node")');let t=this.edges.get(R);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!==R&&!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 Ce(e){return new W(e)}function be(e){let t=e.content;return JSON.parse(t[0]?.text??"")}async function _e(e,t){let n=t?.stateStore,r=[],o=`test-session-${Math.random().toString(36).slice(2,10)}`,s={registerTool:(...d)=>{r.push(d)}};await e.register(s);let i=r[0]?.[2];if(!i)throw new Error(`Flow "${e.id}" did not register a handler`);let a={_meta:{sessionId:o}};async function c(d){return{...d,decodedState:n?await n.get(o):null}}return{async start(d){let f=await i({action:"start",...d?{stateUpdates:d}:{}},a);return c(be(f))},async continueWith(d){let f=await i({action:"continue",...d?{stateUpdates:d}:{}},a);return c(be(f))},async lastState(){return n?n.get(o):null}}}var $="text/html+skybridge",B="text/html;profile=mcp-app",Pe=async(e,t)=>{let n=e.endsWith("/")?e.slice(0,-1):e;return await(await fetch(`${n}${t}`)).text()};function Me(e){return{"openai/widgetDescription":e.description,"openai/widgetPrefersBorder":e.prefersBorder,"openai/widgetDomain":e.widgetDomain,...e.widgetCSP&&{"openai/widgetCSP":e.widgetCSP}}}function Fe(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 Z(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 We(e){let{id:t,title:n,description:r,baseUrl:o,htmlPath:s,widgetDomain:i,prefersBorder:a=!0,autoHeight:c=!0}=e,d=e.widgetCSP??{connect_domains:[o],resource_domains:[o]};if(process.env.NODE_ENV==="development")try{let{hostname:u}=new URL(o);(u==="localhost"||u==="127.0.0.1")&&(d={...d,connect_domains:[...d.connect_domains||[],`ws://${u}:*`,`wss://${u}:*`],resource_domains:[...d.resource_domains||[],`http://${u}:*`]})}catch{}let f=`ui://widgets/apps-sdk/${t}.html`,h=`ui://widgets/ext-apps/${t}.html`,l=null,p=()=>(l||(l=Pe(o,s)),l),w=r;async function y(u){let T=await p();u.registerResource(`${t}-openai-widget`,f,{title:n,description:w,mimeType:$,_meta:{"openai/widgetDescription":w,"openai/widgetPrefersBorder":a}},async x=>({contents:[{uri:x.href,mimeType:$,text:T,_meta:Me({description:w,prefersBorder:a,widgetDomain:i,widgetCSP:d})}]})),u.registerResource(`${t}-mcp-widget`,h,{title:n,description:w,mimeType:B,_meta:{ui:{prefersBorder:a}}},async x=>({contents:[{uri:x.href,mimeType:B,text:T,_meta:Fe({description:w,prefersBorder:a,widgetDomain:i,widgetCSP:d})}]}))}return{id:t,title:n,description:r,openaiUri:f,mcpUri:h,autoHeight:c,register:y}}function rt(e,t){let{resource:n,description:r,inputSchema:o,annotations:s,autoInjectResultText:i=!0}=e,a=e.id??n?.id,c=e.title??n?.title;if(!a)throw new Error("createTool: `id` is required when no resource is provided");if(!c)throw new Error("createTool: `title` is required when no resource is provided");let d=n?Z({openaiTemplateUri:n.openaiUri,mcpTemplateUri:n.mcpUri,invoking:e.invoking??"Loading...",invoked:e.invoked??"Loaded",autoHeight:n.autoHeight}):void 0;return{id:a,title:c,description:r,async register(f){f.registerTool(a,{title:c,description:r,inputSchema:o,annotations:s,...d&&{_meta:d}},(async(h,l)=>{let p=l,w=p._meta??{},y=O(p),u=await t(h,{extra:{_meta:w},waniwani:y});return n&&u.data?{content:[{type:"text",text:u.text}],structuredContent:u.data,_meta:{...d,...w,...i===!1?{"waniwani/autoInjectResultText":!1}:{}}}:{content:[{type:"text",text:u.text}],...u.data?{structuredContent:u.data}:{},...i===!1?{_meta:{"waniwani/autoInjectResultText":!1}}:{}}}))}}}async function ot(e,t){await Promise.all(t.map(n=>n.register(e)))}var q=class extends Error{constructor(n,r){super(n);this.status=r;this.name="WaniWaniError"}};var it="@waniwani/sdk";function Ae(e){let{baseUrl:t,apiKey:n}=e;function r(){if(!n)throw new Error("WANIWANI_API_KEY is not set");return n}async function o(s,i,a){let c=r(),d=`${t.replace(/\/$/,"")}${i}`,f={Authorization:`Bearer ${c}`,"X-WaniWani-SDK":it},h={method:s,headers:f};a!==void 0&&(f["Content-Type"]="application/json",h.body=JSON.stringify(a));let l=await fetch(d,h);if(!l.ok){let w=await l.text().catch(()=>"");throw new q(w||`KB API error: HTTP ${l.status}`,l.status)}return(await l.json()).data}return{async ingest(s){return o("POST","/api/mcp/kb/ingest",{files:s})},async search(s,i){return o("POST","/api/mcp/kb/search",{query:s,...i})},async sources(){return o("GET","/api/mcp/kb/sources")}}}var st="@waniwani/sdk";function V(e,t={}){let n=t.now??(()=>new Date),r=t.generateId??Ne,o=dt(e),s=L(e.meta),i=L(e.metadata),a=ut(e,s),c=v(e.eventId)??r(),d=pt(e.timestamp,n),f=v(e.source)??N(s)??t.source??st,h=Q(e)?{...e}:void 0,l={...i};return Object.keys(s).length>0&&(l.meta=s),h&&(l.rawLegacy=h),{id:c,type:"mcp.event",name:o,source:f,timestamp:d,correlation:a,properties:at(e,o),metadata:l,rawLegacy:h}}function Ne(){return typeof crypto<"u"&&typeof crypto.randomUUID=="function"?`evt_${crypto.randomUUID()}`:`evt_${Math.random().toString(36).slice(2,10)}_${Date.now().toString(36)}`}function at(e,t){if(!Q(e))return L(e.properties);let n=ct(e,t),r=L(e.properties);return{...n,...r}}function ct(e,t){switch(t){case"tool.called":{let n={};return v(e.toolName)&&(n.name=e.toolName),v(e.toolType)&&(n.type=e.toolType),n}case"quote.succeeded":{let n={};return typeof e.quoteAmount=="number"&&(n.amount=e.quoteAmount),v(e.quoteCurrency)&&(n.currency=e.quoteCurrency),n}case"link.clicked":{let n={};return v(e.linkUrl)&&(n.url=e.linkUrl),n}case"purchase.completed":{let n={};return typeof e.purchaseAmount=="number"&&(n.amount=e.purchaseAmount),v(e.purchaseCurrency)&&(n.currency=e.purchaseCurrency),n}default:return{}}}function dt(e){return Q(e)?e.eventType:e.event}function ut(e,t){let n=v(e.requestId)??he(t),r=v(e.sessionId)??_(t),o=v(e.traceId)??we(t),s=v(e.externalUserId)??ye(t),i=v(e.correlationId)??Te(t)??n,a={};return r&&(a.sessionId=r),o&&(a.traceId=o),n&&(a.requestId=n),i&&(a.correlationId=i),s&&(a.externalUserId=s),a}function pt(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 L(e){return!e||typeof e!="object"||Array.isArray(e)?{}:e}function v(e){if(typeof e=="string"&&e.trim().length!==0)return e}function Q(e){return"eventType"in e}var lt="/api/mcp/events/v2/batch";var De="@waniwani/sdk",ft=new Set([401,403]),gt=new Set([408,425,429,500,502,503,504]);function Ue(e){return new ee(e)}var ee=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=wt(t.baseUrl,t.endpointPath??lt),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":De},body:JSON.stringify(this.makeBatchRequest(t))})}catch(s){return{kind:"retryable",reason:yt(s)}}if(ft.has(n.status))return{kind:"auth",status:n.status};if(gt.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:De,version:this.sdkVersion??"0.0.0"},events:t}}classifyRejectedEvents(t,n){let r=new Map(t.map(i=>[i.id,i])),o=[],s=[];for(let i of n){let a=r.get(i.eventId);if(a){if(mt(i)){o.push(a);continue}s.push(a)}}return{retryable:o,permanent:s}}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 mt(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 wt(e,t){let n=e.endsWith("/")?e:`${e}/`,r=t.startsWith("/")?t.slice(1):t;return`${n}${r}`}function yt(e){return e instanceof Error?e.message:String(e)}function Oe(e){let{baseUrl:t,apiKey:n,tracking:r}=e;function o(){if(!n)throw new Error("WANIWANI_API_KEY is not set");return n}let s=n?Ue({baseUrl: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,i={async identify(a,c,d){o();let f=V({event:"user.identified",externalUserId:a,properties:c,meta:d});return s?.enqueue(f),{eventId:f.id}},async track(a){o();let c=V(a);return s?.enqueue(c),{eventId:c.id}},async flush(){o(),await s?.flush()},async shutdown(a){return o(),await s?.shutdown({timeoutMs:a?.timeoutMs??r.shutdownTimeoutMs})??{timedOut:!1,pendingEvents:0}}};return s&&Tt(i,r.shutdownTimeoutMs),i}function Tt(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 je(e){let t=e?.baseUrl??"https://app.waniwani.ai",n=e?.apiKey??process.env.WANIWANI_API_KEY,r={endpointPath:e?.tracking?.endpointPath??"/api/mcp/events/v2/batch",flushIntervalMs:e?.tracking?.flushIntervalMs??1e3,maxBatchSize:e?.tracking?.maxBatchSize??20,maxBufferSize:e?.tracking?.maxBufferSize??1e3,maxRetries:e?.tracking?.maxRetries??3,retryBaseDelayMs:e?.tracking?.retryBaseDelayMs??200,retryMaxDelayMs:e?.tracking?.retryMaxDelayMs??2e3,shutdownTimeoutMs:e?.tracking?.shutdownTimeoutMs??2e3},o={baseUrl:t,apiKey:n,tracking:r},s=Oe(o),i=Ae(o);return{...s,kb:i,_config:o}}function kt(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 St(e){let t={apiKey:e?.apiKey,baseUrl:e?.baseUrl},n;function r(){return n||(n=je(t)),n}return async function(s){let i;try{i=await s.json()}catch{return new Response(JSON.stringify({error:"Invalid JSON"}),{status:400,headers:{"Content-Type":"application/json"}})}if(!Array.isArray(i.events)||i.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(),c=[];for(let d of i.events){let f=kt(d),h=await a.track(f);c.push(h.eventId)}return await a.flush(),new Response(JSON.stringify({ok:!0,accepted:c.length}),{status:200,headers:{"Content-Type":"application/json"}})}catch(a){let c=a instanceof Error?a.message:"Unknown error";return new Response(JSON.stringify({error:c}),{status:500,headers:{"Content-Type":"application/json"}})}}}var z=class{cached=null;pending=null;config;constructor(t){this.config=t}async getToken(t,n){return this.cached&&Date.now()<this.cached.expiresAt-12e4?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=vt(this.config.baseUrl,"/api/mcp/widget-tokens"),o={};t&&(o.sessionId=t),n&&(o.traceId=n);try{let s=await fetch(r,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${this.config.apiKey}`},body:JSON.stringify(o)});if(!s.ok)return null;let i=await s.json(),a=i.data&&typeof i.data=="object"?i.data:i,c=new Date(a.expiresAt).getTime();return!a.token||Number.isNaN(c)?null:(this.cached={token:a.token,expiresAt:c},a.token)}catch{return null}}};function vt(e,t){return`${e.endsWith("/")?e.slice(0,-1):e}${t}`}var Ke="waniwani/sessionId",te="waniwani/geoLocation",ne="waniwani/userLocation";function S(e){return!!e&&typeof e=="object"&&!Array.isArray(e)}function H(e){if(!S(e))return;let t=e._meta;return S(t)?t:void 0}function re(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 xt(e,t){return typeof t=="function"?t(e)??"other":t??"other"}function oe(e,t,n,r,o,s){let i=xt(e,n.toolType),a=H(t);return console.log("[waniwani:debug] buildTrackInput meta:",JSON.stringify(a),"-> source:",N(a)),{event:"tool.called",properties:{name:e,type:i,...r??{},...s?.input!==void 0&&{input:s.input},...s?.output!==void 0&&{output:s.output}},meta:a,source:N(a),metadata:{...n.metadata??{},...o&&{clientInfo:o}}}}async function ie(e,t,n){try{await e.track(t)}catch(r){n?.(ae(r))}}async function se(e,t){try{await e.flush()}catch(n){t?.(ae(n))}}async function $e(e,t,n,r){if(!S(e))return;S(e._meta)||(e._meta={});let o=e._meta,s=S(o.waniwani)?o.waniwani:void 0,i={...s??{},endpoint:s?.endpoint??`${n.replace(/\/$/,"")}/api/mcp/events/v2/batch`};if(t)try{let d=await t.getToken();d&&(i.token=d)}catch(d){r?.(ae(d))}let a=_(o);a&&(i.sessionId||(i.sessionId=a));let c=qe(o);c!==void 0&&(i.geoLocation||(i.geoLocation=c)),o.waniwani=i}function Be(e,t){let n=H(t);if(!n||!S(e))return;S(e._meta)||(e._meta={});let r=e._meta,o=_(n);o&&!r[Ke]&&(r[Ke]=o);let s=qe(n);s&&(r[te]||(r[te]=s),r[ne]||(r[ne]=s))}function qe(e){if(!e)return;let t=e[te]??e[ne];if(S(t)||typeof t=="string")return t}function ae(e){return e instanceof Error?e:new Error(String(e))}var Le="https://app.waniwani.ai";function Rt(e,t){let n=e;if(n.__waniwaniWrapped)return n;n.__waniwaniWrapped=!0;let r=t.client,o=t.injectWidgetToken!==!1,s=null;function i(){if(s)return s;let c=r._config.apiKey;return c?(s=new z({baseUrl:r._config.baseUrl??Le,apiKey:c}),s):null}let a=e.registerTool.bind(e);return n.registerTool=((...c)=>{let[d,f,h]=c,l=typeof d=="string"&&d.trim().length>0?d:"unknown";if(typeof h!="function")return a(...c);let p=h;return a(d,f,async(y,u)=>{let T=H(u)??{},x=me(r,T);S(u)&&(u[J]=x);let b=performance.now(),E=e.server?.getClientVersion?.();try{let k=await p(y,u),P=Math.round(performance.now()-b),I=S(k)&&k.isError===!0;if(I){let ce=re(k);console.error(`[waniwani] Tool "${l}" returned error${ce?`: ${ce}`:""}`)}return await ie(r,oe(l,u,t,{durationMs:P,status:I?"error":"ok",...I&&{errorMessage:re(k)??"Unknown tool error"}},E,{input:y,output:k}),t.onError),t.flushAfterToolCall&&await se(r,t.onError),Be(k,u),o&&await $e(k,i(),r._config.baseUrl??Le,t.onError),k}catch(k){let P=Math.round(performance.now()-b);throw await ie(r,oe(l,u,t,{durationMs:P,status:"error",errorMessage:k instanceof Error?k.message:String(k)},E,{input:y}),t.onError),t.flushAfterToolCall&&await se(r,t.onError),k}})}),n}export{C as END,R as START,W as StateGraph,Ce as createFlow,_e as createFlowTestHarness,We as createResource,rt as createTool,St as createTrackingRoute,Y as detectPlatform,ze as isMCPApps,Ve as isOpenAI,ot as registerTools,Rt as withWaniwani};
3
+ ${I.context}`:`ERROR: ${E}`}:I),P=ve(k,p.context,i,c);if(P)return P;break}}let u=r.get(i);if(!u)return{content:{status:"error",error:`No outgoing edge from node "${i}"`}};i=await W(u,c);continue}if(ge(p)){let y=p.field;if(y&&G(D(c,y))){let u=r.get(i);if(!u)return{content:{status:"error",error:`No outgoing edge from node "${i}"`}};i=await W(u,c);continue}return{content:{status:"widget",tool:p.tool.id,data:p.data,description:p.description,interactive:p.interactive!==!1},flowTokenContent:{step:i,state:c,field:y,widgetId:p.tool.id}}}c=F(c,p);let w=r.get(i);if(!w)return{content:{status:"error",error:`No outgoing edge from node "${i}"`}};i=await W(w,c)}catch(f){return{content:{status:"error",error:f instanceof Error?f.message:String(f)},flowTokenContent:{step:i,state:c}}}}return{content:{status:"error",error:"Flow exceeded maximum iterations (possible infinite loop)"}}}var et="@waniwani/sdk",tt="https://app.waniwani.ai",U=class{baseUrl;apiKey;constructor(t){this.baseUrl=(t?.baseUrl??process.env.WANIWANI_BASE_URL??tt).replace(/\/$/,""),this.apiKey=t?.apiKey??process.env.WANIWANI_API_KEY}async get(t){if(!this.apiKey)return null;try{return await this.request("/api/mcp/redis/get",{key:t})??null}catch{return null}}async set(t,n){if(this.apiKey)try{await this.request("/api/mcp/redis/set",{key:t,value:n})}catch{}}async delete(t){if(this.apiKey)try{await this.request("/api/mcp/redis/delete",{key:t})}catch{}}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 a=await o.text().catch(()=>"");throw new Error(a||`Flow state API error: HTTP ${o.status}`)}return(await o.json()).data}};function xe(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 Re(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. 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.state){let n=[];for(let[r,o]of Object.entries(e.state)){let s=ke(o);if(s){let a=o.description??"",i=Object.entries(s).map(([c,d])=>{let l=xe(d);return l?`\`${r}.${c}\` (${l})`:`\`${r}.${c}\``}).join(", ");n.push(a?`\`${r}\` (${a}): ${i}`:`\`${r}\`: ${i}`)}else{let a=xe(o);n.push(a?`\`${r}\` (${a})`:`\`${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
+ `)}var nt={action:$.enum(["start","continue"]).describe('"start" to begin the flow, "continue" to resume after a pause (interrupt or widget)'),stateUpdates:$.record($.string(),$.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 Ee(e){let{config:t,nodes:n,edges:r}=e,o=Re(t),s=`${t.description}
5
+ ${o}`,a=e.store??new U,i=new Map;async function c(d,l,g,f){if(d.action==="start"){let p=r.get(R);if(!p)return{content:{status:"error",error:"No start edge"}};let w=X(d.stateUpdates??{}),y=await W(p,w);return K(y,w,n,r,i,g,f)}if(d.action==="continue"){if(!l)return{content:{status:"error",error:"No session ID available for continue action."}};let p=await a.get(l);if(!p)return{content:{status:"error",error:"Flow state not found. The flow may have expired."}};let w=p.state,y=p.step;if(!y)return{content:{status:"error",error:"Flow state is missing the current step. The flow may have expired."}};let u=F(w,X(d.stateUpdates??{}));if(p.widgetId){let T=r.get(y);if(!T)return{content:{status:"error",error:`No edge from step "${y}"`}};let x=await W(T,u);return K(x,u,n,r,i,g,f)}return K(y,u,n,r,i,g,f)}return{content:{status:"error",error:`Unknown action: "${d.action}"`}}}return{id:t.id,title:t.title,description:s,graph:e.graph,async register(d){d.registerTool(t.id,{title:t.title,description:s,inputSchema:nt,annotations:t.annotations},(async(l,g)=>{let f=g,p=f._meta??{},w=_(p),y=j(f),u=await c(l,w,p,y);return u.flowTokenContent&&w&&await a.set(w,u.flowTokenContent),{content:[{type:"text",text:JSON.stringify(u.content,null,2)}],_meta:p,...u.content.status==="error"?{isError:!0}:{}}}))}}}function Ie(e,t){let n=["flowchart TD"];n.push(` ${R}((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 A=class{nodes=new Map;edges=new Map;config;constructor(t){this.config=t}addNode(t,n){if(t===R||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),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 Ie(this.nodes,this.edges)}compile(t){this.validate();let n=new Map(this.nodes),r=new Map(this.edges);return Ee({config:this.config,nodes:n,edges:r,store:t?.store,graph:()=>Ie(n,r)})}validate(){if(!this.edges.has(R))throw new Error('Flow must have an entry point. Add an edge from START: .addEdge(START, "first_node")');let t=this.edges.get(R);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!==R&&!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 Ce(e){return new A(e)}function be(e){let t=e.content;return JSON.parse(t[0]?.text??"")}async function _e(e,t){let n=t?.stateStore,r=[],o=`test-session-${Math.random().toString(36).slice(2,10)}`,s={registerTool:(...d)=>{r.push(d)}};await e.register(s);let a=r[0]?.[2];if(!a)throw new Error(`Flow "${e.id}" did not register a handler`);let i={_meta:{sessionId:o}};async function c(d){return{...d,decodedState:n?await n.get(o):null}}return{async start(d){let l=await a({action:"start",...d?{stateUpdates:d}:{}},i);return c(be(l))},async continueWith(d){let l=await a({action:"continue",...d?{stateUpdates:d}:{}},i);return c(be(l))},async lastState(){return n?n.get(o):null}}}var B="text/html+skybridge",q="text/html;profile=mcp-app",Pe=async(e,t)=>{let n=e.endsWith("/")?e.slice(0,-1):e;return await(await fetch(`${n}${t}`)).text()};function Me(e){return{"openai/widgetDescription":e.description,"openai/widgetPrefersBorder":e.prefersBorder,"openai/widgetDomain":e.widgetDomain,...e.widgetCSP&&{"openai/widgetCSP":e.widgetCSP}}}function Fe(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 Z(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 We(e){let{id:t,title:n,description:r,baseUrl:o,htmlPath:s,widgetDomain:a,prefersBorder:i=!0,autoHeight:c=!0}=e,d=e.widgetCSP??{connect_domains:[o],resource_domains:[o]};if(process.env.NODE_ENV==="development")try{let{hostname:u}=new URL(o);(u==="localhost"||u==="127.0.0.1")&&(d={...d,connect_domains:[...d.connect_domains||[],`ws://${u}:*`,`wss://${u}:*`],resource_domains:[...d.resource_domains||[],`http://${u}:*`]})}catch{}let l=`ui://widgets/apps-sdk/${t}.html`,g=`ui://widgets/ext-apps/${t}.html`,f=null,p=()=>(f||(f=Pe(o,s)),f),w=r;async function y(u){let T=await p();u.registerResource(`${t}-openai-widget`,l,{title:n,description:w,mimeType:B,_meta:{"openai/widgetDescription":w,"openai/widgetPrefersBorder":i}},async x=>({contents:[{uri:x.href,mimeType:B,text:T,_meta:Me({description:w,prefersBorder:i,widgetDomain:a,widgetCSP:d})}]})),u.registerResource(`${t}-mcp-widget`,g,{title:n,description:w,mimeType:q,_meta:{ui:{prefersBorder:i}}},async x=>({contents:[{uri:x.href,mimeType:q,text:T,_meta:Fe({description:w,prefersBorder:i,widgetDomain:a,widgetCSP:d})}]}))}return{id:t,title:n,description:r,openaiUri:l,mcpUri:g,autoHeight:c,register:y}}function rt(e,t){let{resource:n,description:r,inputSchema:o,annotations:s,autoInjectResultText:a=!0}=e,i=e.id??n?.id,c=e.title??n?.title;if(!i)throw new Error("createTool: `id` is required when no resource is provided");if(!c)throw new Error("createTool: `title` is required when no resource is provided");let d=n?Z({openaiTemplateUri:n.openaiUri,mcpTemplateUri:n.mcpUri,invoking:e.invoking??"Loading...",invoked:e.invoked??"Loaded",autoHeight:n.autoHeight}):void 0;return{id:i,title:c,description:r,async register(l){l.registerTool(i,{title:c,description:r,inputSchema:o,annotations:s,...d&&{_meta:d}},(async(g,f)=>{let p=f,w=p._meta??{},y=j(p),u=await t(g,{extra:{_meta:w},waniwani:y});return n&&u.data?{content:[{type:"text",text:u.text}],structuredContent:u.data,_meta:{...d,...w,...a===!1?{"waniwani/autoInjectResultText":!1}:{}}}:{content:[{type:"text",text:u.text}],...u.data?{structuredContent:u.data}:{},...a===!1?{_meta:{"waniwani/autoInjectResultText":!1}}:{}}}))}}}async function ot(e,t){await Promise.all(t.map(n=>n.register(e)))}var L=class extends Error{constructor(n,r){super(n);this.status=r;this.name="WaniWaniError"}};var it="@waniwani/sdk";function Ae(e){let{baseUrl:t,apiKey:n}=e;function r(){if(!n)throw new Error("WANIWANI_API_KEY is not set");return n}async function o(s,a,i){let c=r(),d=`${t.replace(/\/$/,"")}${a}`,l={Authorization:`Bearer ${c}`,"X-WaniWani-SDK":it},g={method:s,headers:l};i!==void 0&&(l["Content-Type"]="application/json",g.body=JSON.stringify(i));let f=await fetch(d,g);if(!f.ok){let w=await f.text().catch(()=>"");throw new L(w||`KB API error: HTTP ${f.status}`,f.status)}return(await f.json()).data}return{async ingest(s){return o("POST","/api/mcp/kb/ingest",{files:s})},async search(s,a){return o("POST","/api/mcp/kb/search",{query:s,...a})},async sources(){return o("GET","/api/mcp/kb/sources")}}}var st="@waniwani/sdk";function z(e,t={}){let n=t.now??(()=>new Date),r=t.generateId??Ne,o=dt(e),s=V(e.meta),a=V(e.metadata),i=ut(e,s),c=v(e.eventId)??r(),d=pt(e.timestamp,n),l=v(e.source)??M(s)??t.source??st,g=Q(e)?{...e}:void 0,f={...a};return Object.keys(s).length>0&&(f.meta=s),g&&(f.rawLegacy=g),{id:c,type:"mcp.event",name:o,source:l,timestamp:d,correlation:i,properties:at(e,o),metadata:f,rawLegacy:g}}function Ne(){return typeof crypto<"u"&&typeof crypto.randomUUID=="function"?`evt_${crypto.randomUUID()}`:`evt_${Math.random().toString(36).slice(2,10)}_${Date.now().toString(36)}`}function at(e,t){if(!Q(e))return V(e.properties);let n=ct(e,t),r=V(e.properties);return{...n,...r}}function ct(e,t){switch(t){case"tool.called":{let n={};return v(e.toolName)&&(n.name=e.toolName),v(e.toolType)&&(n.type=e.toolType),n}case"quote.succeeded":{let n={};return typeof e.quoteAmount=="number"&&(n.amount=e.quoteAmount),v(e.quoteCurrency)&&(n.currency=e.quoteCurrency),n}case"link.clicked":{let n={};return v(e.linkUrl)&&(n.url=e.linkUrl),n}case"purchase.completed":{let n={};return typeof e.purchaseAmount=="number"&&(n.amount=e.purchaseAmount),v(e.purchaseCurrency)&&(n.currency=e.purchaseCurrency),n}default:return{}}}function dt(e){return Q(e)?e.eventType:e.event}function ut(e,t){let n=v(e.requestId)??he(t),r=v(e.sessionId)??_(t),o=v(e.traceId)??we(t),s=v(e.externalUserId)??ye(t),a=v(e.correlationId)??Te(t)??n,i={};return r&&(i.sessionId=r),o&&(i.traceId=o),n&&(i.requestId=n),a&&(i.correlationId=a),s&&(i.externalUserId=s),i}function pt(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 V(e){return!e||typeof e!="object"||Array.isArray(e)?{}:e}function v(e){if(typeof e=="string"&&e.trim().length!==0)return e}function Q(e){return"eventType"in e}var lt="/api/mcp/events/v2/batch";var De="@waniwani/sdk",ft=new Set([401,403]),gt=new Set([408,425,429,500,502,503,504]);function Ue(e){return new ee(e)}var ee=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=wt(t.baseUrl,t.endpointPath??lt),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":De},body:JSON.stringify(this.makeBatchRequest(t))})}catch(s){return{kind:"retryable",reason:yt(s)}}if(ft.has(n.status))return{kind:"auth",status:n.status};if(gt.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:De,version:this.sdkVersion??"0.0.0"},events:t}}classifyRejectedEvents(t,n){let r=new Map(t.map(a=>[a.id,a])),o=[],s=[];for(let a of n){let i=r.get(a.eventId);if(i){if(mt(a)){o.push(i);continue}s.push(i)}}return{retryable:o,permanent:s}}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 mt(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 wt(e,t){let n=e.endsWith("/")?e:`${e}/`,r=t.startsWith("/")?t.slice(1):t;return`${n}${r}`}function yt(e){return e instanceof Error?e.message:String(e)}function Oe(e){let{baseUrl:t,apiKey:n,tracking:r}=e;function o(){if(!n)throw new Error("WANIWANI_API_KEY is not set");return n}let s=n?Ue({baseUrl: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,a={async identify(i,c,d){o();let l=z({event:"user.identified",externalUserId:i,properties:c,meta:d});return s?.enqueue(l),{eventId:l.id}},async track(i){o();let c=z(i);return s?.enqueue(c),{eventId:c.id}},async flush(){o(),await s?.flush()},async shutdown(i){return o(),await s?.shutdown({timeoutMs:i?.timeoutMs??r.shutdownTimeoutMs})??{timedOut:!1,pendingEvents:0}}};return s&&Tt(a,r.shutdownTimeoutMs),a}function Tt(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 je(e){let t=e?.baseUrl??"https://app.waniwani.ai",n=e?.apiKey??process.env.WANIWANI_API_KEY,r={endpointPath:e?.tracking?.endpointPath??"/api/mcp/events/v2/batch",flushIntervalMs:e?.tracking?.flushIntervalMs??1e3,maxBatchSize:e?.tracking?.maxBatchSize??20,maxBufferSize:e?.tracking?.maxBufferSize??1e3,maxRetries:e?.tracking?.maxRetries??3,retryBaseDelayMs:e?.tracking?.retryBaseDelayMs??200,retryMaxDelayMs:e?.tracking?.retryMaxDelayMs??2e3,shutdownTimeoutMs:e?.tracking?.shutdownTimeoutMs??2e3},o={baseUrl:t,apiKey:n,tracking:r},s=Oe(o),a=Ae(o);return{...s,kb:a,_config:o}}function kt(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 St(e){let t={apiKey:e?.apiKey,baseUrl:e?.baseUrl},n;function r(){return n||(n=je(t)),n}return async function(s){let a;try{a=await s.json()}catch{return new Response(JSON.stringify({error:"Invalid JSON"}),{status:400,headers:{"Content-Type":"application/json"}})}if(!Array.isArray(a.events)||a.events.length===0)return new Response(JSON.stringify({error:"Missing or empty events array"}),{status:400,headers:{"Content-Type":"application/json"}});try{let i=r(),c=[];for(let d of a.events){let l=kt(d),g=await i.track(l);c.push(g.eventId)}return await i.flush(),new Response(JSON.stringify({ok:!0,accepted:c.length}),{status:200,headers:{"Content-Type":"application/json"}})}catch(i){let c=i instanceof Error?i.message:"Unknown error";return new Response(JSON.stringify({error:c}),{status:500,headers:{"Content-Type":"application/json"}})}}}var H=class{cached=null;pending=null;config;constructor(t){this.config=t}async getToken(t,n){return this.cached&&Date.now()<this.cached.expiresAt-12e4?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=vt(this.config.baseUrl,"/api/mcp/widget-tokens"),o={};t&&(o.sessionId=t),n&&(o.traceId=n);try{let s=await fetch(r,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${this.config.apiKey}`},body:JSON.stringify(o)});if(!s.ok)return null;let a=await s.json(),i=a.data&&typeof a.data=="object"?a.data:a,c=new Date(i.expiresAt).getTime();return!i.token||Number.isNaN(c)?null:(this.cached={token:i.token,expiresAt:c},i.token)}catch{return null}}};function vt(e,t){return`${e.endsWith("/")?e.slice(0,-1):e}${t}`}var Ke="waniwani/sessionId",te="waniwani/geoLocation",ne="waniwani/userLocation";function S(e){return!!e&&typeof e=="object"&&!Array.isArray(e)}function O(e){if(!S(e))return;let t=e._meta;return S(t)?t:void 0}function re(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 xt(e,t){return typeof t=="function"?t(e)??"other":t??"other"}function oe(e,t,n,r,o,s){let a=xt(e,n.toolType),i=O(t);return console.log("[waniwani:debug] buildTrackInput meta:",JSON.stringify(i),"-> source:",M(i)),{event:"tool.called",properties:{name:e,type:a,...r??{},...s?.input!==void 0&&{input:s.input},...s?.output!==void 0&&{output:s.output}},meta:i,source:M(i),metadata:{...n.metadata??{},...o&&{clientInfo:o}}}}async function ie(e,t,n){try{await e.track(t)}catch(r){n?.(ae(r))}}async function se(e,t){try{await e.flush()}catch(n){t?.(ae(n))}}async function $e(e,t,n,r,o){if(!S(e))return;S(e._meta)||(e._meta={});let s=e._meta,a=S(s.waniwani)?s.waniwani:void 0,i={...a??{},endpoint:a?.endpoint??`${n.replace(/\/$/,"")}/api/mcp/events/v2/batch`};if(t)try{let g=await t.getToken();g&&(i.token=g)}catch(g){o?.(ae(g))}let c=_(s);c&&(i.sessionId||(i.sessionId=c));let d=qe(s);d!==void 0&&(i.geoLocation||(i.geoLocation=d));let l=M(O(r));l&&!i.source&&(i.source=l),s.waniwani=i}function Be(e,t){let n=O(t);if(!n||!S(e))return;S(e._meta)||(e._meta={});let r=e._meta,o=_(n);o&&!r[Ke]&&(r[Ke]=o);let s=qe(n);s&&(r[te]||(r[te]=s),r[ne]||(r[ne]=s))}function qe(e){if(!e)return;let t=e[te]??e[ne];if(S(t)||typeof t=="string")return t}function ae(e){return e instanceof Error?e:new Error(String(e))}var Le="https://app.waniwani.ai";function Rt(e,t){let n=e;if(n.__waniwaniWrapped)return n;n.__waniwaniWrapped=!0;let r=t.client,o=t.injectWidgetToken!==!1,s=null;function a(){if(s)return s;let c=r._config.apiKey;return c?(s=new H({baseUrl:r._config.baseUrl??Le,apiKey:c}),s):null}let i=e.registerTool.bind(e);return n.registerTool=((...c)=>{let[d,l,g]=c,f=typeof d=="string"&&d.trim().length>0?d:"unknown";if(typeof g!="function")return i(...c);let p=g;return i(d,l,async(y,u)=>{let T=O(u)??{},x=me(r,T);S(u)&&(u[J]=x);let b=performance.now(),E=e.server?.getClientVersion?.();try{let k=await p(y,u),P=Math.round(performance.now()-b),I=S(k)&&k.isError===!0;if(I){let ce=re(k);console.error(`[waniwani] Tool "${f}" returned error${ce?`: ${ce}`:""}`)}return await ie(r,oe(f,u,t,{durationMs:P,status:I?"error":"ok",...I&&{errorMessage:re(k)??"Unknown tool error"}},E,{input:y,output:k}),t.onError),t.flushAfterToolCall&&await se(r,t.onError),Be(k,u),o&&await $e(k,a(),r._config.baseUrl??Le,u,t.onError),k}catch(k){let P=Math.round(performance.now()-b);throw await ie(r,oe(f,u,t,{durationMs:P,status:"error",errorMessage:k instanceof Error?k.message:String(k)},E,{input:y}),t.onError),t.flushAfterToolCall&&await se(r,t.onError),k}})}),n}export{C as END,R as START,A as StateGraph,Ce as createFlow,_e as createFlowTestHarness,We as createResource,rt as createTool,St as createTrackingRoute,Y as detectPlatform,ze as isMCPApps,Ve as isOpenAI,ot as registerTools,Rt as withWaniwani};
7
7
  //# sourceMappingURL=index.js.map