skybridge 0.0.0-dev.f421483 → 0.0.0-dev.f423b34

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 (460) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +153 -0
  3. package/bin/run.js +5 -0
  4. package/dist/cli/detect-port.d.ts +18 -0
  5. package/dist/cli/detect-port.js +61 -0
  6. package/dist/cli/detect-port.js.map +1 -0
  7. package/dist/cli/header.d.ts +4 -0
  8. package/dist/cli/header.js +6 -0
  9. package/dist/cli/header.js.map +1 -0
  10. package/dist/cli/run-command.d.ts +2 -0
  11. package/dist/cli/run-command.js +43 -0
  12. package/dist/cli/run-command.js.map +1 -0
  13. package/dist/cli/telemetry.d.ts +7 -0
  14. package/dist/cli/telemetry.js +123 -0
  15. package/dist/cli/telemetry.js.map +1 -0
  16. package/dist/cli/tunnel-control-server.d.ts +9 -0
  17. package/dist/cli/tunnel-control-server.js +31 -0
  18. package/dist/cli/tunnel-control-server.js.map +1 -0
  19. package/dist/cli/tunnel-control-server.test.js +39 -0
  20. package/dist/cli/tunnel-control-server.test.js.map +1 -0
  21. package/dist/cli/tunnel-handler.d.ts +3 -0
  22. package/dist/cli/tunnel-handler.js +48 -0
  23. package/dist/cli/tunnel-handler.js.map +1 -0
  24. package/dist/cli/tunnel-handler.test.js +105 -0
  25. package/dist/cli/tunnel-handler.test.js.map +1 -0
  26. package/dist/cli/tunnel.d.ts +57 -0
  27. package/dist/cli/tunnel.js +154 -0
  28. package/dist/cli/tunnel.js.map +1 -0
  29. package/dist/cli/tunnel.test.js +190 -0
  30. package/dist/cli/tunnel.test.js.map +1 -0
  31. package/dist/cli/types.d.ts +5 -0
  32. package/dist/cli/types.js.map +1 -0
  33. package/dist/cli/use-execute-steps.d.ts +11 -0
  34. package/dist/cli/use-execute-steps.js +36 -0
  35. package/dist/cli/use-execute-steps.js.map +1 -0
  36. package/dist/cli/use-messages.d.ts +3 -0
  37. package/dist/cli/use-messages.js +11 -0
  38. package/dist/cli/use-messages.js.map +1 -0
  39. package/dist/cli/use-nodemon.d.ts +2 -0
  40. package/dist/cli/use-nodemon.js +73 -0
  41. package/dist/cli/use-nodemon.js.map +1 -0
  42. package/dist/cli/use-open-browser.d.ts +1 -0
  43. package/dist/cli/use-open-browser.js +44 -0
  44. package/dist/cli/use-open-browser.js.map +1 -0
  45. package/dist/cli/use-tunnel.d.ts +14 -0
  46. package/dist/cli/use-tunnel.js +131 -0
  47. package/dist/cli/use-tunnel.js.map +1 -0
  48. package/dist/cli/use-typescript-check.d.ts +9 -0
  49. package/dist/cli/use-typescript-check.js +94 -0
  50. package/dist/cli/use-typescript-check.js.map +1 -0
  51. package/dist/commands/build.d.ts +9 -0
  52. package/dist/commands/build.js +102 -0
  53. package/dist/commands/build.js.map +1 -0
  54. package/dist/commands/dev.d.ts +12 -0
  55. package/dist/commands/dev.js +80 -0
  56. package/dist/commands/dev.js.map +1 -0
  57. package/dist/commands/start.d.ts +9 -0
  58. package/dist/commands/start.js +49 -0
  59. package/dist/commands/start.js.map +1 -0
  60. package/dist/commands/telemetry/disable.d.ts +5 -0
  61. package/dist/commands/telemetry/disable.js +14 -0
  62. package/dist/commands/telemetry/disable.js.map +1 -0
  63. package/dist/commands/telemetry/enable.d.ts +5 -0
  64. package/dist/commands/telemetry/enable.js +14 -0
  65. package/dist/commands/telemetry/enable.js.map +1 -0
  66. package/dist/commands/telemetry/status.d.ts +5 -0
  67. package/dist/commands/telemetry/status.js +14 -0
  68. package/dist/commands/telemetry/status.js.map +1 -0
  69. package/dist/server/asset-base-url-transform-plugin.d.ts +10 -0
  70. package/dist/server/asset-base-url-transform-plugin.js +33 -0
  71. package/dist/server/asset-base-url-transform-plugin.js.map +1 -0
  72. package/dist/server/asset-base-url-transform-plugin.test.js +84 -0
  73. package/dist/server/asset-base-url-transform-plugin.test.js.map +1 -0
  74. package/dist/server/content-helpers.d.ts +27 -0
  75. package/dist/server/content-helpers.js +46 -0
  76. package/dist/server/content-helpers.js.map +1 -0
  77. package/dist/server/content-helpers.test.js +70 -0
  78. package/dist/server/content-helpers.test.js.map +1 -0
  79. package/dist/server/express.d.ts +11 -0
  80. package/dist/server/express.js +101 -0
  81. package/dist/server/express.js.map +1 -0
  82. package/dist/server/express.test.js +430 -0
  83. package/dist/server/express.test.js.map +1 -0
  84. package/dist/server/file-ref.d.ts +8 -0
  85. package/dist/server/file-ref.js +8 -0
  86. package/dist/server/file-ref.js.map +1 -0
  87. package/dist/server/index.d.ts +7 -0
  88. package/dist/server/index.js +5 -0
  89. package/dist/server/index.js.map +1 -0
  90. package/dist/{src/server → server}/inferUtilityTypes.d.ts +6 -6
  91. package/dist/server/inferUtilityTypes.js.map +1 -0
  92. package/dist/server/metric.d.ts +14 -0
  93. package/dist/server/metric.js +62 -0
  94. package/dist/server/metric.js.map +1 -0
  95. package/dist/server/middleware.d.ts +124 -0
  96. package/dist/server/middleware.js +93 -0
  97. package/dist/server/middleware.js.map +1 -0
  98. package/dist/server/middleware.test-d.js +75 -0
  99. package/dist/server/middleware.test-d.js.map +1 -0
  100. package/dist/server/middleware.test.js +493 -0
  101. package/dist/server/middleware.test.js.map +1 -0
  102. package/dist/server/server.d.ts +199 -0
  103. package/dist/server/server.js +468 -0
  104. package/dist/server/server.js.map +1 -0
  105. package/dist/{src/server → server}/templateHelper.d.ts +5 -7
  106. package/dist/server/templateHelper.js +11 -0
  107. package/dist/server/templateHelper.js.map +1 -0
  108. package/dist/server/templates.generated.d.ts +4 -0
  109. package/dist/server/templates.generated.js +47 -0
  110. package/dist/server/templates.generated.js.map +1 -0
  111. package/dist/server/tunnel-proxy-router.d.ts +7 -0
  112. package/dist/server/tunnel-proxy-router.js +110 -0
  113. package/dist/server/tunnel-proxy-router.js.map +1 -0
  114. package/dist/server/tunnel-proxy-router.test.js +229 -0
  115. package/dist/server/tunnel-proxy-router.test.js.map +1 -0
  116. package/dist/server/viewsDevServer.d.ts +14 -0
  117. package/dist/server/viewsDevServer.js +45 -0
  118. package/dist/server/viewsDevServer.js.map +1 -0
  119. package/dist/{src/test → test}/utils.d.ts +13 -21
  120. package/dist/{src/test → test}/utils.js +42 -37
  121. package/dist/test/utils.js.map +1 -0
  122. package/dist/test/view.test.js +523 -0
  123. package/dist/test/view.test.js.map +1 -0
  124. package/dist/version.d.ts +1 -0
  125. package/dist/version.js +3 -0
  126. package/dist/version.js.map +1 -0
  127. package/dist/web/bridges/apps-sdk/adaptor.d.ts +26 -0
  128. package/dist/web/bridges/apps-sdk/adaptor.js +102 -0
  129. package/dist/web/bridges/apps-sdk/adaptor.js.map +1 -0
  130. package/dist/web/bridges/apps-sdk/bridge.d.ts +10 -0
  131. package/dist/{src/web/bridges/apps-sdk-bridge.js → web/bridges/apps-sdk/bridge.js} +2 -2
  132. package/dist/web/bridges/apps-sdk/bridge.js.map +1 -0
  133. package/dist/web/bridges/apps-sdk/index.d.ts +5 -0
  134. package/dist/web/bridges/apps-sdk/index.js +5 -0
  135. package/dist/web/bridges/apps-sdk/index.js.map +1 -0
  136. package/dist/web/bridges/apps-sdk/types.d.ts +133 -0
  137. package/dist/{src/web → web/bridges/apps-sdk}/types.js +0 -1
  138. package/dist/web/bridges/apps-sdk/types.js.map +1 -0
  139. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.d.ts +2 -0
  140. package/dist/{src/web/bridges/hooks/use-apps-sdk-bridge.js → web/bridges/apps-sdk/use-apps-sdk-context.js} +3 -3
  141. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.js.map +1 -0
  142. package/dist/web/bridges/get-adaptor.d.ts +2 -0
  143. package/dist/web/bridges/get-adaptor.js +8 -0
  144. package/dist/web/bridges/get-adaptor.js.map +1 -0
  145. package/dist/web/bridges/index.d.ts +5 -0
  146. package/dist/web/bridges/index.js +6 -0
  147. package/dist/web/bridges/index.js.map +1 -0
  148. package/dist/web/bridges/mcp-app/adaptor.d.ts +50 -0
  149. package/dist/web/bridges/mcp-app/adaptor.js +271 -0
  150. package/dist/web/bridges/mcp-app/adaptor.js.map +1 -0
  151. package/dist/web/bridges/mcp-app/bridge.d.ts +26 -0
  152. package/dist/web/bridges/mcp-app/bridge.js +102 -0
  153. package/dist/web/bridges/mcp-app/bridge.js.map +1 -0
  154. package/dist/web/bridges/mcp-app/index.d.ts +4 -0
  155. package/dist/web/bridges/mcp-app/index.js +4 -0
  156. package/dist/web/bridges/mcp-app/index.js.map +1 -0
  157. package/dist/web/bridges/mcp-app/types.d.ts +8 -0
  158. package/dist/web/bridges/mcp-app/types.js +2 -0
  159. package/dist/web/bridges/mcp-app/types.js.map +1 -0
  160. package/dist/web/bridges/mcp-app/use-mcp-app-context.d.ts +7 -0
  161. package/dist/web/bridges/mcp-app/use-mcp-app-context.js +7 -0
  162. package/dist/web/bridges/mcp-app/use-mcp-app-context.js.map +1 -0
  163. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js +26 -0
  164. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js.map +1 -0
  165. package/dist/web/bridges/types.d.ts +117 -0
  166. package/dist/web/bridges/types.js +2 -0
  167. package/dist/web/bridges/types.js.map +1 -0
  168. package/dist/web/bridges/use-host-context.d.ts +2 -0
  169. package/dist/web/bridges/use-host-context.js +8 -0
  170. package/dist/web/bridges/use-host-context.js.map +1 -0
  171. package/dist/web/components/modal-provider.d.ts +4 -0
  172. package/dist/web/components/modal-provider.js +45 -0
  173. package/dist/web/components/modal-provider.js.map +1 -0
  174. package/dist/web/create-store.js +38 -0
  175. package/dist/web/create-store.js.map +1 -0
  176. package/dist/web/create-store.test.js +129 -0
  177. package/dist/web/create-store.test.js.map +1 -0
  178. package/dist/{src/web → web}/data-llm.d.ts +1 -1
  179. package/dist/{src/web → web}/data-llm.js +7 -5
  180. package/dist/web/data-llm.js.map +1 -0
  181. package/dist/web/data-llm.test.js +142 -0
  182. package/dist/web/data-llm.test.js.map +1 -0
  183. package/dist/{src/web → web}/generate-helpers.d.ts +22 -19
  184. package/dist/{src/web → web}/generate-helpers.js +20 -18
  185. package/dist/web/generate-helpers.js.map +1 -0
  186. package/dist/{src/web → web}/generate-helpers.test-d.js +26 -26
  187. package/dist/web/generate-helpers.test-d.js.map +1 -0
  188. package/dist/web/generate-helpers.test.js.map +1 -0
  189. package/dist/{src/web → web}/helpers/state.d.ts +2 -2
  190. package/dist/web/helpers/state.js +45 -0
  191. package/dist/web/helpers/state.js.map +1 -0
  192. package/dist/{src/web → web}/helpers/state.test.js +9 -9
  193. package/dist/web/helpers/state.test.js.map +1 -0
  194. package/dist/{src/web → web}/hooks/index.d.ts +5 -3
  195. package/dist/{src/web → web}/hooks/index.js +4 -2
  196. package/dist/web/hooks/index.js.map +1 -0
  197. package/dist/{src/web → web}/hooks/test/utils.d.ts +8 -2
  198. package/dist/web/hooks/test/utils.js +64 -0
  199. package/dist/web/hooks/test/utils.js.map +1 -0
  200. package/dist/{src/web → web}/hooks/use-call-tool.d.ts +2 -1
  201. package/dist/{src/web → web}/hooks/use-call-tool.js +2 -2
  202. package/dist/web/hooks/use-call-tool.js.map +1 -0
  203. package/dist/web/hooks/use-call-tool.test-d.js.map +1 -0
  204. package/dist/{src/web → web}/hooks/use-call-tool.test.js +0 -5
  205. package/dist/web/hooks/use-call-tool.test.js.map +1 -0
  206. package/dist/web/hooks/use-display-mode.d.ts +4 -0
  207. package/dist/web/hooks/use-display-mode.js +9 -0
  208. package/dist/web/hooks/use-display-mode.js.map +1 -0
  209. package/dist/web/hooks/use-display-mode.test-d.js +8 -0
  210. package/dist/web/hooks/use-display-mode.test-d.js.map +1 -0
  211. package/dist/web/hooks/use-display-mode.test.js.map +1 -0
  212. package/dist/web/hooks/use-files.d.ts +7 -0
  213. package/dist/web/hooks/use-files.js +10 -0
  214. package/dist/web/hooks/use-files.js.map +1 -0
  215. package/dist/web/hooks/use-files.test.d.ts +1 -0
  216. package/dist/web/hooks/use-files.test.js +54 -0
  217. package/dist/web/hooks/use-files.test.js.map +1 -0
  218. package/dist/{src/web → web}/hooks/use-layout.d.ts +2 -2
  219. package/dist/{src/web → web}/hooks/use-layout.js +4 -4
  220. package/dist/web/hooks/use-layout.js.map +1 -0
  221. package/dist/web/hooks/use-layout.test.d.ts +1 -0
  222. package/dist/{src/web → web}/hooks/use-layout.test.js +7 -6
  223. package/dist/web/hooks/use-layout.test.js.map +1 -0
  224. package/dist/web/hooks/use-open-external.d.ts +3 -0
  225. package/dist/web/hooks/use-open-external.js +8 -0
  226. package/dist/web/hooks/use-open-external.js.map +1 -0
  227. package/dist/web/hooks/use-open-external.test.d.ts +1 -0
  228. package/dist/{src/web → web}/hooks/use-open-external.test.js +27 -12
  229. package/dist/web/hooks/use-open-external.test.js.map +1 -0
  230. package/dist/web/hooks/use-request-close.d.ts +2 -0
  231. package/dist/web/hooks/use-request-close.js +8 -0
  232. package/dist/web/hooks/use-request-close.js.map +1 -0
  233. package/dist/web/hooks/use-request-close.test.d.ts +1 -0
  234. package/dist/web/hooks/use-request-close.test.js +52 -0
  235. package/dist/web/hooks/use-request-close.test.js.map +1 -0
  236. package/dist/web/hooks/use-request-modal.d.ts +9 -0
  237. package/dist/web/hooks/use-request-modal.js +16 -0
  238. package/dist/web/hooks/use-request-modal.js.map +1 -0
  239. package/dist/web/hooks/use-request-modal.test.d.ts +1 -0
  240. package/dist/{src/web → web}/hooks/use-request-modal.test.js +5 -1
  241. package/dist/web/hooks/use-request-modal.test.js.map +1 -0
  242. package/dist/web/hooks/use-request-size.d.ts +3 -0
  243. package/dist/web/hooks/use-request-size.js +8 -0
  244. package/dist/web/hooks/use-request-size.js.map +1 -0
  245. package/dist/web/hooks/use-request-size.test.d.ts +1 -0
  246. package/dist/web/hooks/use-request-size.test.js +65 -0
  247. package/dist/web/hooks/use-request-size.test.js.map +1 -0
  248. package/dist/web/hooks/use-send-follow-up-message.d.ts +2 -0
  249. package/dist/web/hooks/use-send-follow-up-message.js +8 -0
  250. package/dist/web/hooks/use-send-follow-up-message.js.map +1 -0
  251. package/dist/web/hooks/use-set-open-in-app-url.d.ts +1 -0
  252. package/dist/web/hooks/use-set-open-in-app-url.js +8 -0
  253. package/dist/web/hooks/use-set-open-in-app-url.js.map +1 -0
  254. package/dist/web/hooks/use-set-open-in-app-url.test.d.ts +1 -0
  255. package/dist/web/hooks/use-set-open-in-app-url.test.js +43 -0
  256. package/dist/web/hooks/use-set-open-in-app-url.test.js.map +1 -0
  257. package/dist/{src/web → web}/hooks/use-tool-info.js +4 -4
  258. package/dist/web/hooks/use-tool-info.js.map +1 -0
  259. package/dist/web/hooks/use-tool-info.test-d.d.ts +1 -0
  260. package/dist/web/hooks/use-tool-info.test-d.js.map +1 -0
  261. package/dist/web/hooks/use-tool-info.test.d.ts +1 -0
  262. package/dist/{src/web → web}/hooks/use-tool-info.test.js +5 -5
  263. package/dist/web/hooks/use-tool-info.test.js.map +1 -0
  264. package/dist/{src/web → web}/hooks/use-user.d.ts +1 -1
  265. package/dist/web/hooks/use-user.js +35 -0
  266. package/dist/web/hooks/use-user.js.map +1 -0
  267. package/dist/web/hooks/use-user.test.d.ts +1 -0
  268. package/dist/{src/web → web}/hooks/use-user.test.js +33 -4
  269. package/dist/web/hooks/use-user.test.js.map +1 -0
  270. package/dist/web/hooks/use-view-state.d.ts +4 -0
  271. package/dist/web/hooks/use-view-state.js +32 -0
  272. package/dist/web/hooks/use-view-state.js.map +1 -0
  273. package/dist/web/hooks/use-view-state.test.d.ts +1 -0
  274. package/dist/web/hooks/use-view-state.test.js +177 -0
  275. package/dist/web/hooks/use-view-state.test.js.map +1 -0
  276. package/dist/{src/web → web}/index.d.ts +1 -2
  277. package/dist/{src/web → web}/index.js +1 -2
  278. package/dist/web/index.js.map +1 -0
  279. package/dist/web/mount-view.d.ts +1 -0
  280. package/dist/{src/web/mount-widget.js → web/mount-view.js} +11 -3
  281. package/dist/web/mount-view.js.map +1 -0
  282. package/dist/web/plugin/data-llm.test.d.ts +1 -0
  283. package/dist/web/plugin/data-llm.test.js.map +1 -0
  284. package/dist/web/plugin/plugin.d.ts +5 -0
  285. package/dist/web/plugin/plugin.js +163 -0
  286. package/dist/web/plugin/plugin.js.map +1 -0
  287. package/dist/web/plugin/scan-views.d.ts +16 -0
  288. package/dist/web/plugin/scan-views.js +88 -0
  289. package/dist/web/plugin/scan-views.js.map +1 -0
  290. package/dist/web/plugin/scan-views.test.d.ts +1 -0
  291. package/dist/web/plugin/scan-views.test.js +99 -0
  292. package/dist/web/plugin/scan-views.test.js.map +1 -0
  293. package/dist/{src/web → web}/plugin/transform-data-llm.js +1 -1
  294. package/dist/web/plugin/transform-data-llm.js.map +1 -0
  295. package/dist/web/plugin/transform-data-llm.test.d.ts +1 -0
  296. package/dist/web/plugin/transform-data-llm.test.js.map +1 -0
  297. package/dist/web/plugin/validate-view.d.ts +1 -0
  298. package/dist/web/plugin/validate-view.js +9 -0
  299. package/dist/web/plugin/validate-view.js.map +1 -0
  300. package/dist/web/plugin/validate-view.test.d.ts +1 -0
  301. package/dist/web/plugin/validate-view.test.js +24 -0
  302. package/dist/web/plugin/validate-view.test.js.map +1 -0
  303. package/dist/{src/web → web}/proxy.js +0 -1
  304. package/dist/web/proxy.js.map +1 -0
  305. package/dist/web/types.d.ts +16 -0
  306. package/dist/web/types.js +2 -0
  307. package/dist/web/types.js.map +1 -0
  308. package/package.json +74 -39
  309. package/tsconfig.base.json +33 -0
  310. package/dist/src/server/devtoolsStaticServer.d.ts +0 -15
  311. package/dist/src/server/devtoolsStaticServer.js +0 -38
  312. package/dist/src/server/devtoolsStaticServer.js.map +0 -1
  313. package/dist/src/server/index.d.ts +0 -5
  314. package/dist/src/server/index.js +0 -4
  315. package/dist/src/server/index.js.map +0 -1
  316. package/dist/src/server/inferUtilityTypes.js.map +0 -1
  317. package/dist/src/server/server.d.ts +0 -74
  318. package/dist/src/server/server.js +0 -82
  319. package/dist/src/server/server.js.map +0 -1
  320. package/dist/src/server/templateHelper.js +0 -30
  321. package/dist/src/server/templateHelper.js.map +0 -1
  322. package/dist/src/server/templates/development.hbs +0 -13
  323. package/dist/src/server/templates/production.hbs +0 -7
  324. package/dist/src/server/widgetsDevServer.d.ts +0 -12
  325. package/dist/src/server/widgetsDevServer.js +0 -38
  326. package/dist/src/server/widgetsDevServer.js.map +0 -1
  327. package/dist/src/test/utils.js.map +0 -1
  328. package/dist/src/test/widget.test.js +0 -146
  329. package/dist/src/test/widget.test.js.map +0 -1
  330. package/dist/src/web/bridges/adaptors/apps-sdk-adaptor.d.ts +0 -13
  331. package/dist/src/web/bridges/adaptors/apps-sdk-adaptor.js +0 -33
  332. package/dist/src/web/bridges/adaptors/apps-sdk-adaptor.js.map +0 -1
  333. package/dist/src/web/bridges/adaptors/mcp-app-adaptor.d.ts +0 -16
  334. package/dist/src/web/bridges/adaptors/mcp-app-adaptor.js +0 -115
  335. package/dist/src/web/bridges/adaptors/mcp-app-adaptor.js.map +0 -1
  336. package/dist/src/web/bridges/apps-sdk-bridge.d.ts +0 -10
  337. package/dist/src/web/bridges/apps-sdk-bridge.js.map +0 -1
  338. package/dist/src/web/bridges/hooks/use-adaptor.d.ts +0 -2
  339. package/dist/src/web/bridges/hooks/use-adaptor.js +0 -8
  340. package/dist/src/web/bridges/hooks/use-adaptor.js.map +0 -1
  341. package/dist/src/web/bridges/hooks/use-apps-sdk-bridge.d.ts +0 -2
  342. package/dist/src/web/bridges/hooks/use-apps-sdk-bridge.js.map +0 -1
  343. package/dist/src/web/bridges/hooks/use-bridge.d.ts +0 -2
  344. package/dist/src/web/bridges/hooks/use-bridge.js +0 -8
  345. package/dist/src/web/bridges/hooks/use-bridge.js.map +0 -1
  346. package/dist/src/web/bridges/hooks/use-mcp-app-bridge.d.ts +0 -5
  347. package/dist/src/web/bridges/hooks/use-mcp-app-bridge.js +0 -7
  348. package/dist/src/web/bridges/hooks/use-mcp-app-bridge.js.map +0 -1
  349. package/dist/src/web/bridges/hooks/use-mcp-app-bridge.test.js +0 -41
  350. package/dist/src/web/bridges/hooks/use-mcp-app-bridge.test.js.map +0 -1
  351. package/dist/src/web/bridges/index.d.ts +0 -4
  352. package/dist/src/web/bridges/index.js +0 -5
  353. package/dist/src/web/bridges/index.js.map +0 -1
  354. package/dist/src/web/bridges/mcp-app-bridge.d.ts +0 -38
  355. package/dist/src/web/bridges/mcp-app-bridge.js +0 -162
  356. package/dist/src/web/bridges/mcp-app-bridge.js.map +0 -1
  357. package/dist/src/web/bridges/types.d.ts +0 -57
  358. package/dist/src/web/bridges/types.js.map +0 -1
  359. package/dist/src/web/create-store.js +0 -25
  360. package/dist/src/web/create-store.js.map +0 -1
  361. package/dist/src/web/create-store.test.js +0 -70
  362. package/dist/src/web/create-store.test.js.map +0 -1
  363. package/dist/src/web/data-llm.js.map +0 -1
  364. package/dist/src/web/data-llm.test.js +0 -76
  365. package/dist/src/web/data-llm.test.js.map +0 -1
  366. package/dist/src/web/generate-helpers.js.map +0 -1
  367. package/dist/src/web/generate-helpers.test-d.js.map +0 -1
  368. package/dist/src/web/generate-helpers.test.js.map +0 -1
  369. package/dist/src/web/helpers/state.js +0 -40
  370. package/dist/src/web/helpers/state.js.map +0 -1
  371. package/dist/src/web/helpers/state.test.js.map +0 -1
  372. package/dist/src/web/hooks/index.js.map +0 -1
  373. package/dist/src/web/hooks/test/utils.js +0 -40
  374. package/dist/src/web/hooks/test/utils.js.map +0 -1
  375. package/dist/src/web/hooks/use-call-tool.js.map +0 -1
  376. package/dist/src/web/hooks/use-call-tool.test-d.js.map +0 -1
  377. package/dist/src/web/hooks/use-call-tool.test.js.map +0 -1
  378. package/dist/src/web/hooks/use-display-mode.d.ts +0 -4
  379. package/dist/src/web/hooks/use-display-mode.js +0 -10
  380. package/dist/src/web/hooks/use-display-mode.js.map +0 -1
  381. package/dist/src/web/hooks/use-display-mode.test.js.map +0 -1
  382. package/dist/src/web/hooks/use-files.d.ts +0 -10
  383. package/dist/src/web/hooks/use-files.js +0 -7
  384. package/dist/src/web/hooks/use-files.js.map +0 -1
  385. package/dist/src/web/hooks/use-files.test.js +0 -29
  386. package/dist/src/web/hooks/use-files.test.js.map +0 -1
  387. package/dist/src/web/hooks/use-layout.js.map +0 -1
  388. package/dist/src/web/hooks/use-layout.test.js.map +0 -1
  389. package/dist/src/web/hooks/use-open-external.d.ts +0 -1
  390. package/dist/src/web/hooks/use-open-external.js +0 -8
  391. package/dist/src/web/hooks/use-open-external.js.map +0 -1
  392. package/dist/src/web/hooks/use-open-external.test.js.map +0 -1
  393. package/dist/src/web/hooks/use-openai-global.d.ts +0 -3
  394. package/dist/src/web/hooks/use-openai-global.js +0 -6
  395. package/dist/src/web/hooks/use-openai-global.js.map +0 -1
  396. package/dist/src/web/hooks/use-request-modal.d.ts +0 -9
  397. package/dist/src/web/hooks/use-request-modal.js +0 -14
  398. package/dist/src/web/hooks/use-request-modal.js.map +0 -1
  399. package/dist/src/web/hooks/use-request-modal.test.js.map +0 -1
  400. package/dist/src/web/hooks/use-send-follow-up-message.d.ts +0 -1
  401. package/dist/src/web/hooks/use-send-follow-up-message.js +0 -8
  402. package/dist/src/web/hooks/use-send-follow-up-message.js.map +0 -1
  403. package/dist/src/web/hooks/use-tool-info.js.map +0 -1
  404. package/dist/src/web/hooks/use-tool-info.test-d.js.map +0 -1
  405. package/dist/src/web/hooks/use-tool-info.test.js.map +0 -1
  406. package/dist/src/web/hooks/use-user.js +0 -19
  407. package/dist/src/web/hooks/use-user.js.map +0 -1
  408. package/dist/src/web/hooks/use-user.test.js.map +0 -1
  409. package/dist/src/web/hooks/use-widget-state.d.ts +0 -4
  410. package/dist/src/web/hooks/use-widget-state.js +0 -32
  411. package/dist/src/web/hooks/use-widget-state.js.map +0 -1
  412. package/dist/src/web/hooks/use-widget-state.test.js +0 -61
  413. package/dist/src/web/hooks/use-widget-state.test.js.map +0 -1
  414. package/dist/src/web/index.js.map +0 -1
  415. package/dist/src/web/mount-widget.d.ts +0 -1
  416. package/dist/src/web/mount-widget.js.map +0 -1
  417. package/dist/src/web/plugin/data-llm.test.js.map +0 -1
  418. package/dist/src/web/plugin/plugin.d.ts +0 -2
  419. package/dist/src/web/plugin/plugin.js +0 -63
  420. package/dist/src/web/plugin/plugin.js.map +0 -1
  421. package/dist/src/web/plugin/transform-data-llm.js.map +0 -1
  422. package/dist/src/web/plugin/transform-data-llm.test.js.map +0 -1
  423. package/dist/src/web/proxy.js.map +0 -1
  424. package/dist/src/web/types.d.ts +0 -149
  425. package/dist/src/web/types.js.map +0 -1
  426. package/dist/vitest.config.d.ts +0 -2
  427. package/dist/vitest.config.js +0 -8
  428. package/dist/vitest.config.js.map +0 -1
  429. /package/dist/{src/test/widget.test.d.ts → cli/tunnel-control-server.test.d.ts} +0 -0
  430. /package/dist/{src/web/bridges/hooks/use-mcp-app-bridge.test.d.ts → cli/tunnel-handler.test.d.ts} +0 -0
  431. /package/dist/{src/web/create-store.test.d.ts → cli/tunnel.test.d.ts} +0 -0
  432. /package/dist/{src/web/bridges → cli}/types.js +0 -0
  433. /package/dist/{src/web/data-llm.test.d.ts → server/asset-base-url-transform-plugin.test.d.ts} +0 -0
  434. /package/dist/{src/web/generate-helpers.test-d.d.ts → server/content-helpers.test.d.ts} +0 -0
  435. /package/dist/{src/web/generate-helpers.test.d.ts → server/express.test.d.ts} +0 -0
  436. /package/dist/{src/server → server}/inferUtilityTypes.js +0 -0
  437. /package/dist/{src/web/helpers/state.test.d.ts → server/middleware.test-d.d.ts} +0 -0
  438. /package/dist/{src/web/hooks/use-call-tool.test-d.d.ts → server/middleware.test.d.ts} +0 -0
  439. /package/dist/{src/web/hooks/use-call-tool.test.d.ts → server/tunnel-proxy-router.test.d.ts} +0 -0
  440. /package/dist/{src/web/hooks/use-display-mode.test.d.ts → test/view.test.d.ts} +0 -0
  441. /package/dist/{src/web/hooks/use-files.test.d.ts → web/bridges/mcp-app/use-mcp-app-context.test.d.ts} +0 -0
  442. /package/dist/{src/web → web}/create-store.d.ts +0 -0
  443. /package/dist/{src/web/hooks/use-layout.test.d.ts → web/create-store.test.d.ts} +0 -0
  444. /package/dist/{src/web/plugin → web}/data-llm.test.d.ts +0 -0
  445. /package/dist/{src/web/hooks/use-open-external.test.d.ts → web/generate-helpers.test-d.d.ts} +0 -0
  446. /package/dist/{src/web/hooks/use-request-modal.test.d.ts → web/generate-helpers.test.d.ts} +0 -0
  447. /package/dist/{src/web → web}/generate-helpers.test.js +0 -0
  448. /package/dist/{src/web/hooks/use-tool-info.test-d.d.ts → web/helpers/state.test.d.ts} +0 -0
  449. /package/dist/{src/web/hooks/use-tool-info.test.d.ts → web/hooks/use-call-tool.test-d.d.ts} +0 -0
  450. /package/dist/{src/web → web}/hooks/use-call-tool.test-d.js +0 -0
  451. /package/dist/{src/web/hooks/use-user.test.d.ts → web/hooks/use-call-tool.test.d.ts} +0 -0
  452. /package/dist/{src/web/hooks/use-widget-state.test.d.ts → web/hooks/use-display-mode.test-d.d.ts} +0 -0
  453. /package/dist/{src/web/plugin/transform-data-llm.test.d.ts → web/hooks/use-display-mode.test.d.ts} +0 -0
  454. /package/dist/{src/web → web}/hooks/use-display-mode.test.js +0 -0
  455. /package/dist/{src/web → web}/hooks/use-tool-info.d.ts +0 -0
  456. /package/dist/{src/web → web}/hooks/use-tool-info.test-d.js +0 -0
  457. /package/dist/{src/web → web}/plugin/data-llm.test.js +0 -0
  458. /package/dist/{src/web → web}/plugin/transform-data-llm.d.ts +0 -0
  459. /package/dist/{src/web → web}/plugin/transform-data-llm.test.js +0 -0
  460. /package/dist/{src/web → web}/proxy.d.ts +0 -0
@@ -0,0 +1,117 @@
1
+ import type { CallToolResult } from "@modelcontextprotocol/sdk/types.js";
2
+ import type { useSyncExternalStore } from "react";
3
+ import type { ViewHostType } from "../../server/index.js";
4
+ export type SkybridgeProperties = {
5
+ hostType: ViewHostType;
6
+ serverUrl: string;
7
+ };
8
+ declare global {
9
+ interface Window {
10
+ skybridge: SkybridgeProperties;
11
+ }
12
+ }
13
+ export type CallToolArgs = Record<string, unknown> | null;
14
+ export type CallToolResponse = {
15
+ content: CallToolResult["content"];
16
+ structuredContent: NonNullable<CallToolResult["structuredContent"]>;
17
+ isError: NonNullable<CallToolResult["isError"]>;
18
+ meta?: CallToolResult["_meta"];
19
+ };
20
+ export type DisplayMode = "pip" | "inline" | "fullscreen" | "modal";
21
+ export type RequestDisplayMode = Exclude<DisplayMode, "modal">;
22
+ export type Theme = "light" | "dark";
23
+ export type DeviceType = "mobile" | "tablet" | "desktop" | "unknown";
24
+ export type SafeAreaInsets = {
25
+ top: number;
26
+ right: number;
27
+ bottom: number;
28
+ left: number;
29
+ };
30
+ export type SafeArea = {
31
+ insets: SafeAreaInsets;
32
+ };
33
+ export type UserAgent = {
34
+ device: {
35
+ type: DeviceType;
36
+ };
37
+ capabilities: {
38
+ hover: boolean;
39
+ touch: boolean;
40
+ };
41
+ };
42
+ export interface HostContext {
43
+ theme: Theme;
44
+ locale: string;
45
+ displayMode: DisplayMode;
46
+ safeArea: SafeArea;
47
+ maxHeight: number | undefined;
48
+ userAgent: UserAgent;
49
+ toolInput: Record<string, unknown> | null;
50
+ toolOutput: Record<string, unknown> | null;
51
+ toolResponseMetadata: Record<string, unknown> | null;
52
+ display: {
53
+ mode: DisplayMode;
54
+ params?: Record<string, unknown>;
55
+ };
56
+ viewState: Record<string, unknown> | null;
57
+ }
58
+ export type Subscribe = Parameters<typeof useSyncExternalStore>[0];
59
+ export interface Bridge<Context> {
60
+ subscribe(key: keyof Context): Subscribe;
61
+ subscribe(keys: readonly (keyof Context)[]): Subscribe;
62
+ getSnapshot<K extends keyof Context>(key: K): Context[K] | undefined;
63
+ }
64
+ export type HostContextStore<K extends keyof HostContext> = {
65
+ subscribe: Subscribe;
66
+ getSnapshot: () => HostContext[K];
67
+ };
68
+ export type ViewState = Record<string, unknown>;
69
+ export type SetViewStateAction = ViewState | ((prevState: ViewState | null) => ViewState);
70
+ export type FileMetadata = {
71
+ fileId: string;
72
+ fileName?: string;
73
+ mimeType?: string;
74
+ };
75
+ export type UploadFileOptions = {
76
+ library?: boolean;
77
+ };
78
+ export type RequestModalOptions = {
79
+ title?: string;
80
+ params?: Record<string, unknown>;
81
+ template?: string;
82
+ anchor?: {
83
+ top?: number;
84
+ left?: number;
85
+ width?: number;
86
+ height?: number;
87
+ };
88
+ };
89
+ export type OpenExternalOptions = {
90
+ redirectUrl?: false;
91
+ };
92
+ export type SendFollowUpMessageOptions = {
93
+ scrollToBottom?: boolean;
94
+ };
95
+ export type RequestSizeOptions = {
96
+ width?: number;
97
+ height?: number;
98
+ };
99
+ export interface Adaptor {
100
+ getHostContextStore<K extends keyof HostContext>(key: K): HostContextStore<K>;
101
+ callTool<ToolArgs extends CallToolArgs = null, ToolResponse extends CallToolResponse = CallToolResponse>(name: string, args: ToolArgs): Promise<ToolResponse>;
102
+ requestDisplayMode(mode: RequestDisplayMode): Promise<{
103
+ mode: RequestDisplayMode;
104
+ }>;
105
+ requestClose(): Promise<void>;
106
+ requestSize(size: RequestSizeOptions): Promise<void>;
107
+ sendFollowUpMessage(prompt: string, options?: SendFollowUpMessageOptions): Promise<void>;
108
+ openExternal(href: string, options?: OpenExternalOptions): void;
109
+ setViewState(stateOrUpdater: SetViewStateAction): Promise<void>;
110
+ uploadFile(file: File, options?: UploadFileOptions): Promise<FileMetadata>;
111
+ getFileDownloadUrl(file: FileMetadata): Promise<{
112
+ downloadUrl: string;
113
+ }>;
114
+ selectFiles(): Promise<FileMetadata[]>;
115
+ openModal(options: RequestModalOptions): void;
116
+ setOpenInAppUrl(href: string): Promise<void>;
117
+ }
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/web/bridges/types.ts"],"names":[],"mappings":"","sourcesContent":["import type { CallToolResult } from \"@modelcontextprotocol/sdk/types.js\";\nimport type { useSyncExternalStore } from \"react\";\nimport type { ViewHostType } from \"../../server/index.js\";\n\nexport type SkybridgeProperties = {\n hostType: ViewHostType;\n serverUrl: string;\n};\n\ndeclare global {\n interface Window {\n skybridge: SkybridgeProperties;\n }\n}\n\nexport type CallToolArgs = Record<string, unknown> | null;\n\nexport type CallToolResponse = {\n content: CallToolResult[\"content\"];\n structuredContent: NonNullable<CallToolResult[\"structuredContent\"]>;\n isError: NonNullable<CallToolResult[\"isError\"]>;\n meta?: CallToolResult[\"_meta\"];\n};\n\nexport type DisplayMode = \"pip\" | \"inline\" | \"fullscreen\" | \"modal\";\nexport type RequestDisplayMode = Exclude<DisplayMode, \"modal\">;\n\nexport type Theme = \"light\" | \"dark\";\n\nexport type DeviceType = \"mobile\" | \"tablet\" | \"desktop\" | \"unknown\";\n\nexport type SafeAreaInsets = {\n top: number;\n right: number;\n bottom: number;\n left: number;\n};\n\nexport type SafeArea = {\n insets: SafeAreaInsets;\n};\n\nexport type UserAgent = {\n device: {\n type: DeviceType;\n };\n capabilities: {\n hover: boolean;\n touch: boolean;\n };\n};\n\nexport interface HostContext {\n theme: Theme;\n locale: string;\n displayMode: DisplayMode;\n safeArea: SafeArea;\n maxHeight: number | undefined;\n userAgent: UserAgent;\n toolInput: Record<string, unknown> | null;\n toolOutput: Record<string, unknown> | null;\n toolResponseMetadata: Record<string, unknown> | null;\n display: {\n mode: DisplayMode;\n params?: Record<string, unknown>;\n };\n viewState: Record<string, unknown> | null;\n}\n\nexport type Subscribe = Parameters<typeof useSyncExternalStore>[0];\n\nexport interface Bridge<Context> {\n subscribe(key: keyof Context): Subscribe;\n subscribe(keys: readonly (keyof Context)[]): Subscribe;\n getSnapshot<K extends keyof Context>(key: K): Context[K] | undefined;\n}\n\nexport type HostContextStore<K extends keyof HostContext> = {\n subscribe: Subscribe;\n getSnapshot: () => HostContext[K];\n};\n\nexport type ViewState = Record<string, unknown>;\n\nexport type SetViewStateAction =\n | ViewState\n | ((prevState: ViewState | null) => ViewState);\n\nexport type FileMetadata = {\n fileId: string;\n fileName?: string;\n mimeType?: string;\n};\n\nexport type UploadFileOptions = { library?: boolean };\n\nexport type RequestModalOptions = {\n title?: string;\n params?: Record<string, unknown>;\n template?: string;\n anchor?: { top?: number; left?: number; width?: number; height?: number };\n};\n\nexport type OpenExternalOptions = {\n redirectUrl?: false;\n};\n\nexport type SendFollowUpMessageOptions = { scrollToBottom?: boolean };\n\nexport type RequestSizeOptions = {\n width?: number;\n height?: number;\n};\n\nexport interface Adaptor {\n getHostContextStore<K extends keyof HostContext>(key: K): HostContextStore<K>;\n callTool<\n ToolArgs extends CallToolArgs = null,\n ToolResponse extends CallToolResponse = CallToolResponse,\n >(name: string, args: ToolArgs): Promise<ToolResponse>;\n requestDisplayMode(mode: RequestDisplayMode): Promise<{\n mode: RequestDisplayMode;\n }>;\n requestClose(): Promise<void>;\n requestSize(size: RequestSizeOptions): Promise<void>;\n sendFollowUpMessage(\n prompt: string,\n options?: SendFollowUpMessageOptions,\n ): Promise<void>;\n openExternal(href: string, options?: OpenExternalOptions): void;\n setViewState(stateOrUpdater: SetViewStateAction): Promise<void>;\n uploadFile(file: File, options?: UploadFileOptions): Promise<FileMetadata>;\n getFileDownloadUrl(file: FileMetadata): Promise<{ downloadUrl: string }>;\n selectFiles(): Promise<FileMetadata[]>;\n openModal(options: RequestModalOptions): void;\n setOpenInAppUrl(href: string): Promise<void>;\n}\n"]}
@@ -0,0 +1,2 @@
1
+ import type { HostContext } from "./types.js";
2
+ export declare const useHostContext: <K extends keyof HostContext>(key: K) => HostContext[K];
@@ -0,0 +1,8 @@
1
+ import { useSyncExternalStore } from "react";
2
+ import { getAdaptor } from "./get-adaptor.js";
3
+ export const useHostContext = (key) => {
4
+ const adaptor = getAdaptor();
5
+ const store = adaptor.getHostContextStore(key);
6
+ return useSyncExternalStore(store.subscribe, store.getSnapshot);
7
+ };
8
+ //# sourceMappingURL=use-host-context.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-host-context.js","sourceRoot":"","sources":["../../../src/web/bridges/use-host-context.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,oBAAoB,EAAE,MAAM,OAAO,CAAC;AAC7C,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAG9C,MAAM,CAAC,MAAM,cAAc,GAAG,CAC5B,GAAM,EACU,EAAE;IAClB,MAAM,OAAO,GAAG,UAAU,EAAE,CAAC;IAC7B,MAAM,KAAK,GAAG,OAAO,CAAC,mBAAmB,CAAC,GAAG,CAAC,CAAC;IAE/C,OAAO,oBAAoB,CAAC,KAAK,CAAC,SAAS,EAAE,KAAK,CAAC,WAAW,CAAC,CAAC;AAClE,CAAC,CAAC","sourcesContent":["import { useSyncExternalStore } from \"react\";\nimport { getAdaptor } from \"./get-adaptor.js\";\nimport type { HostContext } from \"./types.js\";\n\nexport const useHostContext = <K extends keyof HostContext>(\n key: K,\n): HostContext[K] => {\n const adaptor = getAdaptor();\n const store = adaptor.getHostContextStore(key);\n\n return useSyncExternalStore(store.subscribe, store.getSnapshot);\n};\n"]}
@@ -0,0 +1,4 @@
1
+ import { type ReactNode } from "react";
2
+ export declare function ModalProvider({ children }: {
3
+ children: ReactNode;
4
+ }): import("react/jsx-runtime").JSX.Element;
@@ -0,0 +1,45 @@
1
+ import { jsx as _jsx, Fragment as _Fragment, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import { useEffect, useSyncExternalStore } from "react";
3
+ import { McpAppAdaptor } from "../bridges/index.js";
4
+ const modalStyles = `
5
+ .sb-modal-backdrop {
6
+ position: fixed;
7
+ inset: 0;
8
+ background: rgba(0, 0, 0, 0.5);
9
+ z-index: 9998;
10
+ }
11
+ .sb-modal-container {
12
+ border-radius: 12px;
13
+ position: fixed;
14
+ inset: 0;
15
+ margin: auto;
16
+ width: fit-content;
17
+ height: fit-content;
18
+ background: white;
19
+ z-index: 9999;
20
+ }
21
+ `;
22
+ export function ModalProvider({ children }) {
23
+ const adaptor = McpAppAdaptor.getInstance();
24
+ const { mode } = useSyncExternalStore(adaptor.getHostContextStore("display").subscribe, adaptor.getHostContextStore("display").getSnapshot);
25
+ const isOpen = mode === "modal";
26
+ const handleBackdropClick = (e) => {
27
+ if (e.target === e.currentTarget) {
28
+ adaptor.closeModal();
29
+ }
30
+ };
31
+ useEffect(() => {
32
+ if (!isOpen) {
33
+ return;
34
+ }
35
+ const handler = (e) => {
36
+ if (e.key === "Escape") {
37
+ adaptor.closeModal();
38
+ }
39
+ };
40
+ document.addEventListener("keydown", handler);
41
+ return () => document.removeEventListener("keydown", handler);
42
+ }, [isOpen, adaptor]);
43
+ return (_jsxs(_Fragment, { children: [_jsx("style", { children: modalStyles }), isOpen && (_jsx("div", { role: "dialog", className: "sb-modal-backdrop", onClick: handleBackdropClick })), _jsx("div", { className: isOpen ? "sb-modal-container" : undefined, children: children })] }));
44
+ }
45
+ //# sourceMappingURL=modal-provider.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"modal-provider.js","sourceRoot":"","sources":["../../../src/web/components/modal-provider.tsx"],"names":[],"mappings":";AAAA,OAAO,EAAkB,SAAS,EAAE,oBAAoB,EAAE,MAAM,OAAO,CAAC;AACxE,OAAO,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AAEpD,MAAM,WAAW,GAAG;;;;;;;;;;;;;;;;;CAiBnB,CAAC;AAEF,MAAM,UAAU,aAAa,CAAC,EAAE,QAAQ,EAA2B;IACjE,MAAM,OAAO,GAAG,aAAa,CAAC,WAAW,EAAE,CAAC;IAE5C,MAAM,EAAE,IAAI,EAAE,GAAG,oBAAoB,CACnC,OAAO,CAAC,mBAAmB,CAAC,SAAS,CAAC,CAAC,SAAS,EAChD,OAAO,CAAC,mBAAmB,CAAC,SAAS,CAAC,CAAC,WAAW,CACnD,CAAC;IACF,MAAM,MAAM,GAAG,IAAI,KAAK,OAAO,CAAC;IAEhC,MAAM,mBAAmB,GAAG,CAAC,CAAmB,EAAE,EAAE;QAClD,IAAI,CAAC,CAAC,MAAM,KAAK,CAAC,CAAC,aAAa,EAAE,CAAC;YACjC,OAAO,CAAC,UAAU,EAAE,CAAC;QACvB,CAAC;IACH,CAAC,CAAC;IAEF,SAAS,CAAC,GAAG,EAAE;QACb,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,OAAO;QACT,CAAC;QACD,MAAM,OAAO,GAAG,CAAC,CAAgB,EAAE,EAAE;YACnC,IAAI,CAAC,CAAC,GAAG,KAAK,QAAQ,EAAE,CAAC;gBACvB,OAAO,CAAC,UAAU,EAAE,CAAC;YACvB,CAAC;QACH,CAAC,CAAC;QACF,QAAQ,CAAC,gBAAgB,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;QAC9C,OAAO,GAAG,EAAE,CAAC,QAAQ,CAAC,mBAAmB,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;IAChE,CAAC,EAAE,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,CAAC;IAEtB,OAAO,CACL,8BACE,0BAAQ,WAAW,GAAS,EAC3B,MAAM,IAAI,CAET,cACE,IAAI,EAAC,QAAQ,EACb,SAAS,EAAC,mBAAmB,EAC7B,OAAO,EAAE,mBAAmB,GAC5B,CACH,EACD,cAAK,SAAS,EAAE,MAAM,CAAC,CAAC,CAAC,oBAAoB,CAAC,CAAC,CAAC,SAAS,YACtD,QAAQ,GACL,IACL,CACJ,CAAC;AACJ,CAAC","sourcesContent":["import { type ReactNode, useEffect, useSyncExternalStore } from \"react\";\nimport { McpAppAdaptor } from \"../bridges/index.js\";\n\nconst modalStyles = `\n.sb-modal-backdrop {\n position: fixed;\n inset: 0;\n background: rgba(0, 0, 0, 0.5);\n z-index: 9998;\n}\n.sb-modal-container {\n border-radius: 12px;\n position: fixed;\n inset: 0;\n margin: auto;\n width: fit-content;\n height: fit-content;\n background: white;\n z-index: 9999;\n}\n`;\n\nexport function ModalProvider({ children }: { children: ReactNode }) {\n const adaptor = McpAppAdaptor.getInstance();\n\n const { mode } = useSyncExternalStore(\n adaptor.getHostContextStore(\"display\").subscribe,\n adaptor.getHostContextStore(\"display\").getSnapshot,\n );\n const isOpen = mode === \"modal\";\n\n const handleBackdropClick = (e: React.MouseEvent) => {\n if (e.target === e.currentTarget) {\n adaptor.closeModal();\n }\n };\n\n useEffect(() => {\n if (!isOpen) {\n return;\n }\n const handler = (e: KeyboardEvent) => {\n if (e.key === \"Escape\") {\n adaptor.closeModal();\n }\n };\n document.addEventListener(\"keydown\", handler);\n return () => document.removeEventListener(\"keydown\", handler);\n }, [isOpen, adaptor]);\n\n return (\n <>\n <style>{modalStyles}</style>\n {isOpen && (\n // biome-ignore lint/a11y/useKeyWithClickEvents: backdrop isn't focusable\n <div\n role=\"dialog\"\n className=\"sb-modal-backdrop\"\n onClick={handleBackdropClick}\n />\n )}\n <div className={isOpen ? \"sb-modal-container\" : undefined}>\n {children}\n </div>\n </>\n );\n}\n"]}
@@ -0,0 +1,38 @@
1
+ import { dequal } from "dequal/lite";
2
+ import { create } from "zustand";
3
+ import { getAdaptor } from "./bridges/index.js";
4
+ import { filterViewContext, getInitialState, injectViewContext, serializeState, } from "./helpers/state.js";
5
+ export function createStore(storeCreator, defaultState) {
6
+ const initialState = getInitialState(defaultState);
7
+ const store = create()((...args) => {
8
+ const baseStore = storeCreator(...args);
9
+ if (initialState !== null) {
10
+ return { ...baseStore, ...initialState };
11
+ }
12
+ return baseStore;
13
+ });
14
+ // Bidirectional sync between the Zustand store and the adaptor's viewState.
15
+ // Store changes persist to the host; external viewState changes rehydrate the store.
16
+ store.subscribe((state) => {
17
+ const serializedState = serializeState(state);
18
+ if (serializedState !== null && serializedState !== undefined) {
19
+ const stateToPersist = injectViewContext(serializedState);
20
+ if (stateToPersist !== null) {
21
+ getAdaptor().setViewState(stateToPersist);
22
+ }
23
+ }
24
+ });
25
+ const viewStateStore = getAdaptor().getHostContextStore("viewState");
26
+ viewStateStore.subscribe(() => {
27
+ const externalState = viewStateStore.getSnapshot();
28
+ if (externalState !== null) {
29
+ const filtered = filterViewContext(externalState);
30
+ const current = serializeState(store.getState());
31
+ if (!dequal(filtered, current)) {
32
+ store.setState(filtered);
33
+ }
34
+ }
35
+ });
36
+ return store;
37
+ }
38
+ //# sourceMappingURL=create-store.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create-store.js","sourceRoot":"","sources":["../../src/web/create-store.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AACrC,OAAO,EAAE,MAAM,EAAqB,MAAM,SAAS,CAAC;AACpD,OAAO,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAChD,OAAO,EACL,iBAAiB,EACjB,eAAe,EACf,iBAAiB,EACjB,cAAc,GACf,MAAM,oBAAoB,CAAC;AAG5B,MAAM,UAAU,WAAW,CACzB,YAAgD,EAChD,YAAoC;IAEpC,MAAM,YAAY,GAAG,eAAe,CAAC,YAAY,CAAC,CAAC;IAEnD,MAAM,KAAK,GAAG,MAAM,EAAS,CAC3B,CAAC,GAAG,IAAoD,EAAE,EAAE;QAC1D,MAAM,SAAS,GAAG,YAAY,CAAC,GAAG,IAAI,CAAC,CAAC;QAExC,IAAI,YAAY,KAAK,IAAI,EAAE,CAAC;YAC1B,OAAO,EAAE,GAAG,SAAS,EAAE,GAAG,YAAY,EAAE,CAAC;QAC3C,CAAC;QAED,OAAO,SAAS,CAAC;IACnB,CAAC,CACF,CAAC;IAEF,4EAA4E;IAC5E,qFAAqF;IACrF,KAAK,CAAC,SAAS,CAAC,CAAC,KAAY,EAAE,EAAE;QAC/B,MAAM,eAAe,GAAG,cAAc,CAAC,KAAK,CAAC,CAAC;QAC9C,IAAI,eAAe,KAAK,IAAI,IAAI,eAAe,KAAK,SAAS,EAAE,CAAC;YAC9D,MAAM,cAAc,GAAG,iBAAiB,CAAC,eAAwB,CAAC,CAAC;YACnE,IAAI,cAAc,KAAK,IAAI,EAAE,CAAC;gBAC5B,UAAU,EAAE,CAAC,YAAY,CAAC,cAAc,CAAC,CAAC;YAC5C,CAAC;QACH,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,MAAM,cAAc,GAAG,UAAU,EAAE,CAAC,mBAAmB,CAAC,WAAW,CAAC,CAAC;IACrE,cAAc,CAAC,SAAS,CAAC,GAAG,EAAE;QAC5B,MAAM,aAAa,GAAG,cAAc,CAAC,WAAW,EAAE,CAAC;QACnD,IAAI,aAAa,KAAK,IAAI,EAAE,CAAC;YAC3B,MAAM,QAAQ,GAAG,iBAAiB,CAAC,aAAa,CAAU,CAAC;YAC3D,MAAM,OAAO,GAAG,cAAc,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAU,CAAC;YAC1D,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,OAAO,CAAC,EAAE,CAAC;gBAC/B,KAAK,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;YAC3B,CAAC;QACH,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,OAAO,KAAK,CAAC;AACf,CAAC","sourcesContent":["import { dequal } from \"dequal/lite\";\nimport { create, type StateCreator } from \"zustand\";\nimport { getAdaptor } from \"./bridges/index.js\";\nimport {\n filterViewContext,\n getInitialState,\n injectViewContext,\n serializeState,\n} from \"./helpers/state.js\";\nimport type { UnknownObject } from \"./types.js\";\n\nexport function createStore<State extends UnknownObject>(\n storeCreator: StateCreator<State, [], [], State>,\n defaultState?: State | (() => State),\n) {\n const initialState = getInitialState(defaultState);\n\n const store = create<State>()(\n (...args: Parameters<StateCreator<State, [], [], State>>) => {\n const baseStore = storeCreator(...args);\n\n if (initialState !== null) {\n return { ...baseStore, ...initialState };\n }\n\n return baseStore;\n },\n );\n\n // Bidirectional sync between the Zustand store and the adaptor's viewState.\n // Store changes persist to the host; external viewState changes rehydrate the store.\n store.subscribe((state: State) => {\n const serializedState = serializeState(state);\n if (serializedState !== null && serializedState !== undefined) {\n const stateToPersist = injectViewContext(serializedState as State);\n if (stateToPersist !== null) {\n getAdaptor().setViewState(stateToPersist);\n }\n }\n });\n\n const viewStateStore = getAdaptor().getHostContextStore(\"viewState\");\n viewStateStore.subscribe(() => {\n const externalState = viewStateStore.getSnapshot();\n if (externalState !== null) {\n const filtered = filterViewContext(externalState) as State;\n const current = serializeState(store.getState()) as State;\n if (!dequal(filtered, current)) {\n store.setState(filtered);\n }\n }\n });\n\n return store;\n}\n"]}
@@ -0,0 +1,129 @@
1
+ import { cleanup } from "@testing-library/react";
2
+ import { afterEach, beforeEach, describe, expect, it, vi, } from "vitest";
3
+ import { McpAppAdaptor } from "./bridges/mcp-app/adaptor.js";
4
+ import { McpAppBridge } from "./bridges/mcp-app/bridge.js";
5
+ import { createStore } from "./create-store.js";
6
+ import { VIEW_CONTEXT_KEY } from "./data-llm.js";
7
+ import { getMcpAppHostPostMessageMock, MockResizeObserver, } from "./hooks/test/utils.js";
8
+ describe("createStore", () => {
9
+ afterEach(() => {
10
+ vi.unstubAllGlobals();
11
+ vi.resetAllMocks();
12
+ });
13
+ describe("apps-sdk mode", () => {
14
+ let OpenaiMock;
15
+ beforeEach(() => {
16
+ OpenaiMock = {
17
+ widgetState: null,
18
+ setWidgetState: vi.fn().mockResolvedValue(undefined),
19
+ };
20
+ vi.stubGlobal("openai", OpenaiMock);
21
+ vi.stubGlobal("skybridge", { hostType: "apps-sdk" });
22
+ });
23
+ it("should create a store without default state", () => {
24
+ const storeCreator = () => ({
25
+ count: 0,
26
+ });
27
+ const store = createStore(storeCreator);
28
+ expect(store.getState()).toEqual({ count: 0 });
29
+ });
30
+ it("should create a store with default state", () => {
31
+ const storeCreator = () => ({
32
+ count: 0,
33
+ name: "initial",
34
+ });
35
+ const defaultState = { count: 5, name: "default" };
36
+ const store = createStore(storeCreator, defaultState);
37
+ expect(store.getState()).toEqual({ count: 5, name: "default" });
38
+ });
39
+ it("should initialize from window.openai.widgetState when available", () => {
40
+ const storeCreator = () => ({
41
+ count: 0,
42
+ name: "initial",
43
+ });
44
+ const windowState = { count: 20, name: "window" };
45
+ OpenaiMock.widgetState = { modelContent: windowState };
46
+ const store = createStore(storeCreator);
47
+ expect(store.getState()).toEqual({ count: 20, name: "window" });
48
+ });
49
+ it("should persist state changes to window.openai.setWidgetState", async () => {
50
+ const storeCreator = (set) => ({
51
+ count: 0,
52
+ increment: () => set((state) => ({ count: state.count + 1 })),
53
+ });
54
+ const store = createStore(storeCreator);
55
+ store.getState().increment();
56
+ await vi.waitFor(() => {
57
+ expect(OpenaiMock.setWidgetState).toHaveBeenCalled();
58
+ });
59
+ const callArgs = OpenaiMock.setWidgetState.mock.calls[0]?.[0];
60
+ expect(callArgs).toEqual({
61
+ modelContent: { count: 1 },
62
+ privateContent: {},
63
+ });
64
+ });
65
+ it("should filter view context from initial state", () => {
66
+ const storeCreator = () => ({
67
+ count: 0,
68
+ });
69
+ const windowState = {
70
+ count: 5,
71
+ [VIEW_CONTEXT_KEY]: "context-value",
72
+ };
73
+ OpenaiMock.widgetState = { modelContent: windowState };
74
+ const store = createStore(storeCreator);
75
+ expect(store.getState()).toEqual({ count: 5 });
76
+ expect(store.getState()[VIEW_CONTEXT_KEY]).toBeUndefined();
77
+ });
78
+ });
79
+ describe("mcp-app mode", () => {
80
+ beforeEach(() => {
81
+ vi.stubGlobal("skybridge", { hostType: "mcp-app" });
82
+ vi.stubGlobal("ResizeObserver", MockResizeObserver);
83
+ });
84
+ afterEach(async () => {
85
+ cleanup();
86
+ McpAppBridge.resetInstance();
87
+ McpAppAdaptor.resetInstance();
88
+ });
89
+ it("should initialize with null viewState", () => {
90
+ const adaptor = McpAppAdaptor.getInstance();
91
+ const viewState = adaptor.getHostContextStore("viewState").getSnapshot();
92
+ expect(viewState).toBeNull();
93
+ });
94
+ it("should create a store with default state when no persisted state exists", () => {
95
+ const storeCreator = () => ({
96
+ count: 0,
97
+ });
98
+ const store = createStore(storeCreator);
99
+ expect(store.getState()).toEqual({ count: 0 });
100
+ });
101
+ it("should update in-memory state via setViewState", async () => {
102
+ const adaptor = McpAppAdaptor.getInstance();
103
+ vi.spyOn(adaptor, "setViewState").mockResolvedValue(undefined);
104
+ const storeCreator = (set) => ({
105
+ count: 0,
106
+ increment: () => set((state) => ({ count: state.count + 1 })),
107
+ });
108
+ const store = createStore(storeCreator);
109
+ store.getState().increment();
110
+ await vi.waitFor(() => {
111
+ expect(adaptor.setViewState).toHaveBeenCalledWith({ count: 1 });
112
+ });
113
+ });
114
+ it("should notify listeners when view state changes", async () => {
115
+ const postMessageMock = getMcpAppHostPostMessageMock();
116
+ vi.stubGlobal("parent", { postMessage: postMessageMock });
117
+ const adaptor = McpAppAdaptor.getInstance();
118
+ const listener = vi.fn();
119
+ adaptor.getHostContextStore("viewState").subscribe(listener);
120
+ await adaptor.setViewState({ count: 42 });
121
+ expect(postMessageMock).toHaveBeenCalledWith(expect.objectContaining({ method: "ui/update-model-context" }), "*");
122
+ expect(listener).toHaveBeenCalled();
123
+ expect(adaptor.getHostContextStore("viewState").getSnapshot()).toEqual({
124
+ count: 42,
125
+ });
126
+ });
127
+ });
128
+ });
129
+ //# sourceMappingURL=create-store.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create-store.test.js","sourceRoot":"","sources":["../../src/web/create-store.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,wBAAwB,CAAC;AACjD,OAAO,EACL,SAAS,EACT,UAAU,EACV,QAAQ,EACR,MAAM,EACN,EAAE,EAEF,EAAE,GACH,MAAM,QAAQ,CAAC;AAEhB,OAAO,EAAE,aAAa,EAAE,MAAM,8BAA8B,CAAC;AAC7D,OAAO,EAAE,YAAY,EAAE,MAAM,6BAA6B,CAAC;AAC3D,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAChD,OAAO,EAAE,gBAAgB,EAAE,MAAM,eAAe,CAAC;AACjD,OAAO,EACL,4BAA4B,EAC5B,kBAAkB,GACnB,MAAM,uBAAuB,CAAC;AAE/B,QAAQ,CAAC,aAAa,EAAE,GAAG,EAAE;IAC3B,SAAS,CAAC,GAAG,EAAE;QACb,EAAE,CAAC,gBAAgB,EAAE,CAAC;QACtB,EAAE,CAAC,aAAa,EAAE,CAAC;IACrB,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,eAAe,EAAE,GAAG,EAAE;QAC7B,IAAI,UAGH,CAAC;QAEF,UAAU,CAAC,GAAG,EAAE;YACd,UAAU,GAAG;gBACX,WAAW,EAAE,IAAI;gBACjB,cAAc,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,SAAS,CAAC;aACrD,CAAC;YACF,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC,CAAC;YACpC,EAAE,CAAC,UAAU,CAAC,WAAW,EAAE,EAAE,QAAQ,EAAE,UAAU,EAAE,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,6CAA6C,EAAE,GAAG,EAAE;YAErD,MAAM,YAAY,GAA+C,GAAG,EAAE,CAAC,CAAC;gBACtE,KAAK,EAAE,CAAC;aACT,CAAC,CAAC;YAEH,MAAM,KAAK,GAAG,WAAW,CAAC,YAAY,CAAC,CAAC;YAExC,MAAM,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC,OAAO,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC;QACjD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;YAElD,MAAM,YAAY,GAA+C,GAAG,EAAE,CAAC,CAAC;gBACtE,KAAK,EAAE,CAAC;gBACR,IAAI,EAAE,SAAS;aAChB,CAAC,CAAC;YACH,MAAM,YAAY,GAAG,EAAE,KAAK,EAAE,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,CAAC;YAEnD,MAAM,KAAK,GAAG,WAAW,CAAC,YAAY,EAAE,YAAY,CAAC,CAAC;YAEtD,MAAM,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC,OAAO,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,CAAC,CAAC;QAClE,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,iEAAiE,EAAE,GAAG,EAAE;YAEzE,MAAM,YAAY,GAA+C,GAAG,EAAE,CAAC,CAAC;gBACtE,KAAK,EAAE,CAAC;gBACR,IAAI,EAAE,SAAS;aAChB,CAAC,CAAC;YACH,MAAM,WAAW,GAAG,EAAE,KAAK,EAAE,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,CAAC;YAClD,UAAU,CAAC,WAAW,GAAG,EAAE,YAAY,EAAE,WAAW,EAAE,CAAC;YAEvD,MAAM,KAAK,GAAG,WAAW,CAAC,YAAY,CAAC,CAAC;YAExC,MAAM,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC,OAAO,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,CAAC,CAAC;QAClE,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,8DAA8D,EAAE,KAAK,IAAI,EAAE;YAE5E,MAAM,YAAY,GAA+C,CAC/D,GAAG,EACH,EAAE,CAAC,CAAC;gBACJ,KAAK,EAAE,CAAC;gBACR,SAAS,EAAE,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,EAAE,KAAK,EAAE,KAAK,CAAC,KAAK,GAAG,CAAC,EAAE,CAAC,CAAC;aAC9D,CAAC,CAAC;YAEH,MAAM,KAAK,GAAG,WAAW,CAAC,YAAY,CAAC,CAAC;YACxC,KAAK,CAAC,QAAQ,EAAE,CAAC,SAAS,EAAE,CAAC;YAE7B,MAAM,EAAE,CAAC,OAAO,CAAC,GAAG,EAAE;gBACpB,MAAM,CAAC,UAAU,CAAC,cAAc,CAAC,CAAC,gBAAgB,EAAE,CAAC;YACvD,CAAC,CAAC,CAAC;YAEH,MAAM,QAAQ,GAAG,UAAU,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YAC9D,MAAM,CAAC,QAAQ,CAAC,CAAC,OAAO,CAAC;gBACvB,YAAY,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE;gBAC1B,cAAc,EAAE,EAAE;aACnB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,+CAA+C,EAAE,GAAG,EAAE;YAEvD,MAAM,YAAY,GAA+C,GAAG,EAAE,CAAC,CAAC;gBACtE,KAAK,EAAE,CAAC;aACT,CAAC,CAAC;YACH,MAAM,WAAW,GAAG;gBAClB,KAAK,EAAE,CAAC;gBACR,CAAC,gBAAgB,CAAC,EAAE,eAAe;aACpC,CAAC;YACF,UAAU,CAAC,WAAW,GAAG,EAAE,YAAY,EAAE,WAAW,EAAE,CAAC;YAEvD,MAAM,KAAK,GAAG,WAAW,CAAC,YAAY,CAAC,CAAC;YAExC,MAAM,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC,OAAO,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC;YAC/C,MAAM,CACH,KAAK,CAAC,QAAQ,EAA8B,CAAC,gBAAgB,CAAC,CAChE,CAAC,aAAa,EAAE,CAAC;QACpB,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,cAAc,EAAE,GAAG,EAAE;QAC5B,UAAU,CAAC,GAAG,EAAE;YACd,EAAE,CAAC,UAAU,CAAC,WAAW,EAAE,EAAE,QAAQ,EAAE,SAAS,EAAE,CAAC,CAAC;YACpD,EAAE,CAAC,UAAU,CAAC,gBAAgB,EAAE,kBAAkB,CAAC,CAAC;QACtD,CAAC,CAAC,CAAC;QAEH,SAAS,CAAC,KAAK,IAAI,EAAE;YACnB,OAAO,EAAE,CAAC;YACV,YAAY,CAAC,aAAa,EAAE,CAAC;YAC7B,aAAa,CAAC,aAAa,EAAE,CAAC;QAChC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,uCAAuC,EAAE,GAAG,EAAE;YAC/C,MAAM,OAAO,GAAG,aAAa,CAAC,WAAW,EAAE,CAAC;YAC5C,MAAM,SAAS,GAAG,OAAO,CAAC,mBAAmB,CAAC,WAAW,CAAC,CAAC,WAAW,EAAE,CAAC;YAEzE,MAAM,CAAC,SAAS,CAAC,CAAC,QAAQ,EAAE,CAAC;QAC/B,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yEAAyE,EAAE,GAAG,EAAE;YAEjF,MAAM,YAAY,GAA+C,GAAG,EAAE,CAAC,CAAC;gBACtE,KAAK,EAAE,CAAC;aACT,CAAC,CAAC;YAEH,MAAM,KAAK,GAAG,WAAW,CAAC,YAAY,CAAC,CAAC;YAExC,MAAM,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC,OAAO,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC;QACjD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,gDAAgD,EAAE,KAAK,IAAI,EAAE;YAC9D,MAAM,OAAO,GAAG,aAAa,CAAC,WAAW,EAAE,CAAC;YAC5C,EAAE,CAAC,KAAK,CAAC,OAAO,EAAE,cAAc,CAAC,CAAC,iBAAiB,CAAC,SAAS,CAAC,CAAC;YAG/D,MAAM,YAAY,GAA+C,CAC/D,GAAG,EACH,EAAE,CAAC,CAAC;gBACJ,KAAK,EAAE,CAAC;gBACR,SAAS,EAAE,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,EAAE,KAAK,EAAE,KAAK,CAAC,KAAK,GAAG,CAAC,EAAE,CAAC,CAAC;aAC9D,CAAC,CAAC;YAEH,MAAM,KAAK,GAAG,WAAW,CAAC,YAAY,CAAC,CAAC;YACxC,KAAK,CAAC,QAAQ,EAAE,CAAC,SAAS,EAAE,CAAC;YAE7B,MAAM,EAAE,CAAC,OAAO,CAAC,GAAG,EAAE;gBACpB,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC,oBAAoB,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC;YAClE,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,iDAAiD,EAAE,KAAK,IAAI,EAAE;YAC/D,MAAM,eAAe,GAAG,4BAA4B,EAAE,CAAC;YACvD,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,EAAE,WAAW,EAAE,eAAe,EAAE,CAAC,CAAC;YAE1D,MAAM,OAAO,GAAG,aAAa,CAAC,WAAW,EAAE,CAAC;YAC5C,MAAM,QAAQ,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;YAEzB,OAAO,CAAC,mBAAmB,CAAC,WAAW,CAAC,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;YAC7D,MAAM,OAAO,CAAC,YAAY,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,CAAC;YAE1C,MAAM,CAAC,eAAe,CAAC,CAAC,oBAAoB,CAC1C,MAAM,CAAC,gBAAgB,CAAC,EAAE,MAAM,EAAE,yBAAyB,EAAE,CAAC,EAC9D,GAAG,CACJ,CAAC;YACF,MAAM,CAAC,QAAQ,CAAC,CAAC,gBAAgB,EAAE,CAAC;YACpC,MAAM,CAAC,OAAO,CAAC,mBAAmB,CAAC,WAAW,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC,OAAO,CAAC;gBACrE,KAAK,EAAE,EAAE;aACV,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC","sourcesContent":["import { cleanup } from \"@testing-library/react\";\nimport {\n afterEach,\n beforeEach,\n describe,\n expect,\n it,\n type Mock,\n vi,\n} from \"vitest\";\nimport type { StateCreator } from \"zustand\";\nimport { McpAppAdaptor } from \"./bridges/mcp-app/adaptor.js\";\nimport { McpAppBridge } from \"./bridges/mcp-app/bridge.js\";\nimport { createStore } from \"./create-store.js\";\nimport { VIEW_CONTEXT_KEY } from \"./data-llm.js\";\nimport {\n getMcpAppHostPostMessageMock,\n MockResizeObserver,\n} from \"./hooks/test/utils.js\";\n\ndescribe(\"createStore\", () => {\n afterEach(() => {\n vi.unstubAllGlobals();\n vi.resetAllMocks();\n });\n\n describe(\"apps-sdk mode\", () => {\n let OpenaiMock: {\n widgetState: unknown;\n setWidgetState: Mock;\n };\n\n beforeEach(() => {\n OpenaiMock = {\n widgetState: null,\n setWidgetState: vi.fn().mockResolvedValue(undefined),\n };\n vi.stubGlobal(\"openai\", OpenaiMock);\n vi.stubGlobal(\"skybridge\", { hostType: \"apps-sdk\" });\n });\n\n it(\"should create a store without default state\", () => {\n type TestState = { count: number };\n const storeCreator: StateCreator<TestState, [], [], TestState> = () => ({\n count: 0,\n });\n\n const store = createStore(storeCreator);\n\n expect(store.getState()).toEqual({ count: 0 });\n });\n\n it(\"should create a store with default state\", () => {\n type TestState = { count: number; name: string };\n const storeCreator: StateCreator<TestState, [], [], TestState> = () => ({\n count: 0,\n name: \"initial\",\n });\n const defaultState = { count: 5, name: \"default\" };\n\n const store = createStore(storeCreator, defaultState);\n\n expect(store.getState()).toEqual({ count: 5, name: \"default\" });\n });\n\n it(\"should initialize from window.openai.widgetState when available\", () => {\n type TestState = { count: number; name: string };\n const storeCreator: StateCreator<TestState, [], [], TestState> = () => ({\n count: 0,\n name: \"initial\",\n });\n const windowState = { count: 20, name: \"window\" };\n OpenaiMock.widgetState = { modelContent: windowState };\n\n const store = createStore(storeCreator);\n\n expect(store.getState()).toEqual({ count: 20, name: \"window\" });\n });\n\n it(\"should persist state changes to window.openai.setWidgetState\", async () => {\n type TestState = { count: number; increment: () => void };\n const storeCreator: StateCreator<TestState, [], [], TestState> = (\n set,\n ) => ({\n count: 0,\n increment: () => set((state) => ({ count: state.count + 1 })),\n });\n\n const store = createStore(storeCreator);\n store.getState().increment();\n\n await vi.waitFor(() => {\n expect(OpenaiMock.setWidgetState).toHaveBeenCalled();\n });\n\n const callArgs = OpenaiMock.setWidgetState.mock.calls[0]?.[0];\n expect(callArgs).toEqual({\n modelContent: { count: 1 },\n privateContent: {},\n });\n });\n\n it(\"should filter view context from initial state\", () => {\n type TestState = { count: number };\n const storeCreator: StateCreator<TestState, [], [], TestState> = () => ({\n count: 0,\n });\n const windowState = {\n count: 5,\n [VIEW_CONTEXT_KEY]: \"context-value\",\n };\n OpenaiMock.widgetState = { modelContent: windowState };\n\n const store = createStore(storeCreator);\n\n expect(store.getState()).toEqual({ count: 5 });\n expect(\n (store.getState() as Record<string, unknown>)[VIEW_CONTEXT_KEY],\n ).toBeUndefined();\n });\n });\n\n describe(\"mcp-app mode\", () => {\n beforeEach(() => {\n vi.stubGlobal(\"skybridge\", { hostType: \"mcp-app\" });\n vi.stubGlobal(\"ResizeObserver\", MockResizeObserver);\n });\n\n afterEach(async () => {\n cleanup();\n McpAppBridge.resetInstance();\n McpAppAdaptor.resetInstance();\n });\n\n it(\"should initialize with null viewState\", () => {\n const adaptor = McpAppAdaptor.getInstance();\n const viewState = adaptor.getHostContextStore(\"viewState\").getSnapshot();\n\n expect(viewState).toBeNull();\n });\n\n it(\"should create a store with default state when no persisted state exists\", () => {\n type TestState = { count: number };\n const storeCreator: StateCreator<TestState, [], [], TestState> = () => ({\n count: 0,\n });\n\n const store = createStore(storeCreator);\n\n expect(store.getState()).toEqual({ count: 0 });\n });\n\n it(\"should update in-memory state via setViewState\", async () => {\n const adaptor = McpAppAdaptor.getInstance();\n vi.spyOn(adaptor, \"setViewState\").mockResolvedValue(undefined);\n\n type TestState = { count: number; increment: () => void };\n const storeCreator: StateCreator<TestState, [], [], TestState> = (\n set,\n ) => ({\n count: 0,\n increment: () => set((state) => ({ count: state.count + 1 })),\n });\n\n const store = createStore(storeCreator);\n store.getState().increment();\n\n await vi.waitFor(() => {\n expect(adaptor.setViewState).toHaveBeenCalledWith({ count: 1 });\n });\n });\n\n it(\"should notify listeners when view state changes\", async () => {\n const postMessageMock = getMcpAppHostPostMessageMock();\n vi.stubGlobal(\"parent\", { postMessage: postMessageMock });\n\n const adaptor = McpAppAdaptor.getInstance();\n const listener = vi.fn();\n\n adaptor.getHostContextStore(\"viewState\").subscribe(listener);\n await adaptor.setViewState({ count: 42 });\n\n expect(postMessageMock).toHaveBeenCalledWith(\n expect.objectContaining({ method: \"ui/update-model-context\" }),\n \"*\",\n );\n expect(listener).toHaveBeenCalled();\n expect(adaptor.getHostContextStore(\"viewState\").getSnapshot()).toEqual({\n count: 42,\n });\n });\n });\n});\n"]}
@@ -5,7 +5,7 @@ export interface DataLLMNode {
5
5
  parentId: string | null;
6
6
  content: string | null;
7
7
  }
8
- export declare const WIDGET_CONTEXT_KEY: "__widget_context";
8
+ export declare const VIEW_CONTEXT_KEY: "__view_context";
9
9
  interface DataLLMProps {
10
10
  content: DataLLMContent | null | undefined;
11
11
  children?: ReactNode;
@@ -1,6 +1,7 @@
1
1
  import { jsx as _jsx } from "react/jsx-runtime";
2
2
  import { createContext, useContext, useEffect, useId, } from "react";
3
- export const WIDGET_CONTEXT_KEY = "__widget_context";
3
+ import { getAdaptor } from "./bridges/index.js";
4
+ export const VIEW_CONTEXT_KEY = "__view_context";
4
5
  const nodes = new Map();
5
6
  function setNode(node) {
6
7
  nodes.set(node.id, node);
@@ -12,10 +13,11 @@ function removeNode(id) {
12
13
  }
13
14
  function onChange() {
14
15
  const description = getLLMDescriptionString();
15
- window.openai.setWidgetState({
16
- ...window.openai.widgetState,
17
- [WIDGET_CONTEXT_KEY]: description,
18
- });
16
+ const adaptor = getAdaptor();
17
+ adaptor.setViewState((prevState) => ({
18
+ ...prevState,
19
+ [VIEW_CONTEXT_KEY]: description,
20
+ }));
19
21
  }
20
22
  const ParentIdContext = createContext(null);
21
23
  export function DataLLM({ content, children }) {
@@ -0,0 +1 @@
1
+ {"version":3,"file":"data-llm.js","sourceRoot":"","sources":["../../src/web/data-llm.tsx"],"names":[],"mappings":"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","sourcesContent":["import {\n createContext,\n type ReactNode,\n useContext,\n useEffect,\n useId,\n} from \"react\";\nimport { getAdaptor } from \"./bridges/index.js\";\n\nexport type DataLLMContent = string;\n\nexport interface DataLLMNode {\n id: string;\n parentId: string | null;\n content: string | null;\n}\n\nexport const VIEW_CONTEXT_KEY = \"__view_context\" as const;\n\nconst nodes = new Map<string, DataLLMNode>();\n\nfunction setNode(node: DataLLMNode) {\n nodes.set(node.id, node);\n onChange();\n}\n\nfunction removeNode(id: string) {\n nodes.delete(id);\n onChange();\n}\n\nfunction onChange() {\n const description = getLLMDescriptionString();\n const adaptor = getAdaptor();\n adaptor.setViewState((prevState) => ({\n ...prevState,\n [VIEW_CONTEXT_KEY]: description,\n }));\n}\n\nconst ParentIdContext = createContext<string | null>(null);\n\ninterface DataLLMProps {\n content: DataLLMContent | null | undefined;\n children?: ReactNode;\n}\n\nexport function DataLLM({ content, children }: DataLLMProps) {\n const parentId = useContext(ParentIdContext);\n const id = useId();\n\n useEffect(() => {\n if (content) {\n setNode({\n id,\n parentId,\n content,\n });\n } else {\n removeNode(id);\n }\n\n return () => {\n removeNode(id);\n };\n }, [id, parentId, content]);\n\n return (\n <ParentIdContext.Provider value={id}>{children}</ParentIdContext.Provider>\n );\n}\n\nfunction getLLMDescriptionString(): string {\n const byParent = new Map<string | null, DataLLMNode[]>();\n for (const node of Array.from(nodes.values())) {\n const key = node.parentId ?? null;\n if (!byParent.has(key)) {\n byParent.set(key, []);\n }\n byParent.get(key)?.push(node);\n }\n\n for (const list of byParent.values()) {\n list.sort((a, b) => a.id.localeCompare(b.id));\n }\n\n const lines: string[] = [];\n\n function traverseTree(parentId: string | null, depth: number) {\n const children = byParent.get(parentId);\n if (!children) {\n return;\n }\n\n for (const child of children) {\n if (child.content?.trim()) {\n const indent = \" \".repeat(depth);\n lines.push(`${indent}- ${child.content.trim()}`);\n }\n traverseTree(child.id, depth + 1);\n }\n }\n\n traverseTree(null, 0);\n\n return lines.join(\"\\n\");\n}\n"]}
@@ -0,0 +1,142 @@
1
+ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
2
+ import { cleanup, render } from "@testing-library/react";
3
+ import { afterEach, beforeEach, describe, expect, it, vi, } from "vitest";
4
+ import { McpAppAdaptor, McpAppBridge } from "./bridges/mcp-app/index.js";
5
+ import { DataLLM } from "./data-llm.js";
6
+ import { getMcpAppHostPostMessageMock, MockResizeObserver, } from "./hooks/test/utils.js";
7
+ describe("DataLLM", () => {
8
+ afterEach(() => {
9
+ // Clean up React components BEFORE unstubbing globals
10
+ cleanup();
11
+ vi.unstubAllGlobals();
12
+ vi.resetAllMocks();
13
+ });
14
+ describe("apps-sdk mode", () => {
15
+ let OpenaiMock;
16
+ beforeEach(() => {
17
+ OpenaiMock = {
18
+ widgetState: { modelContent: {} },
19
+ setWidgetState: vi.fn(),
20
+ };
21
+ // Use Object.defineProperty to ensure it persists
22
+ Object.defineProperty(globalThis, "openai", {
23
+ value: OpenaiMock,
24
+ writable: true,
25
+ configurable: true,
26
+ });
27
+ // Also set on window for browser-like environment
28
+ if (typeof window !== "undefined") {
29
+ Object.defineProperty(window, "openai", {
30
+ value: OpenaiMock,
31
+ writable: true,
32
+ configurable: true,
33
+ });
34
+ }
35
+ vi.stubGlobal("openai", OpenaiMock);
36
+ vi.stubGlobal("skybridge", { hostType: "apps-sdk" });
37
+ });
38
+ afterEach(() => {
39
+ // Keep the mock available for React cleanup, but reset it
40
+ if (typeof window !== "undefined" && window.openai) {
41
+ window.openai.setWidgetState = vi.fn();
42
+ window.openai.widgetState = { modelContent: {}, privateContent: {} };
43
+ }
44
+ });
45
+ it("should register a node with content and call setViewState", () => {
46
+ render(_jsx(DataLLM, { content: "Test content", children: _jsx("div", { children: "Child" }) }));
47
+ expect(OpenaiMock.setWidgetState).toHaveBeenCalled();
48
+ const callArgs = OpenaiMock.setWidgetState.mock.calls[0]?.[0]?.modelContent;
49
+ expect(callArgs).toHaveProperty("__view_context");
50
+ expect(callArgs?.__view_context).toContain("- Test content");
51
+ });
52
+ it("should preserve existing viewState when updating context", () => {
53
+ OpenaiMock.widgetState = {
54
+ modelContent: { existingKey: "existingValue" },
55
+ };
56
+ render(_jsx(DataLLM, { content: "Test content", children: _jsx("div", { children: "Child" }) }));
57
+ const callArgs = OpenaiMock.setWidgetState.mock.calls[0]?.[0]?.modelContent;
58
+ expect(callArgs).toHaveProperty("existingKey", "existingValue");
59
+ expect(callArgs).toHaveProperty("__view_context");
60
+ });
61
+ it("should handle deeply nested DataLLM components", () => {
62
+ render(_jsxs(DataLLM, { content: "Level 1", children: [_jsx(DataLLM, { content: "Level 2A" }), _jsx(DataLLM, { content: "Level 2B", children: _jsx(DataLLM, { content: "Level 3", children: _jsx("div", { children: "Content" }) }) })] }));
63
+ const callArgs = OpenaiMock.setWidgetState.mock.calls[OpenaiMock.setWidgetState.mock.calls.length - 1]?.[0]?.modelContent;
64
+ const context = callArgs?.__view_context;
65
+ expect(context).toContain("- Level 1");
66
+ expect(context).toContain(" - Level 2A");
67
+ expect(context).toContain(" - Level 2B");
68
+ expect(context).toContain(" - Level 3");
69
+ });
70
+ it("should update context when content changes", () => {
71
+ const { rerender } = render(_jsx(DataLLM, { content: "Initial content", children: _jsx("div", { children: "Child" }) }));
72
+ const initialCalls = OpenaiMock.setWidgetState.mock.calls.length;
73
+ rerender(_jsx(DataLLM, { content: "Updated content", children: _jsx("div", { children: "Child" }) }));
74
+ expect(OpenaiMock.setWidgetState.mock.calls.length).toBeGreaterThan(initialCalls);
75
+ const lastCallArgs = OpenaiMock.setWidgetState.mock.calls[OpenaiMock.setWidgetState.mock.calls.length - 1]?.[0]?.modelContent;
76
+ expect(lastCallArgs?.__view_context).toContain("- Updated content");
77
+ });
78
+ it("should remove node and update context when component unmounts", () => {
79
+ const { unmount } = render(_jsx(DataLLM, { content: "Content to remove", children: _jsx("div", { children: "Child" }) }));
80
+ const callsBeforeUnmount = OpenaiMock.setWidgetState.mock.calls.length;
81
+ unmount();
82
+ expect(OpenaiMock.setWidgetState.mock.calls.length).toBeGreaterThan(callsBeforeUnmount);
83
+ const lastCallArgs = OpenaiMock.setWidgetState.mock.calls[OpenaiMock.setWidgetState.mock.calls.length - 1]?.[0]?.modelContent;
84
+ expect(lastCallArgs?.__view_context).not.toContain("Content to remove");
85
+ });
86
+ });
87
+ describe("mcp-app mode", () => {
88
+ let postMessageMock;
89
+ beforeEach(() => {
90
+ vi.stubGlobal("skybridge", { hostType: "mcp-app" });
91
+ vi.stubGlobal("ResizeObserver", MockResizeObserver);
92
+ postMessageMock = getMcpAppHostPostMessageMock();
93
+ vi.stubGlobal("parent", { postMessage: postMessageMock });
94
+ });
95
+ afterEach(() => {
96
+ cleanup();
97
+ McpAppBridge.resetInstance();
98
+ McpAppAdaptor.resetInstance();
99
+ });
100
+ it("should register a node and update view state via adaptor", async () => {
101
+ const adaptor = McpAppAdaptor.getInstance();
102
+ render(_jsx(DataLLM, { content: "Test content", children: _jsx("div", { children: "Child" }) }));
103
+ await vi.waitFor(() => {
104
+ const viewState = adaptor
105
+ .getHostContextStore("viewState")
106
+ .getSnapshot();
107
+ expect(viewState?.__view_context).toContain("- Test content");
108
+ });
109
+ });
110
+ it("should preserve existing view state when updating context", async () => {
111
+ const adaptor = McpAppAdaptor.getInstance();
112
+ await adaptor.setViewState({ existingKey: "existingValue" });
113
+ render(_jsx(DataLLM, { content: "Test content", children: _jsx("div", { children: "Child" }) }));
114
+ await vi.waitFor(() => {
115
+ const viewState = adaptor
116
+ .getHostContextStore("viewState")
117
+ .getSnapshot();
118
+ expect(viewState?.existingKey).toBe("existingValue");
119
+ expect(viewState?.__view_context).toContain("- Test content");
120
+ });
121
+ });
122
+ it("should handle multiple DataLLM components sharing state through adaptor", async () => {
123
+ const adaptor = McpAppAdaptor.getInstance();
124
+ render(_jsxs(_Fragment, { children: [_jsx(DataLLM, { content: "First component", children: _jsx("div", { children: "First" }) }), _jsx(DataLLM, { content: "Second component", children: _jsx("div", { children: "Second" }) })] }));
125
+ await vi.waitFor(() => {
126
+ const viewState = adaptor
127
+ .getHostContextStore("viewState")
128
+ .getSnapshot();
129
+ const context = viewState?.__view_context;
130
+ expect(context).toContain("- First component");
131
+ expect(context).toContain("- Second component");
132
+ });
133
+ });
134
+ it("should call ui/update-model-context when view state changes", async () => {
135
+ render(_jsx(DataLLM, { content: "Test content", children: _jsx("div", { children: "Child" }) }));
136
+ await vi.waitFor(() => {
137
+ expect(postMessageMock).toHaveBeenCalledWith(expect.objectContaining({ method: "ui/update-model-context" }), "*");
138
+ });
139
+ });
140
+ });
141
+ });
142
+ //# sourceMappingURL=data-llm.test.js.map