skybridge 0.0.0-dev.5d2997d → 0.0.0-dev.5d51265

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 (298) hide show
  1. package/LICENSE +21 -674
  2. package/bin/run.js +9 -0
  3. package/dist/cli/header.d.ts +4 -0
  4. package/dist/cli/header.js +6 -0
  5. package/dist/cli/header.js.map +1 -0
  6. package/dist/cli/run-command.d.ts +2 -0
  7. package/dist/cli/run-command.js +43 -0
  8. package/dist/cli/run-command.js.map +1 -0
  9. package/dist/cli/use-execute-steps.d.ts +10 -0
  10. package/dist/cli/use-execute-steps.js +31 -0
  11. package/dist/cli/use-execute-steps.js.map +1 -0
  12. package/dist/commands/build.d.ts +9 -0
  13. package/dist/commands/build.js +44 -0
  14. package/dist/commands/build.js.map +1 -0
  15. package/dist/commands/dev.d.ts +7 -0
  16. package/dist/commands/dev.js +20 -0
  17. package/dist/commands/dev.js.map +1 -0
  18. package/dist/commands/start.d.ts +7 -0
  19. package/dist/commands/start.js +33 -0
  20. package/dist/commands/start.js.map +1 -0
  21. package/dist/commands/telemetry/disable.d.ts +5 -0
  22. package/dist/commands/telemetry/disable.js +14 -0
  23. package/dist/commands/telemetry/disable.js.map +1 -0
  24. package/dist/commands/telemetry/enable.d.ts +5 -0
  25. package/dist/commands/telemetry/enable.js +14 -0
  26. package/dist/commands/telemetry/enable.js.map +1 -0
  27. package/dist/commands/telemetry/status.d.ts +5 -0
  28. package/dist/commands/telemetry/status.js +14 -0
  29. package/dist/commands/telemetry/status.js.map +1 -0
  30. package/dist/hooks/telemetry.d.ts +7 -0
  31. package/dist/hooks/telemetry.js +124 -0
  32. package/dist/hooks/telemetry.js.map +1 -0
  33. package/dist/server/index.d.ts +4 -0
  34. package/dist/server/index.js.map +1 -0
  35. package/dist/server/inferUtilityTypes.d.ts +64 -0
  36. package/dist/server/inferUtilityTypes.js +2 -0
  37. package/dist/server/inferUtilityTypes.js.map +1 -0
  38. package/dist/server/server.d.ts +99 -0
  39. package/dist/server/server.js +138 -0
  40. package/dist/server/server.js.map +1 -0
  41. package/dist/{src/server → server}/templateHelper.d.ts +3 -0
  42. package/dist/{src/server → server}/templateHelper.js +5 -4
  43. package/dist/server/templateHelper.js.map +1 -0
  44. package/dist/server/templates/development.hbs +66 -0
  45. package/dist/{src/server → server}/templates/production.hbs +1 -0
  46. package/dist/{src/server → server}/widgetsDevServer.d.ts +2 -2
  47. package/dist/{src/server → server}/widgetsDevServer.js +13 -5
  48. package/dist/server/widgetsDevServer.js.map +1 -0
  49. package/dist/test/utils.d.ts +135 -0
  50. package/dist/test/utils.js +242 -0
  51. package/dist/test/utils.js.map +1 -0
  52. package/dist/test/widget.test.js +255 -0
  53. package/dist/test/widget.test.js.map +1 -0
  54. package/dist/web/bridges/apps-sdk/adaptor.d.ts +14 -0
  55. package/dist/web/bridges/apps-sdk/adaptor.js +39 -0
  56. package/dist/web/bridges/apps-sdk/adaptor.js.map +1 -0
  57. package/dist/web/bridges/apps-sdk/bridge.d.ts +10 -0
  58. package/dist/web/bridges/apps-sdk/bridge.js +46 -0
  59. package/dist/web/bridges/apps-sdk/bridge.js.map +1 -0
  60. package/dist/web/bridges/apps-sdk/index.d.ts +5 -0
  61. package/dist/web/bridges/apps-sdk/index.js +5 -0
  62. package/dist/web/bridges/apps-sdk/index.js.map +1 -0
  63. package/dist/{src/web → web/bridges/apps-sdk}/types.d.ts +53 -41
  64. package/dist/web/bridges/apps-sdk/types.js.map +1 -0
  65. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.d.ts +2 -0
  66. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.js +7 -0
  67. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.js.map +1 -0
  68. package/dist/web/bridges/get-adaptor.d.ts +2 -0
  69. package/dist/web/bridges/get-adaptor.js +8 -0
  70. package/dist/web/bridges/get-adaptor.js.map +1 -0
  71. package/dist/web/bridges/index.d.ts +5 -0
  72. package/dist/web/bridges/index.js +6 -0
  73. package/dist/web/bridges/index.js.map +1 -0
  74. package/dist/web/bridges/mcp-app/adaptor.d.ts +19 -0
  75. package/dist/web/bridges/mcp-app/adaptor.js +145 -0
  76. package/dist/web/bridges/mcp-app/adaptor.js.map +1 -0
  77. package/dist/web/bridges/mcp-app/bridge.d.ts +43 -0
  78. package/dist/web/bridges/mcp-app/bridge.js +255 -0
  79. package/dist/web/bridges/mcp-app/bridge.js.map +1 -0
  80. package/dist/web/bridges/mcp-app/index.d.ts +4 -0
  81. package/dist/web/bridges/mcp-app/index.js +4 -0
  82. package/dist/web/bridges/mcp-app/index.js.map +1 -0
  83. package/dist/web/bridges/mcp-app/types.d.ts +8 -0
  84. package/dist/web/bridges/mcp-app/types.js +2 -0
  85. package/dist/web/bridges/mcp-app/types.js.map +1 -0
  86. package/dist/web/bridges/mcp-app/use-mcp-app-context.d.ts +5 -0
  87. package/dist/web/bridges/mcp-app/use-mcp-app-context.js +7 -0
  88. package/dist/web/bridges/mcp-app/use-mcp-app-context.js.map +1 -0
  89. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js +66 -0
  90. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js.map +1 -0
  91. package/dist/web/bridges/types.d.ts +74 -0
  92. package/dist/web/bridges/types.js +2 -0
  93. package/dist/web/bridges/types.js.map +1 -0
  94. package/dist/web/bridges/use-host-context.d.ts +2 -0
  95. package/dist/web/bridges/use-host-context.js +8 -0
  96. package/dist/web/bridges/use-host-context.js.map +1 -0
  97. package/dist/web/create-store.d.ts +3 -0
  98. package/dist/web/create-store.js +24 -0
  99. package/dist/web/create-store.js.map +1 -0
  100. package/dist/web/create-store.test.js +126 -0
  101. package/dist/web/create-store.test.js.map +1 -0
  102. package/dist/web/data-llm.d.ts +14 -0
  103. package/dist/web/data-llm.js +72 -0
  104. package/dist/web/data-llm.js.map +1 -0
  105. package/dist/web/data-llm.test.js +139 -0
  106. package/dist/web/data-llm.test.js.map +1 -0
  107. package/dist/web/generate-helpers.d.ts +116 -0
  108. package/dist/web/generate-helpers.js +111 -0
  109. package/dist/web/generate-helpers.js.map +1 -0
  110. package/dist/web/generate-helpers.test-d.js +209 -0
  111. package/dist/web/generate-helpers.test-d.js.map +1 -0
  112. package/dist/web/generate-helpers.test.js +17 -0
  113. package/dist/web/generate-helpers.test.js.map +1 -0
  114. package/dist/web/helpers/state.d.ts +7 -0
  115. package/dist/web/helpers/state.js +45 -0
  116. package/dist/web/helpers/state.js.map +1 -0
  117. package/dist/web/helpers/state.test.js +53 -0
  118. package/dist/web/helpers/state.test.js.map +1 -0
  119. package/dist/web/hooks/index.d.ts +10 -0
  120. package/dist/web/hooks/index.js +11 -0
  121. package/dist/web/hooks/index.js.map +1 -0
  122. package/dist/web/hooks/test/utils.d.ts +16 -0
  123. package/dist/web/hooks/test/utils.js +60 -0
  124. package/dist/web/hooks/test/utils.js.map +1 -0
  125. package/dist/web/hooks/use-call-tool.d.ts +101 -0
  126. package/dist/web/hooks/use-call-tool.js +68 -0
  127. package/dist/web/hooks/use-call-tool.js.map +1 -0
  128. package/dist/web/hooks/use-call-tool.test-d.js +104 -0
  129. package/dist/web/hooks/use-call-tool.test-d.js.map +1 -0
  130. package/dist/{src/web → web}/hooks/use-call-tool.test.js +40 -13
  131. package/dist/web/hooks/use-call-tool.test.js.map +1 -0
  132. package/dist/{src/web → web}/hooks/use-display-mode.d.ts +1 -1
  133. package/dist/web/hooks/use-display-mode.js +9 -0
  134. package/dist/web/hooks/use-display-mode.js.map +1 -0
  135. package/dist/{src/web → web}/hooks/use-display-mode.test.js +3 -2
  136. package/dist/web/hooks/use-display-mode.test.js.map +1 -0
  137. package/dist/web/hooks/use-files.d.ts +10 -0
  138. package/dist/web/hooks/use-files.js +7 -0
  139. package/dist/web/hooks/use-files.js.map +1 -0
  140. package/dist/web/hooks/use-files.test.d.ts +1 -0
  141. package/dist/web/hooks/use-files.test.js +29 -0
  142. package/dist/web/hooks/use-files.test.js.map +1 -0
  143. package/dist/web/hooks/use-layout.d.ts +22 -0
  144. package/dist/web/hooks/use-layout.js +23 -0
  145. package/dist/web/hooks/use-layout.js.map +1 -0
  146. package/dist/web/hooks/use-layout.test.d.ts +1 -0
  147. package/dist/web/hooks/use-layout.test.js +96 -0
  148. package/dist/web/hooks/use-layout.test.js.map +1 -0
  149. package/dist/web/hooks/use-open-external.js +8 -0
  150. package/dist/web/hooks/use-open-external.js.map +1 -0
  151. package/dist/web/hooks/use-open-external.test.d.ts +1 -0
  152. package/dist/web/hooks/use-open-external.test.js +50 -0
  153. package/dist/web/hooks/use-open-external.test.js.map +1 -0
  154. package/dist/web/hooks/use-request-modal.d.ts +9 -0
  155. package/dist/web/hooks/use-request-modal.js +14 -0
  156. package/dist/web/hooks/use-request-modal.js.map +1 -0
  157. package/dist/web/hooks/use-request-modal.test.d.ts +1 -0
  158. package/dist/web/hooks/use-request-modal.test.js +57 -0
  159. package/dist/web/hooks/use-request-modal.test.js.map +1 -0
  160. package/dist/web/hooks/use-send-follow-up-message.js +8 -0
  161. package/dist/web/hooks/use-send-follow-up-message.js.map +1 -0
  162. package/dist/web/hooks/use-tool-info.d.ts +36 -0
  163. package/dist/web/hooks/use-tool-info.js +26 -0
  164. package/dist/web/hooks/use-tool-info.js.map +1 -0
  165. package/dist/web/hooks/use-tool-info.test-d.d.ts +1 -0
  166. package/dist/web/hooks/use-tool-info.test-d.js +109 -0
  167. package/dist/web/hooks/use-tool-info.test-d.js.map +1 -0
  168. package/dist/web/hooks/use-tool-info.test.d.ts +1 -0
  169. package/dist/web/hooks/use-tool-info.test.js +130 -0
  170. package/dist/web/hooks/use-tool-info.test.js.map +1 -0
  171. package/dist/web/hooks/use-user.d.ts +18 -0
  172. package/dist/web/hooks/use-user.js +19 -0
  173. package/dist/web/hooks/use-user.js.map +1 -0
  174. package/dist/web/hooks/use-user.test.d.ts +1 -0
  175. package/dist/web/hooks/use-user.test.js +94 -0
  176. package/dist/web/hooks/use-user.test.js.map +1 -0
  177. package/dist/web/hooks/use-widget-state.js +32 -0
  178. package/dist/web/hooks/use-widget-state.js.map +1 -0
  179. package/dist/web/hooks/use-widget-state.test.d.ts +1 -0
  180. package/dist/{src/web → web}/hooks/use-widget-state.test.js +3 -2
  181. package/dist/web/hooks/use-widget-state.test.js.map +1 -0
  182. package/dist/web/index.d.ts +8 -0
  183. package/dist/web/index.js +9 -0
  184. package/dist/web/index.js.map +1 -0
  185. package/dist/{src/web → web}/mount-widget.js +5 -0
  186. package/dist/web/mount-widget.js.map +1 -0
  187. package/dist/web/plugin/data-llm.test.d.ts +1 -0
  188. package/dist/web/plugin/data-llm.test.js +81 -0
  189. package/dist/web/plugin/data-llm.test.js.map +1 -0
  190. package/dist/web/plugin/plugin.js +39 -0
  191. package/dist/web/plugin/plugin.js.map +1 -0
  192. package/dist/web/plugin/transform-data-llm.d.ts +12 -0
  193. package/dist/web/plugin/transform-data-llm.js +96 -0
  194. package/dist/web/plugin/transform-data-llm.js.map +1 -0
  195. package/dist/web/plugin/transform-data-llm.test.d.ts +1 -0
  196. package/dist/web/plugin/transform-data-llm.test.js +81 -0
  197. package/dist/web/plugin/transform-data-llm.test.js.map +1 -0
  198. package/dist/web/proxy.d.ts +1 -0
  199. package/dist/web/proxy.js +53 -0
  200. package/dist/web/proxy.js.map +1 -0
  201. package/dist/web/types.d.ts +16 -0
  202. package/dist/web/types.js +2 -0
  203. package/dist/web/types.js.map +1 -0
  204. package/package.json +59 -30
  205. package/README.md +0 -198
  206. package/dist/src/server/index.d.ts +0 -2
  207. package/dist/src/server/index.js.map +0 -1
  208. package/dist/src/server/server.d.ts +0 -13
  209. package/dist/src/server/server.js +0 -57
  210. package/dist/src/server/server.js.map +0 -1
  211. package/dist/src/server/templateHelper.js.map +0 -1
  212. package/dist/src/server/templates/development.hbs +0 -12
  213. package/dist/src/server/widgetsDevServer.js.map +0 -1
  214. package/dist/src/test/utils.d.ts +0 -28
  215. package/dist/src/test/utils.js +0 -43
  216. package/dist/src/test/utils.js.map +0 -1
  217. package/dist/src/test/widget.test.js +0 -90
  218. package/dist/src/test/widget.test.js.map +0 -1
  219. package/dist/src/web/hooks/index.d.ts +0 -13
  220. package/dist/src/web/hooks/index.js +0 -14
  221. package/dist/src/web/hooks/index.js.map +0 -1
  222. package/dist/src/web/hooks/use-call-tool.d.ts +0 -62
  223. package/dist/src/web/hooks/use-call-tool.js +0 -66
  224. package/dist/src/web/hooks/use-call-tool.js.map +0 -1
  225. package/dist/src/web/hooks/use-call-tool.test.js.map +0 -1
  226. package/dist/src/web/hooks/use-display-mode.js +0 -7
  227. package/dist/src/web/hooks/use-display-mode.js.map +0 -1
  228. package/dist/src/web/hooks/use-display-mode.test.js.map +0 -1
  229. package/dist/src/web/hooks/use-locale.d.ts +0 -1
  230. package/dist/src/web/hooks/use-locale.js +0 -5
  231. package/dist/src/web/hooks/use-locale.js.map +0 -1
  232. package/dist/src/web/hooks/use-locale.test.js +0 -21
  233. package/dist/src/web/hooks/use-locale.test.js.map +0 -1
  234. package/dist/src/web/hooks/use-open-external.js +0 -6
  235. package/dist/src/web/hooks/use-open-external.js.map +0 -1
  236. package/dist/src/web/hooks/use-open-external.test.js +0 -24
  237. package/dist/src/web/hooks/use-open-external.test.js.map +0 -1
  238. package/dist/src/web/hooks/use-openai-global.d.ts +0 -2
  239. package/dist/src/web/hooks/use-openai-global.js +0 -23
  240. package/dist/src/web/hooks/use-openai-global.js.map +0 -1
  241. package/dist/src/web/hooks/use-request-modal.d.ts +0 -5
  242. package/dist/src/web/hooks/use-request-modal.js +0 -9
  243. package/dist/src/web/hooks/use-request-modal.js.map +0 -1
  244. package/dist/src/web/hooks/use-request-modal.test.js +0 -24
  245. package/dist/src/web/hooks/use-request-modal.test.js.map +0 -1
  246. package/dist/src/web/hooks/use-send-follow-up-message.js +0 -11
  247. package/dist/src/web/hooks/use-send-follow-up-message.js.map +0 -1
  248. package/dist/src/web/hooks/use-theme.d.ts +0 -1
  249. package/dist/src/web/hooks/use-theme.js +0 -5
  250. package/dist/src/web/hooks/use-theme.js.map +0 -1
  251. package/dist/src/web/hooks/use-theme.test.js +0 -26
  252. package/dist/src/web/hooks/use-theme.test.js.map +0 -1
  253. package/dist/src/web/hooks/use-tool-info.d.ts +0 -23
  254. package/dist/src/web/hooks/use-tool-info.js +0 -22
  255. package/dist/src/web/hooks/use-tool-info.js.map +0 -1
  256. package/dist/src/web/hooks/use-tool-info.test.js +0 -133
  257. package/dist/src/web/hooks/use-tool-info.test.js.map +0 -1
  258. package/dist/src/web/hooks/use-tool-output.d.ts +0 -4
  259. package/dist/src/web/hooks/use-tool-output.js +0 -9
  260. package/dist/src/web/hooks/use-tool-output.js.map +0 -1
  261. package/dist/src/web/hooks/use-tool-response-metadata.d.ts +0 -4
  262. package/dist/src/web/hooks/use-tool-response-metadata.js +0 -8
  263. package/dist/src/web/hooks/use-tool-response-metadata.js.map +0 -1
  264. package/dist/src/web/hooks/use-user-agent.d.ts +0 -1
  265. package/dist/src/web/hooks/use-user-agent.js +0 -5
  266. package/dist/src/web/hooks/use-user-agent.js.map +0 -1
  267. package/dist/src/web/hooks/use-user-agent.test.js +0 -31
  268. package/dist/src/web/hooks/use-user-agent.test.js.map +0 -1
  269. package/dist/src/web/hooks/use-widget-state.js +0 -30
  270. package/dist/src/web/hooks/use-widget-state.js.map +0 -1
  271. package/dist/src/web/hooks/use-widget-state.test.js.map +0 -1
  272. package/dist/src/web/index.d.ts +0 -4
  273. package/dist/src/web/index.js +0 -5
  274. package/dist/src/web/index.js.map +0 -1
  275. package/dist/src/web/mount-widget.js.map +0 -1
  276. package/dist/src/web/plugin.js +0 -28
  277. package/dist/src/web/plugin.js.map +0 -1
  278. package/dist/src/web/types.js.map +0 -1
  279. package/dist/vitest.config.d.ts +0 -2
  280. package/dist/vitest.config.js +0 -8
  281. package/dist/vitest.config.js.map +0 -1
  282. /package/dist/{src/server → server}/index.js +0 -0
  283. /package/dist/{src/test → test}/widget.test.d.ts +0 -0
  284. /package/dist/{src/web → web/bridges/apps-sdk}/types.js +0 -0
  285. /package/dist/{src/web/hooks/use-call-tool.test.d.ts → web/bridges/mcp-app/use-mcp-app-context.test.d.ts} +0 -0
  286. /package/dist/{src/web/hooks/use-display-mode.test.d.ts → web/create-store.test.d.ts} +0 -0
  287. /package/dist/{src/web/hooks/use-locale.test.d.ts → web/data-llm.test.d.ts} +0 -0
  288. /package/dist/{src/web/hooks/use-open-external.test.d.ts → web/generate-helpers.test-d.d.ts} +0 -0
  289. /package/dist/{src/web/hooks/use-request-modal.test.d.ts → web/generate-helpers.test.d.ts} +0 -0
  290. /package/dist/{src/web/hooks/use-theme.test.d.ts → web/helpers/state.test.d.ts} +0 -0
  291. /package/dist/{src/web/hooks/use-tool-info.test.d.ts → web/hooks/use-call-tool.test-d.d.ts} +0 -0
  292. /package/dist/{src/web/hooks/use-user-agent.test.d.ts → web/hooks/use-call-tool.test.d.ts} +0 -0
  293. /package/dist/{src/web/hooks/use-widget-state.test.d.ts → web/hooks/use-display-mode.test.d.ts} +0 -0
  294. /package/dist/{src/web → web}/hooks/use-open-external.d.ts +0 -0
  295. /package/dist/{src/web → web}/hooks/use-send-follow-up-message.d.ts +0 -0
  296. /package/dist/{src/web → web}/hooks/use-widget-state.d.ts +0 -0
  297. /package/dist/{src/web → web}/mount-widget.d.ts +0 -0
  298. /package/dist/{src/web → web/plugin}/plugin.d.ts +0 -0
@@ -0,0 +1,99 @@
1
+ import type { McpUiResourceMeta } from "@modelcontextprotocol/ext-apps";
2
+ import { McpServer as McpServerBase, type RegisteredTool } from "@modelcontextprotocol/sdk/server/mcp.js";
3
+ import type { AnySchema, SchemaOutput, ZodRawShapeCompat } from "@modelcontextprotocol/sdk/server/zod-compat.js";
4
+ import type { RequestHandlerExtra } from "@modelcontextprotocol/sdk/shared/protocol.js";
5
+ import type { CallToolResult, Resource, ServerNotification, ServerRequest, ToolAnnotations } from "@modelcontextprotocol/sdk/types.js";
6
+ export type ToolDef<TInput = unknown, TOutput = unknown, TResponseMetadata = unknown> = {
7
+ input: TInput;
8
+ output: TOutput;
9
+ responseMetadata: TResponseMetadata;
10
+ };
11
+ /**
12
+ * Extended MCP Apps CSP with upcoming fields from ext-apps PR #158
13
+ * and Skybridge-specific fields for OpenAI compatibility
14
+ * @see https://github.com/modelcontextprotocol/ext-apps/pull/158
15
+ */
16
+ type ExtendedMcpUiResourceCsp = McpUiResourceMeta["csp"] & {
17
+ /**
18
+ * Origins that can receive openExternal redirects without safe-link modal (OpenAI-specific)
19
+ * @see https://developers.openai.com/apps-sdk/reference#component-resource-_meta-fields
20
+ */
21
+ redirectDomains?: string[];
22
+ };
23
+ /** Extended MCP Apps resource metadata with upcoming CSP fields */
24
+ type ExtendedMcpUiResourceMeta = Omit<McpUiResourceMeta, "csp"> & {
25
+ csp?: ExtendedMcpUiResourceCsp;
26
+ };
27
+ /** User-provided resource configuration with optional CSP override */
28
+ export type WidgetResourceMeta = {
29
+ ui?: ExtendedMcpUiResourceMeta;
30
+ } & Resource["_meta"];
31
+ export type WidgetHostType = "apps-sdk" | "mcp-app";
32
+ type McpServerOriginalResourceConfig = Omit<Resource, "uri" | "name" | "mimeType" | "_meta"> & {
33
+ _meta?: WidgetResourceMeta;
34
+ /** Restrict host types to a specific subset */
35
+ hosts?: WidgetHostType[];
36
+ };
37
+ type McpServerOriginalToolConfig = Omit<Parameters<typeof McpServerBase.prototype.registerTool<ZodRawShapeCompat, ZodRawShapeCompat>>[1], "inputSchema" | "outputSchema">;
38
+ type Simplify<T> = {
39
+ [K in keyof T]: T[K];
40
+ };
41
+ type ExtractStructuredContent<T> = T extends {
42
+ structuredContent: infer SC;
43
+ } ? Simplify<SC> : never;
44
+ type ExtractMeta<T> = [Extract<T, {
45
+ _meta: unknown;
46
+ }>] extends [never] ? unknown : Extract<T, {
47
+ _meta: unknown;
48
+ }> extends {
49
+ _meta: infer M;
50
+ } ? Simplify<M> : unknown;
51
+ /**
52
+ * Type-level marker interface for cross-package type inference.
53
+ * This enables TypeScript to infer tool types across package boundaries
54
+ * using structural typing on the $types property, rather than relying on
55
+ * class generic inference which fails when McpServer comes from different
56
+ * package installations.
57
+ *
58
+ * Inspired by tRPC's _def pattern and Hono's type markers.
59
+ */
60
+ export interface McpServerTypes<TTools extends Record<string, ToolDef>> {
61
+ readonly tools: TTools;
62
+ }
63
+ type ShapeOutput<Shape extends ZodRawShapeCompat> = Simplify<{
64
+ [K in keyof Shape as undefined extends SchemaOutput<Shape[K]> ? never : K]: SchemaOutput<Shape[K]>;
65
+ } & {
66
+ [K in keyof Shape as undefined extends SchemaOutput<Shape[K]> ? K : never]?: SchemaOutput<Shape[K]>;
67
+ }>;
68
+ type AddTool<TTools, TName extends string, TInput extends ZodRawShapeCompat, TOutput, TResponseMetadata = unknown> = McpServer<TTools & {
69
+ [K in TName]: ToolDef<ShapeOutput<TInput>, TOutput, TResponseMetadata>;
70
+ }>;
71
+ type ToolConfig<TInput extends ZodRawShapeCompat | AnySchema> = {
72
+ title?: string;
73
+ description?: string;
74
+ inputSchema?: TInput;
75
+ outputSchema?: ZodRawShapeCompat | AnySchema;
76
+ annotations?: ToolAnnotations;
77
+ _meta?: Record<string, unknown>;
78
+ };
79
+ type ToolHandler<TInput extends ZodRawShapeCompat, TReturn extends {
80
+ content: CallToolResult["content"];
81
+ } = CallToolResult> = (args: ShapeOutput<TInput>, extra: RequestHandlerExtra<ServerRequest, ServerNotification>) => TReturn | Promise<TReturn>;
82
+ export declare class McpServer<TTools extends Record<string, ToolDef> = Record<never, ToolDef>> extends McpServerBase {
83
+ readonly $types: McpServerTypes<TTools>;
84
+ registerWidget<TName extends string, TInput extends ZodRawShapeCompat, TReturn extends {
85
+ content: CallToolResult["content"];
86
+ }>(name: TName, resourceConfig: McpServerOriginalResourceConfig, toolConfig: McpServerOriginalToolConfig & {
87
+ inputSchema?: TInput;
88
+ outputSchema?: ZodRawShapeCompat | AnySchema;
89
+ }, toolCallback: ToolHandler<TInput, TReturn>): AddTool<TTools, TName, TInput, ExtractStructuredContent<TReturn>, ExtractMeta<TReturn>>;
90
+ registerTool<TName extends string, InputArgs extends ZodRawShapeCompat, TReturn extends {
91
+ content: CallToolResult["content"];
92
+ }>(name: TName, config: ToolConfig<InputArgs>, cb: ToolHandler<InputArgs, TReturn>): AddTool<TTools, TName, InputArgs, ExtractStructuredContent<TReturn>, ExtractMeta<TReturn>>;
93
+ registerTool<InputArgs extends ZodRawShapeCompat>(name: string, config: ToolConfig<InputArgs>, cb: ToolHandler<InputArgs>): RegisteredTool;
94
+ private registerWidgetResource;
95
+ private lookupDistFile;
96
+ private lookupDistFileWithIndexFallback;
97
+ private readManifest;
98
+ }
99
+ export {};
@@ -0,0 +1,138 @@
1
+ import { readFileSync } from "node:fs";
2
+ import path from "node:path";
3
+ import { McpServer as McpServerBase, } from "@modelcontextprotocol/sdk/server/mcp.js";
4
+ import { mergeWith, union } from "es-toolkit";
5
+ import { templateHelper } from "./templateHelper.js";
6
+ const mergeWithUnion = (target, source) => {
7
+ return mergeWith(target, source, (targetVal, sourceVal) => {
8
+ if (Array.isArray(targetVal) && Array.isArray(sourceVal)) {
9
+ return union(targetVal, sourceVal);
10
+ }
11
+ });
12
+ };
13
+ export class McpServer extends McpServerBase {
14
+ registerWidget(name, resourceConfig, toolConfig, toolCallback) {
15
+ const userMeta = resourceConfig._meta;
16
+ const toolMeta = {
17
+ ...toolConfig._meta,
18
+ };
19
+ if (!resourceConfig.hosts || resourceConfig.hosts.includes("apps-sdk")) {
20
+ const widgetConfig = {
21
+ hostType: "apps-sdk",
22
+ uri: `ui://widgets/apps-sdk/${name}.html`,
23
+ mimeType: "text/html+skybridge",
24
+ buildContentMeta: ({ resourceDomains, connectDomains, domain }) => {
25
+ const userUi = userMeta?.ui;
26
+ const userCsp = userUi?.csp;
27
+ const defaults = {
28
+ "openai/widgetCSP": {
29
+ resource_domains: resourceDomains,
30
+ connect_domains: connectDomains,
31
+ },
32
+ "openai/widgetDomain": domain,
33
+ "openai/widgetDescription": toolConfig.description,
34
+ };
35
+ const fromUi = {
36
+ "openai/widgetCSP": {
37
+ resource_domains: userCsp?.resourceDomains,
38
+ connect_domains: userCsp?.connectDomains,
39
+ frame_domains: userCsp?.frameDomains,
40
+ redirect_domains: userCsp?.redirectDomains,
41
+ },
42
+ "openai/widgetDomain": userUi?.domain,
43
+ "openai/widgetPrefersBorder": userUi?.prefersBorder,
44
+ };
45
+ const directOpenaiKeys = Object.fromEntries(Object.entries(userMeta ?? {}).filter(([key]) => key.startsWith("openai/")));
46
+ return mergeWithUnion(mergeWithUnion(defaults, fromUi), directOpenaiKeys);
47
+ },
48
+ };
49
+ this.registerWidgetResource({
50
+ name,
51
+ widgetConfig,
52
+ resourceConfig,
53
+ });
54
+ toolMeta["openai/outputTemplate"] = widgetConfig.uri;
55
+ }
56
+ if (!resourceConfig.hosts || resourceConfig.hosts.includes("mcp-app")) {
57
+ const widgetConfig = {
58
+ hostType: "mcp-app",
59
+ uri: `ui://widgets/ext-apps/${name}.html`,
60
+ mimeType: "text/html;profile=mcp-app",
61
+ buildContentMeta: ({ resourceDomains, connectDomains, domain }) => {
62
+ const defaults = {
63
+ ui: {
64
+ csp: {
65
+ resourceDomains,
66
+ connectDomains,
67
+ },
68
+ domain,
69
+ },
70
+ };
71
+ return mergeWithUnion(defaults, { ui: userMeta?.ui });
72
+ },
73
+ };
74
+ this.registerWidgetResource({
75
+ name,
76
+ widgetConfig,
77
+ resourceConfig,
78
+ });
79
+ toolMeta.ui = { resourceUri: widgetConfig.uri };
80
+ }
81
+ this.registerTool(name, {
82
+ ...toolConfig,
83
+ _meta: toolMeta,
84
+ }, toolCallback);
85
+ return this;
86
+ }
87
+ registerTool(name, config, cb) {
88
+ super.registerTool(name, config, cb);
89
+ return this;
90
+ }
91
+ registerWidgetResource({ name, widgetConfig, resourceConfig, }) {
92
+ const { hostType, uri: widgetUri, mimeType, buildContentMeta, } = widgetConfig;
93
+ this.registerResource(name, widgetUri, { ...resourceConfig, _meta: resourceConfig._meta }, async (uri, extra) => {
94
+ const isProduction = process.env.NODE_ENV === "production";
95
+ const serverUrl = isProduction
96
+ ? `https://${extra?.requestInfo?.headers?.["x-forwarded-host"] ?? extra?.requestInfo?.headers?.host}`
97
+ : "http://localhost:3000";
98
+ const html = isProduction
99
+ ? templateHelper.renderProduction({
100
+ hostType,
101
+ serverUrl,
102
+ widgetFile: this.lookupDistFileWithIndexFallback(`src/widgets/${name}`),
103
+ styleFile: this.lookupDistFile("style.css"),
104
+ })
105
+ : templateHelper.renderDevelopment({
106
+ hostType,
107
+ serverUrl,
108
+ widgetName: name,
109
+ });
110
+ const VITE_HMR_WEBSOCKET_DEFAULT_URL = "ws://localhost:24678";
111
+ const contentMeta = buildContentMeta({
112
+ resourceDomains: [serverUrl],
113
+ connectDomains: !isProduction ? [VITE_HMR_WEBSOCKET_DEFAULT_URL] : [],
114
+ domain: serverUrl,
115
+ baseUriDomains: [serverUrl],
116
+ });
117
+ return {
118
+ contents: [
119
+ { uri: uri.href, mimeType, text: html, _meta: contentMeta },
120
+ ],
121
+ };
122
+ });
123
+ }
124
+ lookupDistFile(key) {
125
+ const manifest = this.readManifest();
126
+ return manifest[key]?.file;
127
+ }
128
+ lookupDistFileWithIndexFallback(basePath) {
129
+ const manifest = this.readManifest();
130
+ const flatFileKey = `${basePath}.tsx`;
131
+ const indexFileKey = `${basePath}/index.tsx`;
132
+ return manifest[flatFileKey]?.file ?? manifest[indexFileKey]?.file;
133
+ }
134
+ readManifest() {
135
+ return JSON.parse(readFileSync(path.join(process.cwd(), "dist", "assets", ".vite", "manifest.json"), "utf-8"));
136
+ }
137
+ }
138
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../../src/server/server.ts"],"names":[],"mappings":"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"}
@@ -1,12 +1,15 @@
1
+ import type { WidgetHostType } from "./server.js";
1
2
  declare class TemplateHelper {
2
3
  private templateCache;
3
4
  private loadTemplate;
4
5
  renderProduction(data: {
6
+ hostType: WidgetHostType;
5
7
  serverUrl: string;
6
8
  widgetFile: string;
7
9
  styleFile: string;
8
10
  }): string;
9
11
  renderDevelopment(data: {
12
+ hostType: WidgetHostType;
10
13
  serverUrl: string;
11
14
  widgetName: string;
12
15
  }): string;
@@ -1,14 +1,15 @@
1
- import Handlebars from "handlebars";
2
1
  import { readFileSync } from "node:fs";
3
- import { join, dirname } from "node:path";
2
+ import { dirname, join } from "node:path";
4
3
  import { fileURLToPath } from "node:url";
4
+ import Handlebars from "handlebars";
5
5
  const __filename = fileURLToPath(import.meta.url);
6
6
  const __dirname = dirname(__filename);
7
7
  class TemplateHelper {
8
8
  templateCache = new Map();
9
9
  loadTemplate(templateName) {
10
- if (this.templateCache.has(templateName)) {
11
- return this.templateCache.get(templateName);
10
+ const cached = this.templateCache.get(templateName);
11
+ if (cached) {
12
+ return cached;
12
13
  }
13
14
  const templatePath = join(__dirname, "templates", `${templateName}.hbs`);
14
15
  const templateSource = readFileSync(templatePath, "utf-8");
@@ -0,0 +1 @@
1
+ {"version":3,"file":"templateHelper.js","sourceRoot":"","sources":["../../src/server/templateHelper.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AACvC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AACzC,OAAO,UAAU,MAAM,YAAY,CAAC;AAGpC,MAAM,UAAU,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AAClD,MAAM,SAAS,GAAG,OAAO,CAAC,UAAU,CAAC,CAAC;AAEtC,MAAM,cAAc;IACV,aAAa,GAAG,IAAI,GAAG,EAAsC,CAAC;IAE9D,YAAY,CAAC,YAAoB;QACvC,MAAM,MAAM,GAAG,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;QACpD,IAAI,MAAM,EAAE,CAAC;YACX,OAAO,MAAM,CAAC;QAChB,CAAC;QAED,MAAM,YAAY,GAAG,IAAI,CAAC,SAAS,EAAE,WAAW,EAAE,GAAG,YAAY,MAAM,CAAC,CAAC;QACzE,MAAM,cAAc,GAAG,YAAY,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;QAC3D,MAAM,QAAQ,GAAG,UAAU,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;QAEpD,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,YAAY,EAAE,QAAQ,CAAC,CAAC;QAC/C,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED,gBAAgB,CAAC,IAKhB;QACC,MAAM,QAAQ,GAAG,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC,CAAC;QACjD,OAAO,QAAQ,CAAC,IAAI,CAAC,CAAC;IACxB,CAAC;IAED,iBAAiB,CAAC,IAIjB;QACC,MAAM,QAAQ,GAAG,IAAI,CAAC,YAAY,CAAC,aAAa,CAAC,CAAC;QAClD,OAAO,QAAQ,CAAC,IAAI,CAAC,CAAC;IACxB,CAAC;CACF;AAED,MAAM,CAAC,MAAM,cAAc,GAAG,IAAI,cAAc,EAAE,CAAC"}
@@ -0,0 +1,66 @@
1
+ <base href="{{serverUrl}}" />
2
+ <script type="module">window.skybridge = { hostType: "{{hostType}}" };</script>
3
+ <script type="module">
4
+ import { injectIntoGlobalHook } from "{{serverUrl}}/assets/@react-refresh";
5
+ injectIntoGlobalHook(window); window.$RefreshReg$ = () => {};
6
+ window.$RefreshSig$ = () => (type) => type;
7
+ window.__vite_plugin_react_preamble_installed__ = true;
8
+ </script>
9
+ <script type="module" src="{{serverUrl}}/@vite/client"></script>
10
+ <script type="module">
11
+ // Checks for browser support and shows error if local network access is denied
12
+ (async () => {
13
+ if (!navigator.permissions?.query) {
14
+ return;
15
+ }
16
+
17
+ // Skip for non-http(s) protocols (file://, custom protocols in Electron, etc.)
18
+ const protocol = window.location.protocol;
19
+ const isNonHttpProtocol = protocol !== 'http:' && protocol !== 'https:'
20
+ if (isNonHttpProtocol) {
21
+ return;
22
+ }
23
+
24
+ const host = window.location.hostname;
25
+ const isLoopback = host === 'localhost'
26
+ || /^127\.(\d{1,3})\.(\d{1,3})\.(\d{1,3})$/.test(host)
27
+ || host === '::1';
28
+ if (isLoopback) {
29
+ return;
30
+ }
31
+
32
+ try {
33
+ const status = await navigator.permissions.query({ name: "local-network-access" });
34
+ if (status.state === "denied") {
35
+ const errorDiv = document.createElement("div");
36
+ errorDiv.style.cssText = "background: #fef2f2; border: 2px solid #ef4444; border-radius: 8px; padding: 16px; text-align: center; z-index: 10000; font-family: system-ui, sans-serif;";
37
+
38
+ const errorTitle = document.createElement("div");
39
+ errorTitle.style.cssText = "color: #ef4444; font-size: 18px; font-weight: 600; margin-bottom: 8px;";
40
+ errorTitle.textContent = "Error: Local network access permission is denied.";
41
+
42
+ const errorMessage = document.createElement("div");
43
+ errorMessage.style.cssText = "color: #ef4444; font-size: 14px;";
44
+ errorMessage.textContent = "Local network access is required for your widget to connect to the local dev server. Please enable it in your browser settings. ";
45
+
46
+ const link = document.createElement("a");
47
+ link.href = "https://developer.chrome.com/blog/local-network-access";
48
+ link.target = "_blank";
49
+ link.rel = "noopener noreferrer";
50
+ link.style.cssText = "color: #ef4444; text-decoration: underline;";
51
+ link.textContent = "Learn more";
52
+ errorMessage.appendChild(link);
53
+
54
+ errorDiv.appendChild(errorTitle);
55
+ errorDiv.appendChild(errorMessage);
56
+ document.body.appendChild(errorDiv);
57
+ }
58
+ } catch (e) {
59
+ // Permission API doesn't support local-network-access, ignore silently
60
+ }
61
+ })();
62
+ </script>
63
+ <div id="root"></div>
64
+ <script type="module" id="dev-widget-entry">
65
+ import('{{serverUrl}}/src/widgets/{{widgetName}}');
66
+ </script>
@@ -1,4 +1,5 @@
1
1
  <base href="{{serverUrl}}" />
2
+ <script type="module">window.skybridge = { hostType: "{{hostType}}" };</script>
2
3
  <div id="root"></div>
3
4
  <script type="module">
4
5
  import('{{serverUrl}}/assets/{{widgetFile}}');
@@ -1,4 +1,4 @@
1
- import { type RequestHandler } from "express";
1
+ import { type Router } from "express";
2
2
  /**
3
3
  * Install Vite dev server
4
4
  * This router MUST be installed at the application root, like so:
@@ -9,4 +9,4 @@ import { type RequestHandler } from "express";
9
9
  * app.use(await widgetsRouter());
10
10
  * }
11
11
  */
12
- export declare const widgetsDevServer: () => Promise<RequestHandler>;
12
+ export declare const widgetsDevServer: () => Promise<Router>;
@@ -1,6 +1,7 @@
1
- import express, {} from "express";
2
- import cors from "cors";
1
+ import { existsSync } from "node:fs";
3
2
  import path from "node:path";
3
+ import cors from "cors";
4
+ import express, {} from "express";
4
5
  /**
5
6
  * Install Vite dev server
6
7
  * This router MUST be installed at the application root, like so:
@@ -14,10 +15,17 @@ import path from "node:path";
14
15
  export const widgetsDevServer = async () => {
15
16
  const router = express.Router();
16
17
  const { createServer, searchForWorkspaceRoot, loadConfigFromFile } = await import("vite");
17
- const workspaceRoot = searchForWorkspaceRoot(process.cwd());
18
- const webAppRoot = path.join(workspaceRoot, "web");
18
+ // Since 0.16.0, the template is a single package that does not rely on workspace.
19
+ // It means that, when starting the server, the working dir is the template root
20
+ // hence we don't need to walk up the tree to find the workspace, which does not exist anymore.
21
+ let webAppRoot = path.join(process.cwd(), "web");
22
+ // fallback to the old behavior for backward compatibility
23
+ const hasWebAppRoot = existsSync(webAppRoot);
24
+ if (!hasWebAppRoot) {
25
+ const workspaceRoot = searchForWorkspaceRoot(process.cwd());
26
+ webAppRoot = path.join(workspaceRoot, "web");
27
+ }
19
28
  const configResult = await loadConfigFromFile({ command: "serve", mode: "development" }, path.join(webAppRoot, "vite.config.ts"), webAppRoot);
20
- // Remove build-specific options that don't apply to dev server
21
29
  const { build, preview, ...devConfig } = configResult?.config || {};
22
30
  const vite = await createServer({
23
31
  ...devConfig,
@@ -0,0 +1 @@
1
+ {"version":3,"file":"widgetsDevServer.js","sourceRoot":"","sources":["../../src/server/widgetsDevServer.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACrC,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,OAAO,EAAE,EAAe,MAAM,SAAS,CAAC;AAC/C;;;;;;;;;GASG;AACH,MAAM,CAAC,MAAM,gBAAgB,GAAG,KAAK,IAAqB,EAAE;IAC1D,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IAEhC,MAAM,EAAE,YAAY,EAAE,sBAAsB,EAAE,kBAAkB,EAAE,GAChE,MAAM,MAAM,CAAC,MAAM,CAAC,CAAC;IAEvB,kFAAkF;IAClF,gFAAgF;IAChF,+FAA+F;IAC/F,IAAI,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,KAAK,CAAC,CAAC;IAEjD,0DAA0D;IAC1D,MAAM,aAAa,GAAG,UAAU,CAAC,UAAU,CAAC,CAAC;IAC7C,IAAI,CAAC,aAAa,EAAE,CAAC;QACnB,MAAM,aAAa,GAAG,sBAAsB,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;QAC5D,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,KAAK,CAAC,CAAC;IAC/C,CAAC;IAED,MAAM,YAAY,GAAG,MAAM,kBAAkB,CAC3C,EAAE,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,aAAa,EAAE,EACzC,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,gBAAgB,CAAC,EACvC,UAAU,CACX,CAAC;IAEF,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,GAAG,SAAS,EAAE,GAAG,YAAY,EAAE,MAAM,IAAI,EAAE,CAAC;IAEpE,MAAM,IAAI,GAAG,MAAM,YAAY,CAAC;QAC9B,GAAG,SAAS;QACZ,UAAU,EAAE,KAAK,EAAE,kFAAkF;QACrG,OAAO,EAAE,QAAQ;QACjB,MAAM,EAAE;YACN,YAAY,EAAE,IAAI;YAClB,cAAc,EAAE,IAAI;SACrB;QACD,IAAI,EAAE,UAAU;QAChB,YAAY,EAAE;YACZ,OAAO,EAAE,CAAC,OAAO,EAAE,kBAAkB,CAAC;SACvC;KACF,CAAC,CAAC;IAEH,MAAM,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC;IACnB,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;IAElC,OAAO,MAAM,CAAC;AAChB,CAAC,CAAC"}
@@ -0,0 +1,135 @@
1
+ import { type MockInstance } from "vitest";
2
+ import { McpServer } from "../server/server.js";
3
+ /**
4
+ * Creates a real McpServer instance for testing
5
+ */
6
+ export declare function createMockMcpServer(): {
7
+ server: McpServer;
8
+ mockRegisterResource: MockInstance<McpServer["registerResource"]>;
9
+ mockRegisterTool: MockInstance<McpServer["registerTool"]>;
10
+ };
11
+ export declare function createTestServer(): McpServer<Record<never, import("../server/server.js").ToolDef<unknown, unknown, unknown>> & {
12
+ "search-voyage": import("../server/server.js").ToolDef<{
13
+ destination: string;
14
+ departureDate?: string | undefined;
15
+ maxPrice?: number | undefined;
16
+ }, {
17
+ results: {
18
+ id: string;
19
+ name: string;
20
+ price: number;
21
+ }[];
22
+ totalCount: number;
23
+ }, unknown>;
24
+ } & {
25
+ "get-trip-details": import("../server/server.js").ToolDef<{
26
+ tripId: string;
27
+ }, {
28
+ name: string;
29
+ description: string;
30
+ images: string[];
31
+ }, unknown>;
32
+ } & {
33
+ "no-input-widget": import("../server/server.js").ToolDef<{}, {}, unknown>;
34
+ } & {
35
+ "inferred-output-widget": import("../server/server.js").ToolDef<{
36
+ query: string;
37
+ }, {
38
+ inferredResults: {
39
+ id: string;
40
+ score: number;
41
+ }[];
42
+ inferredCount: number;
43
+ }, unknown>;
44
+ } & {
45
+ "calculate-price": import("../server/server.js").ToolDef<{
46
+ tripId: string;
47
+ passengers: number;
48
+ }, {
49
+ totalPrice: number;
50
+ currency: string;
51
+ }, unknown>;
52
+ } & {
53
+ "inferred-tool": import("../server/server.js").ToolDef<{
54
+ itemId: string;
55
+ }, {
56
+ itemDetails: {
57
+ name: string;
58
+ available: boolean;
59
+ };
60
+ fetchedAt: string;
61
+ }, unknown>;
62
+ } & {
63
+ "widget-with-metadata": import("../server/server.js").ToolDef<{
64
+ resourceId: string;
65
+ }, {
66
+ data: {
67
+ id: string;
68
+ loaded: boolean;
69
+ };
70
+ }, {
71
+ requestId: string;
72
+ timestamp: number;
73
+ cached: boolean;
74
+ }>;
75
+ } & {
76
+ "tool-with-metadata": import("../server/server.js").ToolDef<{
77
+ query: string;
78
+ }, {
79
+ results: string[];
80
+ }, {
81
+ executionTime: number;
82
+ source: string;
83
+ }>;
84
+ } & {
85
+ "widget-with-mixed-returns": import("../server/server.js").ToolDef<{
86
+ shouldSucceed: boolean;
87
+ }, {
88
+ error: string;
89
+ data?: undefined;
90
+ } | {
91
+ data: string;
92
+ error?: undefined;
93
+ }, {
94
+ processedAt: number;
95
+ region: string;
96
+ }>;
97
+ }>;
98
+ export declare function createMinimalTestServer(): McpServer<Record<never, import("../server/server.js").ToolDef<unknown, unknown, unknown>> & {
99
+ "search-voyage": import("../server/server.js").ToolDef<{
100
+ destination: string;
101
+ }, {
102
+ results: {
103
+ id: string;
104
+ }[];
105
+ }, unknown>;
106
+ }>;
107
+ export declare function createInterfaceTestServer(): McpServer<Record<never, import("../server/server.js").ToolDef<unknown, unknown, unknown>> & {
108
+ "interface-widget": import("../server/server.js").ToolDef<{
109
+ id: string;
110
+ }, {
111
+ itemName: string;
112
+ quantity: number;
113
+ }, {
114
+ processedBy: string;
115
+ version: number;
116
+ }>;
117
+ }>;
118
+ /**
119
+ * Mock extra parameter for resource callback
120
+ */
121
+ export declare function createMockExtra(host: string): {
122
+ requestInfo: {
123
+ headers: {
124
+ host: string;
125
+ };
126
+ };
127
+ };
128
+ /**
129
+ * Sets up environment variables for testing
130
+ */
131
+ export declare function setTestEnv(env: Record<string, string>): void;
132
+ /**
133
+ * Resets environment variables
134
+ */
135
+ export declare function resetTestEnv(): void;