sunpeak 0.19.2 → 0.19.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.
Files changed (90) hide show
  1. package/README.md +6 -4
  2. package/bin/commands/dev.mjs +1 -1
  3. package/bin/commands/inspect.mjs +1 -1
  4. package/bin/commands/new.mjs +9 -5
  5. package/bin/commands/start.mjs +3 -1
  6. package/bin/commands/test-init.mjs +478 -76
  7. package/bin/commands/test.mjs +357 -4
  8. package/bin/lib/eval/eval-reporter.mjs +105 -0
  9. package/bin/lib/eval/eval-runner.mjs +310 -0
  10. package/bin/lib/eval/eval-types.d.mts +168 -0
  11. package/bin/lib/eval/eval-vitest-plugin.mjs +158 -0
  12. package/bin/lib/eval/model-registry.mjs +73 -0
  13. package/bin/lib/sandbox-server.mjs +5 -2
  14. package/bin/sunpeak.js +1 -0
  15. package/dist/chatgpt/index.cjs +1 -1
  16. package/dist/chatgpt/index.js +1 -1
  17. package/dist/claude/index.cjs +1 -1
  18. package/dist/claude/index.js +1 -1
  19. package/dist/host/chatgpt/index.cjs +1 -1
  20. package/dist/host/chatgpt/index.js +1 -1
  21. package/dist/index.cjs +134 -124
  22. package/dist/index.cjs.map +1 -1
  23. package/dist/index.d.ts +3 -1
  24. package/dist/index.js +71 -62
  25. package/dist/index.js.map +1 -1
  26. package/dist/inspector/index.cjs +1 -1
  27. package/dist/inspector/index.js +1 -1
  28. package/dist/{inspector-Cdo5BK2D.js → inspector-D5DckQuU.js} +236 -98
  29. package/dist/inspector-D5DckQuU.js.map +1 -0
  30. package/dist/{inspector-8nPV2A-z.cjs → inspector-jY9O18z9.cjs} +237 -99
  31. package/dist/inspector-jY9O18z9.cjs.map +1 -0
  32. package/dist/mcp/index.cjs +237 -140
  33. package/dist/mcp/index.cjs.map +1 -1
  34. package/dist/mcp/index.d.ts +1 -1
  35. package/dist/mcp/index.js +230 -134
  36. package/dist/mcp/index.js.map +1 -1
  37. package/dist/mcp/production-server.d.ts +31 -0
  38. package/dist/{protocol-C7kTcBr_.cjs → protocol-C8pFDmcy.cjs} +8194 -8187
  39. package/dist/protocol-C8pFDmcy.cjs.map +1 -0
  40. package/dist/{protocol-BfAACnv0.js → protocol-CRqiPTLT.js} +8186 -8185
  41. package/dist/protocol-CRqiPTLT.js.map +1 -0
  42. package/dist/{use-app-CfP9VypY.js → use-app-Bfargfa3.js} +194 -94
  43. package/dist/use-app-Bfargfa3.js.map +1 -0
  44. package/dist/{use-app-CzcYw1Kz.cjs → use-app-CbsBEmwv.cjs} +254 -148
  45. package/dist/use-app-CbsBEmwv.cjs.map +1 -0
  46. package/package.json +27 -3
  47. package/template/README.md +17 -7
  48. package/template/_gitignore +2 -0
  49. package/template/dist/albums/albums.html +15 -15
  50. package/template/dist/albums/albums.json +1 -1
  51. package/template/dist/carousel/carousel.html +19 -19
  52. package/template/dist/carousel/carousel.json +1 -1
  53. package/template/dist/map/map.html +14 -14
  54. package/template/dist/map/map.json +1 -1
  55. package/template/dist/review/review.html +11 -11
  56. package/template/dist/review/review.json +1 -1
  57. package/template/node_modules/.bin/vitest +2 -2
  58. package/template/node_modules/.vite/deps/_metadata.json +3 -3
  59. package/template/node_modules/.vite-mcp/deps/@modelcontextprotocol_ext-apps.js +192 -91
  60. package/template/node_modules/.vite-mcp/deps/@modelcontextprotocol_ext-apps.js.map +1 -1
  61. package/template/node_modules/.vite-mcp/deps/@modelcontextprotocol_ext-apps_app-bridge.js +231 -92
  62. package/template/node_modules/.vite-mcp/deps/@modelcontextprotocol_ext-apps_app-bridge.js.map +1 -1
  63. package/template/node_modules/.vite-mcp/deps/@modelcontextprotocol_ext-apps_react.js +208 -105
  64. package/template/node_modules/.vite-mcp/deps/@modelcontextprotocol_ext-apps_react.js.map +1 -1
  65. package/template/node_modules/.vite-mcp/deps/_metadata.json +25 -25
  66. package/template/node_modules/.vite-mcp/deps/{protocol-B_qKkui_.js → protocol-BqGB4zBx.js} +45 -45
  67. package/template/node_modules/.vite-mcp/deps/protocol-BqGB4zBx.js.map +1 -0
  68. package/template/node_modules/.vite-mcp/deps/vitest.js +7 -7
  69. package/template/node_modules/.vite-mcp/deps/vitest.js.map +1 -1
  70. package/template/tests/e2e/visual.spec.ts-snapshots/albums-dark-chatgpt-darwin.png +0 -0
  71. package/template/tests/e2e/visual.spec.ts-snapshots/albums-dark-claude-darwin.png +0 -0
  72. package/template/tests/e2e/visual.spec.ts-snapshots/albums-fullscreen-chatgpt-darwin.png +0 -0
  73. package/template/tests/e2e/visual.spec.ts-snapshots/albums-fullscreen-claude-darwin.png +0 -0
  74. package/template/tests/e2e/visual.spec.ts-snapshots/albums-light-chatgpt-darwin.png +0 -0
  75. package/template/tests/e2e/visual.spec.ts-snapshots/albums-light-claude-darwin.png +0 -0
  76. package/template/tests/e2e/visual.spec.ts-snapshots/albums-page-light-chatgpt-darwin.png +0 -0
  77. package/template/tests/e2e/visual.spec.ts-snapshots/albums-page-light-claude-darwin.png +0 -0
  78. package/template/tests/evals/.env.example +5 -0
  79. package/template/tests/evals/albums.eval.ts +28 -0
  80. package/template/tests/evals/carousel.eval.ts +26 -0
  81. package/template/tests/evals/eval.config.ts +26 -0
  82. package/template/tests/evals/map.eval.ts +23 -0
  83. package/template/tests/evals/review.eval.ts +48 -0
  84. package/dist/inspector-8nPV2A-z.cjs.map +0 -1
  85. package/dist/inspector-Cdo5BK2D.js.map +0 -1
  86. package/dist/protocol-BfAACnv0.js.map +0 -1
  87. package/dist/protocol-C7kTcBr_.cjs.map +0 -1
  88. package/dist/use-app-CfP9VypY.js.map +0 -1
  89. package/dist/use-app-CzcYw1Kz.cjs.map +0 -1
  90. package/template/node_modules/.vite-mcp/deps/protocol-B_qKkui_.js.map +0 -1
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-app-CbsBEmwv.cjs","names":["f","QQ","YQ","$Q","C","XQ","ZQ","DQ","b","bQ","RQ","CQ","UQ","kQ","MQ","HQ"],"sources":["../../../node_modules/.pnpm/@modelcontextprotocol+ext-apps@1.5.0_@modelcontextprotocol+sdk@1.29.0_zod@4.3.6__react-_f4871531d9cf52c692eb6edc1ee416ef/node_modules/@modelcontextprotocol/ext-apps/dist/src/app.js","../src/hooks/app-context.tsx","../src/hooks/use-app.ts"],"sourcesContent":["import{CallToolRequestSchema as RQ,CallToolResultSchema as UQ,EmptyResultSchema as HQ,ListResourcesResultSchema as MQ,ListToolsRequestSchema as CQ,PingRequestSchema as bQ,ReadResourceResultSchema as kQ}from\"@modelcontextprotocol/sdk/types.js\";import{Protocol as f}from\"@modelcontextprotocol/sdk/shared/protocol.js\";class z extends f{_registeredMethods=new Set;_eventSlots=new Map;onEventDispatch(X,Y){}_ensureEventSlot(X){let Y=this._eventSlots.get(X);if(!Y){let Z=this.eventSchemas[X];if(!Z)throw Error(`Unknown event: ${String(X)}`);Y={listeners:[]},this._eventSlots.set(X,Y);let $=Z.shape.method.value;this._registeredMethods.add($);let D=Y;super.setNotificationHandler(Z,(J)=>{let G=J.params;this.onEventDispatch(X,G),D.onHandler?.(G);for(let V of[...D.listeners])V(G)})}return Y}setEventHandler(X,Y){let Z=this._ensureEventSlot(X);if(Z.onHandler&&Y)console.warn(`[MCP Apps] on${String(X)} handler replaced. Use addEventListener(\"${String(X)}\", …) to add multiple listeners without replacing.`);Z.onHandler=Y}getEventHandler(X){return this._eventSlots.get(X)?.onHandler}addEventListener(X,Y){this._ensureEventSlot(X).listeners.push(Y)}removeEventListener(X,Y){let Z=this._eventSlots.get(X);if(!Z)return;let $=Z.listeners.indexOf(Y);if($!==-1)Z.listeners.splice($,1)}setRequestHandler=(X,Y)=>{this._assertMethodNotRegistered(X,\"setRequestHandler\"),super.setRequestHandler(X,Y)};setNotificationHandler=(X,Y)=>{this._assertMethodNotRegistered(X,\"setNotificationHandler\"),super.setNotificationHandler(X,Y)};warnIfRequestHandlerReplaced(X,Y,Z){if(Y&&Z)console.warn(`[MCP Apps] ${X} handler replaced. Previous handler will no longer be called.`)}replaceRequestHandler=(X,Y)=>{let Z=X.shape.method.value;this._registeredMethods.add(Z),super.setRequestHandler(X,Y)};_assertMethodNotRegistered(X,Y){let Z=X.shape.method.value;if(this._registeredMethods.has(Z))throw Error(`Handler for \"${Z}\" already registered (via ${Y}). Use addEventListener() to attach multiple listeners, or the on* setter for replace semantics.`);this._registeredMethods.add(Z)}}import{JSONRPCMessageSchema as QQ}from\"@modelcontextprotocol/sdk/types.js\";var B=\"2026-01-26\",u=\"ui/open-link\",d=\"ui/download-file\",h=\"ui/message\",i=\"ui/notifications/sandbox-proxy-ready\",m=\"ui/notifications/sandbox-resource-ready\",p=\"ui/notifications/size-changed\",r=\"ui/notifications/tool-input\",E=\"ui/notifications/tool-input-partial\",c=\"ui/notifications/tool-result\",l=\"ui/notifications/tool-cancelled\",n=\"ui/notifications/host-context-changed\",a=\"ui/notifications/request-teardown\",o=\"ui/resource-teardown\",s=\"ui/initialize\",t=\"ui/notifications/initialized\",e=\"ui/request-display-mode\";class N{eventTarget;eventSource;messageListener;constructor(X=window.parent,Y){this.eventTarget=X;this.eventSource=Y;this.messageListener=(Z)=>{if(Y&&Z.source!==this.eventSource){console.debug(\"Ignoring message from unknown source\",Z);return}let $=QQ.safeParse(Z.data);if($.success)console.debug(\"Parsed message\",$.data),this.onmessage?.($.data);else if(Z.data?.jsonrpc!==\"2.0\")console.debug(\"Ignoring non-JSON-RPC message\",$.error.message,Z);else console.error(\"Failed to parse message\",$.error.message,Z),this.onerror?.(Error(\"Invalid JSON-RPC message received: \"+$.error.message))}}async start(){window.addEventListener(\"message\",this.messageListener)}async send(X,Y){if(X.method!==E)console.debug(\"Sending message\",X);this.eventTarget.postMessage(X,\"*\")}async close(){window.removeEventListener(\"message\",this.messageListener),this.onclose?.()}onclose;onerror;onmessage;sessionId;setProtocolVersion}import{z as Q}from\"zod/v4\";import{ContentBlockSchema as C,CallToolResultSchema as XQ,EmbeddedResourceSchema as YQ,ImplementationSchema as b,RequestIdSchema as ZQ,ResourceLinkSchema as $Q,ToolSchema as DQ}from\"@modelcontextprotocol/sdk/types.js\";var k=Q.union([Q.literal(\"light\"),Q.literal(\"dark\")]).describe(\"Color theme preference for the host environment.\"),K=Q.union([Q.literal(\"inline\"),Q.literal(\"fullscreen\"),Q.literal(\"pip\")]).describe(\"Display mode for UI presentation.\"),JQ=Q.union([Q.literal(\"--color-background-primary\"),Q.literal(\"--color-background-secondary\"),Q.literal(\"--color-background-tertiary\"),Q.literal(\"--color-background-inverse\"),Q.literal(\"--color-background-ghost\"),Q.literal(\"--color-background-info\"),Q.literal(\"--color-background-danger\"),Q.literal(\"--color-background-success\"),Q.literal(\"--color-background-warning\"),Q.literal(\"--color-background-disabled\"),Q.literal(\"--color-text-primary\"),Q.literal(\"--color-text-secondary\"),Q.literal(\"--color-text-tertiary\"),Q.literal(\"--color-text-inverse\"),Q.literal(\"--color-text-ghost\"),Q.literal(\"--color-text-info\"),Q.literal(\"--color-text-danger\"),Q.literal(\"--color-text-success\"),Q.literal(\"--color-text-warning\"),Q.literal(\"--color-text-disabled\"),Q.literal(\"--color-border-primary\"),Q.literal(\"--color-border-secondary\"),Q.literal(\"--color-border-tertiary\"),Q.literal(\"--color-border-inverse\"),Q.literal(\"--color-border-ghost\"),Q.literal(\"--color-border-info\"),Q.literal(\"--color-border-danger\"),Q.literal(\"--color-border-success\"),Q.literal(\"--color-border-warning\"),Q.literal(\"--color-border-disabled\"),Q.literal(\"--color-ring-primary\"),Q.literal(\"--color-ring-secondary\"),Q.literal(\"--color-ring-inverse\"),Q.literal(\"--color-ring-info\"),Q.literal(\"--color-ring-danger\"),Q.literal(\"--color-ring-success\"),Q.literal(\"--color-ring-warning\"),Q.literal(\"--font-sans\"),Q.literal(\"--font-mono\"),Q.literal(\"--font-weight-normal\"),Q.literal(\"--font-weight-medium\"),Q.literal(\"--font-weight-semibold\"),Q.literal(\"--font-weight-bold\"),Q.literal(\"--font-text-xs-size\"),Q.literal(\"--font-text-sm-size\"),Q.literal(\"--font-text-md-size\"),Q.literal(\"--font-text-lg-size\"),Q.literal(\"--font-heading-xs-size\"),Q.literal(\"--font-heading-sm-size\"),Q.literal(\"--font-heading-md-size\"),Q.literal(\"--font-heading-lg-size\"),Q.literal(\"--font-heading-xl-size\"),Q.literal(\"--font-heading-2xl-size\"),Q.literal(\"--font-heading-3xl-size\"),Q.literal(\"--font-text-xs-line-height\"),Q.literal(\"--font-text-sm-line-height\"),Q.literal(\"--font-text-md-line-height\"),Q.literal(\"--font-text-lg-line-height\"),Q.literal(\"--font-heading-xs-line-height\"),Q.literal(\"--font-heading-sm-line-height\"),Q.literal(\"--font-heading-md-line-height\"),Q.literal(\"--font-heading-lg-line-height\"),Q.literal(\"--font-heading-xl-line-height\"),Q.literal(\"--font-heading-2xl-line-height\"),Q.literal(\"--font-heading-3xl-line-height\"),Q.literal(\"--border-radius-xs\"),Q.literal(\"--border-radius-sm\"),Q.literal(\"--border-radius-md\"),Q.literal(\"--border-radius-lg\"),Q.literal(\"--border-radius-xl\"),Q.literal(\"--border-radius-full\"),Q.literal(\"--border-width-regular\"),Q.literal(\"--shadow-hairline\"),Q.literal(\"--shadow-sm\"),Q.literal(\"--shadow-md\"),Q.literal(\"--shadow-lg\")]).describe(\"CSS variable keys available to MCP apps for theming.\"),KQ=Q.record(JQ.describe(`Style variables for theming MCP apps.\n\nIndividual style keys are optional - hosts may provide any subset of these values.\nValues are strings containing CSS values (colors, sizes, font stacks, etc.).\n\nNote: This type uses \\`Record<K, string | undefined>\\` rather than \\`Partial<Record<K, string>>\\`\nfor compatibility with Zod schema generation. Both are functionally equivalent for validation.`),Q.union([Q.string(),Q.undefined()]).describe(`Style variables for theming MCP apps.\n\nIndividual style keys are optional - hosts may provide any subset of these values.\nValues are strings containing CSS values (colors, sizes, font stacks, etc.).\n\nNote: This type uses \\`Record<K, string | undefined>\\` rather than \\`Partial<Record<K, string>>\\`\nfor compatibility with Zod schema generation. Both are functionally equivalent for validation.`)).describe(`Style variables for theming MCP apps.\n\nIndividual style keys are optional - hosts may provide any subset of these values.\nValues are strings containing CSS values (colors, sizes, font stacks, etc.).\n\nNote: This type uses \\`Record<K, string | undefined>\\` rather than \\`Partial<Record<K, string>>\\`\nfor compatibility with Zod schema generation. Both are functionally equivalent for validation.`),GQ=Q.object({method:Q.literal(\"ui/open-link\"),params:Q.object({url:Q.string().describe(\"URL to open in the host's browser\")})}),O=Q.object({isError:Q.boolean().optional().describe(\"True if the host failed to open the URL (e.g., due to security policy).\")}).passthrough(),I=Q.object({isError:Q.boolean().optional().describe(\"True if the download failed (e.g., user cancelled or host denied).\")}).passthrough(),w=Q.object({isError:Q.boolean().optional().describe(\"True if the host rejected or failed to deliver the message.\")}).passthrough(),VQ=Q.object({method:Q.literal(\"ui/notifications/sandbox-proxy-ready\"),params:Q.object({})}),W=Q.object({connectDomains:Q.array(Q.string()).optional().describe(`Origins for network requests (fetch/XHR/WebSocket).\n\n- Maps to CSP \\`connect-src\\` directive\n- Empty or omitted → no network connections (secure default)`),resourceDomains:Q.array(Q.string()).optional().describe(\"Origins for static resources (images, scripts, stylesheets, fonts, media).\\n\\n- Maps to CSP `img-src`, `script-src`, `style-src`, `font-src`, `media-src` directives\\n- Wildcard subdomains supported: `https://*.example.com`\\n- Empty or omitted → no network resources (secure default)\"),frameDomains:Q.array(Q.string()).optional().describe(\"Origins for nested iframes.\\n\\n- Maps to CSP `frame-src` directive\\n- Empty or omitted → no nested iframes allowed (`frame-src 'none'`)\"),baseUriDomains:Q.array(Q.string()).optional().describe(\"Allowed base URIs for the document.\\n\\n- Maps to CSP `base-uri` directive\\n- Empty or omitted → only same origin allowed (`base-uri 'self'`)\")}),j=Q.object({camera:Q.object({}).optional().describe(\"Request camera access.\\n\\nMaps to Permission Policy `camera` feature.\"),microphone:Q.object({}).optional().describe(\"Request microphone access.\\n\\nMaps to Permission Policy `microphone` feature.\"),geolocation:Q.object({}).optional().describe(\"Request geolocation access.\\n\\nMaps to Permission Policy `geolocation` feature.\"),clipboardWrite:Q.object({}).optional().describe(\"Request clipboard write access.\\n\\nMaps to Permission Policy `clipboard-write` feature.\")}),NQ=Q.object({method:Q.literal(\"ui/notifications/size-changed\"),params:Q.object({width:Q.number().optional().describe(\"New width in pixels.\"),height:Q.number().optional().describe(\"New height in pixels.\")})}),A=Q.object({method:Q.literal(\"ui/notifications/tool-input\"),params:Q.object({arguments:Q.record(Q.string(),Q.unknown().describe(\"Complete tool call arguments as key-value pairs.\")).optional().describe(\"Complete tool call arguments as key-value pairs.\")})}),F=Q.object({method:Q.literal(\"ui/notifications/tool-input-partial\"),params:Q.object({arguments:Q.record(Q.string(),Q.unknown().describe(\"Partial tool call arguments (incomplete, may change).\")).optional().describe(\"Partial tool call arguments (incomplete, may change).\")})}),P=Q.object({method:Q.literal(\"ui/notifications/tool-cancelled\"),params:Q.object({reason:Q.string().optional().describe('Optional reason for the cancellation (e.g., \"user action\", \"timeout\").')})}),g=Q.object({fonts:Q.string().optional()}),x=Q.object({variables:KQ.optional().describe(\"CSS variables for theming the app.\"),css:g.optional().describe(\"CSS blocks that apps can inject.\")}),q=Q.object({method:Q.literal(\"ui/resource-teardown\"),params:Q.object({})}),WQ=Q.record(Q.string(),Q.unknown()),_=Q.object({text:Q.object({}).optional().describe(\"Host supports text content blocks.\"),image:Q.object({}).optional().describe(\"Host supports image content blocks.\"),audio:Q.object({}).optional().describe(\"Host supports audio content blocks.\"),resource:Q.object({}).optional().describe(\"Host supports resource content blocks.\"),resourceLink:Q.object({}).optional().describe(\"Host supports resource link content blocks.\"),structuredContent:Q.object({}).optional().describe(\"Host supports structured content.\")}),jQ=Q.object({method:Q.literal(\"ui/notifications/request-teardown\"),params:Q.object({}).optional()}),S=Q.object({experimental:Q.object({}).optional().describe(\"Experimental features (structure TBD).\"),openLinks:Q.object({}).optional().describe(\"Host supports opening external URLs.\"),downloadFile:Q.object({}).optional().describe(\"Host supports file downloads via ui/download-file.\"),serverTools:Q.object({listChanged:Q.boolean().optional().describe(\"Host supports tools/list_changed notifications.\")}).optional().describe(\"Host can proxy tool calls to the MCP server.\"),serverResources:Q.object({listChanged:Q.boolean().optional().describe(\"Host supports resources/list_changed notifications.\")}).optional().describe(\"Host can proxy resource reads to the MCP server.\"),logging:Q.object({}).optional().describe(\"Host accepts log messages.\"),sandbox:Q.object({permissions:j.optional().describe(\"Permissions granted by the host (camera, microphone, geolocation).\"),csp:W.optional().describe(\"CSP domains approved by the host.\")}).optional().describe(\"Sandbox configuration applied by the host.\"),updateModelContext:_.optional().describe(\"Host accepts context updates (ui/update-model-context) to be included in the model's context for future turns.\"),message:_.optional().describe(\"Host supports receiving content messages (ui/message) from the view.\")}),y=Q.object({experimental:Q.object({}).optional().describe(\"Experimental features (structure TBD).\"),tools:Q.object({listChanged:Q.boolean().optional().describe(\"App supports tools/list_changed notifications.\")}).optional().describe(\"App exposes MCP-style tools that the host can call.\"),availableDisplayModes:Q.array(K).optional().describe(\"Display modes the app supports.\")}),LQ=Q.object({method:Q.literal(\"ui/notifications/initialized\"),params:Q.object({}).optional()}),zQ=Q.object({csp:W.optional().describe(\"Content Security Policy configuration for UI resources.\"),permissions:j.optional().describe(\"Sandbox permissions requested by the UI resource.\"),domain:Q.string().optional().describe(`Dedicated origin for view sandbox.\n\nUseful when views need stable, dedicated origins for OAuth callbacks, CORS policies, or API key allowlists.\n\n**Host-dependent:** The format and validation rules for this field are determined by each host. Servers MUST consult host-specific documentation for the expected domain format. Common patterns include:\n- Hash-based subdomains (e.g., \\`{hash}.claudemcpcontent.com\\`)\n- URL-derived subdomains (e.g., \\`www-example-com.oaiusercontent.com\\`)\n\nIf omitted, host uses default sandbox origin (typically per-conversation).`),prefersBorder:Q.boolean().optional().describe(`Visual boundary preference - true if view prefers a visible border.\n\nBoolean requesting whether a visible border and background is provided by the host. Specifying an explicit value for this is recommended because hosts' defaults may vary.\n\n- \\`true\\`: request visible border + background\n- \\`false\\`: request no visible border + background\n- omitted: host decides border`)}),BQ=Q.object({method:Q.literal(\"ui/request-display-mode\"),params:Q.object({mode:K.describe(\"The display mode being requested.\")})}),T=Q.object({mode:K.describe(\"The display mode that was actually set. May differ from requested if not supported.\")}).passthrough(),v=Q.union([Q.literal(\"model\"),Q.literal(\"app\")]).describe(\"Tool visibility scope - who can access the tool.\"),EQ=Q.object({resourceUri:Q.string().optional(),visibility:Q.array(v).optional().describe(`Who can access this tool. Default: [\"model\", \"app\"]\n- \"model\": Tool visible to and callable by the agent\n- \"app\": Tool callable by the app from this server only`)}),pQ=Q.object({mimeTypes:Q.array(Q.string()).optional().describe('Array of supported MIME types for UI resources.\\nMust include `\"text/html;profile=mcp-app\"` for MCP Apps support.')}),_Q=Q.object({method:Q.literal(\"ui/download-file\"),params:Q.object({contents:Q.array(Q.union([YQ,$Q])).describe(\"Resource contents to download — embedded (inline data) or linked (host fetches). Uses standard MCP resource types.\")})}),OQ=Q.object({method:Q.literal(\"ui/message\"),params:Q.object({role:Q.literal(\"user\").describe('Message role, currently only \"user\" is supported.'),content:Q.array(C).describe(\"Message content blocks (text, image, etc.).\")})}),IQ=Q.object({method:Q.literal(\"ui/notifications/sandbox-resource-ready\"),params:Q.object({html:Q.string().describe(\"HTML content to load into the inner iframe.\"),sandbox:Q.string().optional().describe(\"Optional override for the inner iframe's sandbox attribute.\"),csp:W.optional().describe(\"CSP configuration from resource metadata.\"),permissions:j.optional().describe(\"Sandbox permissions from resource metadata.\")})}),R=Q.object({method:Q.literal(\"ui/notifications/tool-result\"),params:XQ.describe(\"Standard MCP tool execution result.\")}),U=Q.object({toolInfo:Q.object({id:ZQ.optional().describe(\"JSON-RPC id of the tools/call request.\"),tool:DQ.describe(\"Tool definition including name, inputSchema, etc.\")}).optional().describe(\"Metadata of the tool call that instantiated this App.\"),theme:k.optional().describe(\"Current color theme preference.\"),styles:x.optional().describe(\"Style configuration for theming the app.\"),displayMode:K.optional().describe(\"How the UI is currently displayed.\"),availableDisplayModes:Q.array(K).optional().describe(\"Display modes the host supports.\"),containerDimensions:Q.union([Q.object({height:Q.number().describe(\"Fixed container height in pixels.\")}),Q.object({maxHeight:Q.union([Q.number(),Q.undefined()]).optional().describe(\"Maximum container height in pixels.\")})]).and(Q.union([Q.object({width:Q.number().describe(\"Fixed container width in pixels.\")}),Q.object({maxWidth:Q.union([Q.number(),Q.undefined()]).optional().describe(\"Maximum container width in pixels.\")})])).optional().describe(`Container dimensions. Represents the dimensions of the iframe or other\ncontainer holding the app. Specify either width or maxWidth, and either height or maxHeight.`),locale:Q.string().optional().describe(\"User's language and region preference in BCP 47 format.\"),timeZone:Q.string().optional().describe(\"User's timezone in IANA format.\"),userAgent:Q.string().optional().describe(\"Host application identifier.\"),platform:Q.union([Q.literal(\"web\"),Q.literal(\"desktop\"),Q.literal(\"mobile\")]).optional().describe(\"Platform type for responsive design decisions.\"),deviceCapabilities:Q.object({touch:Q.boolean().optional().describe(\"Whether the device supports touch input.\"),hover:Q.boolean().optional().describe(\"Whether the device supports hover interactions.\")}).optional().describe(\"Device input capabilities.\"),safeAreaInsets:Q.object({top:Q.number().describe(\"Top safe area inset in pixels.\"),right:Q.number().describe(\"Right safe area inset in pixels.\"),bottom:Q.number().describe(\"Bottom safe area inset in pixels.\"),left:Q.number().describe(\"Left safe area inset in pixels.\")}).optional().describe(\"Mobile safe area boundaries in pixels.\")}).passthrough(),H=Q.object({method:Q.literal(\"ui/notifications/host-context-changed\"),params:U.describe(\"Partial context update containing only changed fields.\")}),wQ=Q.object({method:Q.literal(\"ui/update-model-context\"),params:Q.object({content:Q.array(C).optional().describe(\"Context content blocks (text, image, etc.).\"),structuredContent:Q.record(Q.string(),Q.unknown().describe(\"Structured content for machine-readable context data.\")).optional().describe(\"Structured content for machine-readable context data.\")})}),AQ=Q.object({method:Q.literal(\"ui/initialize\"),params:Q.object({appInfo:b.describe(\"App identification (name and version).\"),appCapabilities:y.describe(\"Features and capabilities this app provides.\"),protocolVersion:Q.string().describe(\"Protocol version this app supports.\")})}),M=Q.object({protocolVersion:Q.string().describe('Negotiated protocol version string (e.g., \"2025-11-21\").'),hostInfo:b.describe(\"Host application identification and version.\"),hostCapabilities:S.describe(\"Features and capabilities provided by the host.\"),hostContext:U.describe(\"Rich context about the host environment.\")}).passthrough();function FQ(){let X=document.documentElement.getAttribute(\"data-theme\");if(X===\"dark\"||X===\"light\")return X;return document.documentElement.classList.contains(\"dark\")?\"dark\":\"light\"}function PQ(X){let Y=document.documentElement;Y.setAttribute(\"data-theme\",X),Y.style.colorScheme=X}function qQ(X,Y=document.documentElement){for(let[Z,$]of Object.entries(X))if($!==void 0)Y.style.setProperty(Z,$)}function TQ(X){if(document.getElementById(\"__mcp-host-fonts\"))return;let Z=document.createElement(\"style\");Z.id=\"__mcp-host-fonts\",Z.textContent=X,document.head.appendChild(Z)}var MX=\"ui/resourceUri\",CX=\"text/html;profile=mcp-app\";class gQ extends z{_appInfo;_capabilities;options;_hostCapabilities;_hostInfo;_hostContext;eventSchemas={toolinput:A,toolinputpartial:F,toolresult:R,toolcancelled:P,hostcontextchanged:H};onEventDispatch(X,Y){if(X===\"hostcontextchanged\")this._hostContext={...this._hostContext,...Y}}constructor(X,Y={},Z={autoResize:!0}){super(Z);this._appInfo=X;this._capabilities=Y;this.options=Z;this.setRequestHandler(bQ,($)=>{return console.log(\"Received ping:\",$.params),{}}),this.setEventHandler(\"hostcontextchanged\",void 0)}getHostCapabilities(){return this._hostCapabilities}getHostVersion(){return this._hostInfo}getHostContext(){return this._hostContext}get ontoolinput(){return this.getEventHandler(\"toolinput\")}set ontoolinput(X){this.setEventHandler(\"toolinput\",X)}get ontoolinputpartial(){return this.getEventHandler(\"toolinputpartial\")}set ontoolinputpartial(X){this.setEventHandler(\"toolinputpartial\",X)}get ontoolresult(){return this.getEventHandler(\"toolresult\")}set ontoolresult(X){this.setEventHandler(\"toolresult\",X)}get ontoolcancelled(){return this.getEventHandler(\"toolcancelled\")}set ontoolcancelled(X){this.setEventHandler(\"toolcancelled\",X)}get onhostcontextchanged(){return this.getEventHandler(\"hostcontextchanged\")}set onhostcontextchanged(X){this.setEventHandler(\"hostcontextchanged\",X)}_onteardown;get onteardown(){return this._onteardown}set onteardown(X){this.warnIfRequestHandlerReplaced(\"onteardown\",this._onteardown,X),this._onteardown=X,this.replaceRequestHandler(q,(Y,Z)=>{if(!this._onteardown)throw Error(\"No onteardown handler set\");return this._onteardown(Y.params,Z)})}_oncalltool;get oncalltool(){return this._oncalltool}set oncalltool(X){this.warnIfRequestHandlerReplaced(\"oncalltool\",this._oncalltool,X),this._oncalltool=X,this.replaceRequestHandler(RQ,(Y,Z)=>{if(!this._oncalltool)throw Error(\"No oncalltool handler set\");return this._oncalltool(Y.params,Z)})}_onlisttools;get onlisttools(){return this._onlisttools}set onlisttools(X){this.warnIfRequestHandlerReplaced(\"onlisttools\",this._onlisttools,X),this._onlisttools=X,this.replaceRequestHandler(CQ,(Y,Z)=>{if(!this._onlisttools)throw Error(\"No onlisttools handler set\");return this._onlisttools(Y.params,Z)})}assertCapabilityForMethod(X){}assertRequestHandlerCapability(X){switch(X){case\"tools/call\":case\"tools/list\":if(!this._capabilities.tools)throw Error(`Client does not support tool capability (required for ${X})`);return;case\"ping\":case\"ui/resource-teardown\":return;default:throw Error(`No handler for method ${X} registered`)}}assertNotificationCapability(X){}assertTaskCapability(X){throw Error(\"Tasks are not supported in MCP Apps\")}assertTaskHandlerCapability(X){throw Error(\"Task handlers are not supported in MCP Apps\")}async callServerTool(X,Y){if(typeof X===\"string\")throw Error(`callServerTool() expects an object as its first argument, but received a string (\"${X}\"). Did you mean: callServerTool({ name: \"${X}\", arguments: { ... } })?`);return await this.request({method:\"tools/call\",params:X},UQ,Y)}async readServerResource(X,Y){return await this.request({method:\"resources/read\",params:X},kQ,Y)}async listServerResources(X,Y){return await this.request({method:\"resources/list\",params:X},MQ,Y)}sendMessage(X,Y){return this.request({method:\"ui/message\",params:X},w,Y)}sendLog(X){return this.notification({method:\"notifications/message\",params:X})}updateModelContext(X,Y){return this.request({method:\"ui/update-model-context\",params:X},HQ,Y)}openLink(X,Y){return this.request({method:\"ui/open-link\",params:X},O,Y)}sendOpenLink=this.openLink;downloadFile(X,Y){return this.request({method:\"ui/download-file\",params:X},I,Y)}requestTeardown(X={}){return this.notification({method:\"ui/notifications/request-teardown\",params:X})}requestDisplayMode(X,Y){return this.request({method:\"ui/request-display-mode\",params:X},T,Y)}sendSizeChanged(X){return this.notification({method:\"ui/notifications/size-changed\",params:X})}setupSizeChangedNotifications(){let X=!1,Y=0,Z=0,$=()=>{if(X)return;X=!0,requestAnimationFrame(()=>{X=!1;let J=document.documentElement,G=J.style.height;J.style.height=\"max-content\";let V=Math.ceil(J.getBoundingClientRect().height);J.style.height=G;let L=Math.ceil(window.innerWidth);if(L!==Y||V!==Z)Y=L,Z=V,this.sendSizeChanged({width:L,height:V})})};$();let D=new ResizeObserver($);return D.observe(document.documentElement),D.observe(document.body),()=>D.disconnect()}async connect(X=new N(window.parent,window.parent),Y){if(this.transport)throw Error(\"App is already connected. Call close() before connecting again.\");await super.connect(X);try{let Z=await this.request({method:\"ui/initialize\",params:{appCapabilities:this._capabilities,appInfo:this._appInfo,protocolVersion:B}},M,Y);if(Z===void 0)throw Error(`Server sent invalid initialize result: ${Z}`);if(this._hostCapabilities=Z.hostCapabilities,this._hostInfo=Z.hostInfo,this._hostContext=Z.hostContext,await this.notification({method:\"ui/notifications/initialized\"}),this.options?.autoResize)this.setupSizeChangedNotifications()}catch(Z){throw this.close(),Z}}}export{FQ as getDocumentTheme,qQ as applyHostStyleVariables,TQ as applyHostFonts,PQ as applyDocumentTheme,c as TOOL_RESULT_METHOD,E as TOOL_INPUT_PARTIAL_METHOD,r as TOOL_INPUT_METHOD,l as TOOL_CANCELLED_METHOD,p as SIZE_CHANGED_METHOD,m as SANDBOX_RESOURCE_READY_METHOD,i as SANDBOX_PROXY_READY_METHOD,MX as RESOURCE_URI_META_KEY,o as RESOURCE_TEARDOWN_METHOD,CX as RESOURCE_MIME_TYPE,a as REQUEST_TEARDOWN_METHOD,e as REQUEST_DISPLAY_MODE_METHOD,z as ProtocolWithEvents,N as PostMessageTransport,u as OPEN_LINK_METHOD,wQ as McpUiUpdateModelContextRequestSchema,v as McpUiToolVisibilitySchema,R as McpUiToolResultNotificationSchema,EQ as McpUiToolMetaSchema,F as McpUiToolInputPartialNotificationSchema,A as McpUiToolInputNotificationSchema,P as McpUiToolCancelledNotificationSchema,k as McpUiThemeSchema,_ as McpUiSupportedContentBlockModalitiesSchema,NQ as McpUiSizeChangedNotificationSchema,IQ as McpUiSandboxResourceReadyNotificationSchema,VQ as McpUiSandboxProxyReadyNotificationSchema,WQ as McpUiResourceTeardownResultSchema,q as McpUiResourceTeardownRequestSchema,j as McpUiResourcePermissionsSchema,zQ as McpUiResourceMetaSchema,W as McpUiResourceCspSchema,jQ as McpUiRequestTeardownNotificationSchema,T as McpUiRequestDisplayModeResultSchema,BQ as McpUiRequestDisplayModeRequestSchema,O as McpUiOpenLinkResultSchema,GQ as McpUiOpenLinkRequestSchema,w as McpUiMessageResultSchema,OQ as McpUiMessageRequestSchema,LQ as McpUiInitializedNotificationSchema,M as McpUiInitializeResultSchema,AQ as McpUiInitializeRequestSchema,x as McpUiHostStylesSchema,g as McpUiHostCssSchema,U as McpUiHostContextSchema,H as McpUiHostContextChangedNotificationSchema,S as McpUiHostCapabilitiesSchema,I as McpUiDownloadFileResultSchema,_Q as McpUiDownloadFileRequestSchema,K as McpUiDisplayModeSchema,y as McpUiAppCapabilitiesSchema,h as MESSAGE_METHOD,B as LATEST_PROTOCOL_VERSION,s as INITIALIZE_METHOD,t as INITIALIZED_METHOD,n as HOST_CONTEXT_CHANGED_METHOD,d as DOWNLOAD_FILE_METHOD,gQ as App};\n","/**\n * React Context for sharing the MCP App instance across the component tree.\n *\n * AppProvider handles connecting to the MCP Apps host and provides the App\n * instance via context. All sunpeak hooks read from this context internally,\n * so consumers never need to pass `app` as a parameter.\n *\n * The provider preserves the App instance across React Fast Refresh (HMR)\n * by storing it at module scope, matching the previous useApp() behavior.\n *\n * Connection resilience: if the initial PostMessage handshake doesn't complete\n * within a timeout, the provider automatically retries with exponential backoff.\n * This handles race conditions where the host iframe bridge isn't ready when the\n * app first mounts (common on first load in ChatGPT and Claude).\n */\nimport { createContext, useState, useEffect, type ReactNode } from 'react';\nimport { App, PostMessageTransport } from '@modelcontextprotocol/ext-apps';\n\nexport interface AppProviderProps {\n appInfo: { name: string; version: string };\n capabilities?: Record<string, unknown>;\n onAppCreated?: (app: App) => void;\n children: ReactNode;\n}\n\nexport interface AppState {\n app: App | null;\n isConnected: boolean;\n error: Error | null;\n}\n\nconst defaultState: AppState = { app: null, isConnected: false, error: null };\n\nexport const AppContext = createContext<AppState>(defaultState);\n\n// Module-level App persistence.\n// During React Fast Refresh the component file is hot-swapped but this module\n// is NOT re-evaluated, so these variables survive across HMR cycles.\n// On a full page reload they reset to null, triggering a fresh connection.\nlet _app: App | null = null;\nlet _connecting: Promise<App> | null = null;\n\n/** Timeout for a single connection attempt (ms). */\nconst CONNECT_TIMEOUT_MS = 5_000;\n/** Maximum number of retry attempts before giving up. */\nconst MAX_RETRIES = 3;\n/** Base delay for exponential backoff between retries (ms). */\nconst RETRY_BASE_DELAY_MS = 500;\n\n/**\n * Race a promise against a timeout. Rejects with a TimeoutError if the\n * promise doesn't settle within `ms` milliseconds.\n */\nfunction withTimeout<T>(promise: Promise<T>, ms: number): Promise<T> {\n return new Promise<T>((resolve, reject) => {\n const timer = setTimeout(() => reject(new Error('Connection timed out')), ms);\n promise.then(\n (v) => {\n clearTimeout(timer);\n resolve(v);\n },\n (e) => {\n clearTimeout(timer);\n reject(e);\n }\n );\n });\n}\n\n/**\n * Attempt to connect to the host with timeout and retries.\n * Each attempt creates a fresh transport + App instance so stale PostMessage\n * listeners from a previous failed attempt don't interfere.\n */\nasync function connectWithRetry(\n appInfo: { name: string; version: string },\n capabilities: Record<string, unknown>,\n onAppCreated?: (app: App) => void\n): Promise<App> {\n let lastError: Error | null = null;\n\n for (let attempt = 0; attempt <= MAX_RETRIES; attempt++) {\n if (attempt > 0) {\n const delay = RETRY_BASE_DELAY_MS * 2 ** (attempt - 1);\n await new Promise((r) => setTimeout(r, delay));\n }\n\n let transport: PostMessageTransport | null = null;\n try {\n transport = new PostMessageTransport(window.parent, window.parent);\n const newApp = new App(appInfo, capabilities);\n onAppCreated?.(newApp);\n await withTimeout(newApp.connect(transport), CONNECT_TIMEOUT_MS);\n return newApp;\n } catch (err) {\n // Clean up the transport's PostMessage listener so it doesn't linger.\n if (transport) {\n try {\n await transport.close();\n } catch {\n /* ignore close errors */\n }\n }\n lastError = err instanceof Error ? err : new Error('Connection failed');\n if (attempt < MAX_RETRIES) {\n console.warn(\n `[sunpeak] Connection attempt ${attempt + 1}/${MAX_RETRIES + 1} failed, retrying...`,\n lastError.message\n );\n }\n }\n }\n\n console.error(\n '[sunpeak] All connection attempts failed. Try refreshing the page or opening a new chat.\\n' +\n 'Troubleshooting: https://sunpeak.ai/docs/app-framework/guides/troubleshooting'\n );\n throw lastError ?? new Error('Failed to connect');\n}\n\nexport function AppProvider({ appInfo, capabilities, onAppCreated, children }: AppProviderProps) {\n const [state, setState] = useState<AppState>(() =>\n _app ? { app: _app, isConnected: true, error: null } : defaultState\n );\n\n useEffect(() => {\n let cancelled = false;\n\n // Already connected (HMR re-run or StrictMode double-mount) — reuse.\n if (_app) {\n setState({ app: _app, isConnected: true, error: null });\n return () => {\n cancelled = true;\n };\n }\n\n // Connection already in flight (StrictMode double-mount) — wait for it.\n if (!_connecting) {\n _connecting = connectWithRetry(appInfo, capabilities ?? {}, onAppCreated);\n }\n\n _connecting.then(\n (connectedApp) => {\n _app = connectedApp;\n if (!cancelled) {\n setState({ app: connectedApp, isConnected: true, error: null });\n }\n },\n (err) => {\n _connecting = null;\n if (!cancelled) {\n setState({\n app: null,\n isConnected: false,\n error: err instanceof Error ? err : new Error('Failed to connect'),\n });\n }\n }\n );\n\n return () => {\n cancelled = true;\n };\n // eslint-disable-next-line react-hooks/exhaustive-deps -- connect once, persist across HMR\n }, []);\n\n return <AppContext.Provider value={state}>{children}</AppContext.Provider>;\n}\n","import { useContext } from 'react';\nimport { AppContext } from './app-context';\nimport type { App } from '@modelcontextprotocol/ext-apps';\n\n/**\n * Access the MCP App instance from context.\n *\n * Returns the connected App instance, or `null` while the connection\n * is being established. Must be used inside an `<AppProvider>`.\n *\n * Most hooks read from context internally, so you only need `useApp()`\n * for direct SDK method calls like `app.requestDisplayMode()`.\n *\n * @example\n * ```tsx\n * import { useApp } from 'sunpeak';\n *\n * function MyComponent() {\n * const app = useApp();\n * const handleFullscreen = () => app?.requestDisplayMode({ mode: 'fullscreen' });\n * return <button onClick={handleFullscreen}>Fullscreen</button>;\n * }\n * ```\n */\nexport function useApp(): App | null {\n return useContext(AppContext).app;\n}\n"],"x_google_ignoreList":[0],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "sunpeak",
3
- "version": "0.19.2",
3
+ "version": "0.19.10",
4
4
  "description": "Inspector, testing framework, and app framework for MCP Apps.",
5
5
  "type": "module",
6
6
  "main": "./dist/index.cjs",
@@ -96,6 +96,12 @@
96
96
  "default": "./bin/lib/inspect/inspect-config.mjs"
97
97
  }
98
98
  },
99
+ "./eval": {
100
+ "import": {
101
+ "types": "./bin/lib/eval/eval-types.d.mts",
102
+ "default": "./bin/lib/eval/eval-runner.mjs"
103
+ }
104
+ },
99
105
  "./package.json": "./package.json"
100
106
  },
101
107
  "bin": {
@@ -131,11 +137,29 @@
131
137
  "license": "MIT",
132
138
  "peerDependencies": {
133
139
  "react": "^18.0.0 || ^19.0.0",
134
- "react-dom": "^18.0.0 || ^19.0.0"
140
+ "react-dom": "^18.0.0 || ^19.0.0",
141
+ "ai": "^4.0.0",
142
+ "@ai-sdk/openai": "^1.0.0",
143
+ "@ai-sdk/anthropic": "^1.0.0",
144
+ "@ai-sdk/google": "^1.0.0"
145
+ },
146
+ "peerDependenciesMeta": {
147
+ "ai": {
148
+ "optional": true
149
+ },
150
+ "@ai-sdk/openai": {
151
+ "optional": true
152
+ },
153
+ "@ai-sdk/anthropic": {
154
+ "optional": true
155
+ },
156
+ "@ai-sdk/google": {
157
+ "optional": true
158
+ }
135
159
  },
136
160
  "dependencies": {
137
161
  "@clack/prompts": "^1.2.0",
138
- "@modelcontextprotocol/ext-apps": "^1.3.2",
162
+ "@modelcontextprotocol/ext-apps": "^1.5.0",
139
163
  "@modelcontextprotocol/sdk": "^1.29.0",
140
164
  "@vitejs/plugin-react": "^6.0.1",
141
165
  "clsx": "^2.1.1",
@@ -2,7 +2,7 @@
2
2
 
3
3
  An MCP App built with [sunpeak](https://github.com/Sunpeak-AI/sunpeak).
4
4
 
5
- For an initial overview of your new app and a detailed API reference, refer to the [documentation](https://sunpeak.ai/docs/template/project-scaffold).
5
+ For an initial overview of your new app and a detailed API reference, refer to the [documentation](https://sunpeak.ai/docs/app-framework/project-scaffold).
6
6
 
7
7
  ## Quickstart
8
8
 
@@ -23,6 +23,7 @@ sunpeak test --e2e # Run e2e tests only (Playwright).
23
23
  sunpeak test --visual # Run e2e tests with visual regression.
24
24
  sunpeak test --visual --update # Update visual regression baselines.
25
25
  sunpeak test --live # Run live tests against real ChatGPT.
26
+ sunpeak test --eval # Run evals against multiple LLM models.
26
27
  ```
27
28
 
28
29
  **Development and production:**
@@ -34,7 +35,16 @@ sunpeak start # Start the production MCP server.
34
35
  sunpeak upgrade # Upgrade sunpeak to latest version.
35
36
  ```
36
37
 
37
- E2e tests use the `mcp` fixture from `sunpeak/test` to call tools and assert against rendered UI across ChatGPT and Claude hosts. Unit tests use Vitest with happy-dom. You can add additional tooling (linting, formatting, type-checking) as needed for your project.
38
+ E2e tests use the `mcp` fixture from `sunpeak/test` to call tools and assert against rendered UI across ChatGPT and Claude hosts. Unit tests use Vitest with happy-dom.
39
+
40
+ **Evals** test whether LLMs (GPT-4o, Claude, Gemini, etc.) call your tools correctly. To set up evals:
41
+
42
+ 1. Install the AI SDK and provider packages: `pnpm add ai @ai-sdk/openai`
43
+ 2. Copy `tests/evals/.env.example` to `tests/evals/.env` and add your API keys
44
+ 3. Uncomment models in `tests/evals/eval.config.ts`
45
+ 4. Run: `sunpeak test --eval`
46
+
47
+ The dev server starts automatically for evals. Each case runs multiple times per model to measure reliability. See the [Evals documentation](https://sunpeak.ai/docs/testing/evals) for details.
38
48
 
39
49
  ## Project Structure
40
50
 
@@ -123,7 +133,7 @@ sunpeak start --host 127.0.0.1 # Bind to localhost only
123
133
  sunpeak start --json-logs # Structured JSON logging for production
124
134
  ```
125
135
 
126
- The server includes a `/health` endpoint for load balancer probes and monitoring. See the [Deployment Guide](https://sunpeak.ai/docs/guides/deployment) for production operations details (reverse proxy, process management, Docker).
136
+ The server includes a `/health` endpoint for load balancer probes and monitoring. See the [Deployment Guide](https://sunpeak.ai/docs/app-framework/guides/deployment) for production operations details (reverse proxy, process management, Docker).
127
137
 
128
138
  ## Add a new UI (Resource)
129
139
 
@@ -140,12 +150,12 @@ Only the resource file (`.tsx`) is required to generate a production build and s
140
150
 
141
151
  Then create a tool file in `src/tools/` and simulation file(s) in `tests/simulations/` to preview your resource in `sunpeak dev`.
142
152
 
143
- ## Coding Agent Skill
153
+ ## Coding Agent Skills
144
154
 
145
- Install the `create-sunpeak-app` skill to give your coding agent built-in knowledge of sunpeak patterns, hooks, simulation files, and testing conventions:
155
+ Install the sunpeak skills to give your coding agent built-in knowledge of sunpeak patterns, hooks, and testing:
146
156
 
147
157
  ```bash
148
- npx skills add Sunpeak-AI/sunpeak@create-sunpeak-app
158
+ npx skills add Sunpeak-AI/sunpeak@create-sunpeak-app Sunpeak-AI/sunpeak@test-mcp-server
149
159
  ```
150
160
 
151
161
  ## Troubleshooting
@@ -158,7 +168,7 @@ If your app doesn't render in ChatGPT or Claude:
158
168
  4. **Hard refresh** the host page (`Cmd+Shift+R` / `Ctrl+Shift+R`)
159
169
  5. **Open a new chat** in the host (cached iframes persist per-conversation)
160
170
 
161
- Full guide: [sunpeak.ai/docs/guides/troubleshooting](https://sunpeak.ai/docs/guides/troubleshooting)
171
+ Full guide: [sunpeak.ai/docs/app-framework/guides/troubleshooting](https://sunpeak.ai/docs/app-framework/guides/troubleshooting)
162
172
 
163
173
  ## Resources
164
174
 
@@ -11,6 +11,8 @@ build
11
11
  # Testing
12
12
  coverage
13
13
  tests/live/.auth/
14
+ tests/evals/.env
15
+ .eval-vitest.config.ts
14
16
 
15
17
  # Environment variables
16
18
  .env