skybridge 0.0.0-dev.f89b570 → 0.0.0-dev.f8d2f8c

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 (259) hide show
  1. package/README.md +24 -16
  2. package/dist/cli/detect-port.js.map +1 -1
  3. package/dist/cli/header.js +1 -1
  4. package/dist/cli/header.js.map +1 -1
  5. package/dist/cli/run-command.js.map +1 -1
  6. package/dist/cli/telemetry.js.map +1 -1
  7. package/dist/cli/tunnel-control-server.d.ts +9 -0
  8. package/dist/cli/tunnel-control-server.js +31 -0
  9. package/dist/cli/tunnel-control-server.js.map +1 -0
  10. package/dist/cli/tunnel-control-server.test.js +39 -0
  11. package/dist/cli/tunnel-control-server.test.js.map +1 -0
  12. package/dist/cli/tunnel-handler.d.ts +3 -0
  13. package/dist/cli/tunnel-handler.js +48 -0
  14. package/dist/cli/tunnel-handler.js.map +1 -0
  15. package/dist/cli/tunnel-handler.test.js +105 -0
  16. package/dist/cli/tunnel-handler.test.js.map +1 -0
  17. package/dist/cli/tunnel.d.ts +57 -0
  18. package/dist/cli/tunnel.js +154 -0
  19. package/dist/cli/tunnel.js.map +1 -0
  20. package/dist/cli/tunnel.test.js +190 -0
  21. package/dist/cli/tunnel.test.js.map +1 -0
  22. package/dist/cli/types.d.ts +5 -0
  23. package/dist/cli/types.js +2 -0
  24. package/dist/cli/types.js.map +1 -0
  25. package/dist/cli/use-execute-steps.js.map +1 -1
  26. package/dist/cli/use-messages.d.ts +3 -0
  27. package/dist/cli/use-messages.js +11 -0
  28. package/dist/cli/use-messages.js.map +1 -0
  29. package/dist/cli/use-nodemon.d.ts +2 -7
  30. package/dist/cli/use-nodemon.js +18 -21
  31. package/dist/cli/use-nodemon.js.map +1 -1
  32. package/dist/cli/use-open-browser.d.ts +1 -0
  33. package/dist/cli/use-open-browser.js +44 -0
  34. package/dist/cli/use-open-browser.js.map +1 -0
  35. package/dist/cli/use-tunnel.d.ts +14 -0
  36. package/dist/cli/use-tunnel.js +131 -0
  37. package/dist/cli/use-tunnel.js.map +1 -0
  38. package/dist/cli/use-typescript-check.d.ts +1 -0
  39. package/dist/cli/use-typescript-check.js +41 -6
  40. package/dist/cli/use-typescript-check.js.map +1 -1
  41. package/dist/commands/build.js +63 -7
  42. package/dist/commands/build.js.map +1 -1
  43. package/dist/commands/dev.d.ts +3 -1
  44. package/dist/commands/dev.js +46 -8
  45. package/dist/commands/dev.js.map +1 -1
  46. package/dist/commands/start.js +7 -10
  47. package/dist/commands/start.js.map +1 -1
  48. package/dist/commands/telemetry/disable.js.map +1 -1
  49. package/dist/commands/telemetry/enable.js.map +1 -1
  50. package/dist/commands/telemetry/status.js.map +1 -1
  51. package/dist/server/asset-base-url-transform-plugin.d.ts +5 -6
  52. package/dist/server/asset-base-url-transform-plugin.js +9 -10
  53. package/dist/server/asset-base-url-transform-plugin.js.map +1 -1
  54. package/dist/server/asset-base-url-transform-plugin.test.js +41 -13
  55. package/dist/server/asset-base-url-transform-plugin.test.js.map +1 -1
  56. package/dist/server/content-helpers.d.ts +27 -0
  57. package/dist/server/content-helpers.js +46 -0
  58. package/dist/server/content-helpers.js.map +1 -0
  59. package/dist/server/content-helpers.test.d.ts +1 -0
  60. package/dist/server/content-helpers.test.js +70 -0
  61. package/dist/server/content-helpers.test.js.map +1 -0
  62. package/dist/server/express.d.ts +7 -5
  63. package/dist/server/express.js +52 -23
  64. package/dist/server/express.js.map +1 -1
  65. package/dist/server/express.test.js +381 -25
  66. package/dist/server/express.test.js.map +1 -1
  67. package/dist/server/index.d.ts +5 -3
  68. package/dist/server/index.js +3 -2
  69. package/dist/server/index.js.map +1 -1
  70. package/dist/server/inferUtilityTypes.d.ts +6 -6
  71. package/dist/server/inferUtilityTypes.js.map +1 -1
  72. package/dist/server/metric.d.ts +14 -0
  73. package/dist/server/metric.js +62 -0
  74. package/dist/server/metric.js.map +1 -0
  75. package/dist/server/middleware.d.ts +124 -0
  76. package/dist/server/middleware.js +93 -0
  77. package/dist/server/middleware.js.map +1 -0
  78. package/dist/server/middleware.test-d.d.ts +1 -0
  79. package/dist/server/middleware.test-d.js +75 -0
  80. package/dist/server/middleware.test-d.js.map +1 -0
  81. package/dist/server/middleware.test.d.ts +1 -0
  82. package/dist/server/middleware.test.js +493 -0
  83. package/dist/server/middleware.test.js.map +1 -0
  84. package/dist/server/server.d.ts +158 -67
  85. package/dist/server/server.js +364 -102
  86. package/dist/server/server.js.map +1 -1
  87. package/dist/server/templateHelper.d.ts +5 -8
  88. package/dist/server/templateHelper.js +3 -22
  89. package/dist/server/templateHelper.js.map +1 -1
  90. package/dist/server/templates.generated.d.ts +4 -0
  91. package/dist/server/templates.generated.js +47 -0
  92. package/dist/server/templates.generated.js.map +1 -0
  93. package/dist/server/tunnel-proxy-router.d.ts +7 -0
  94. package/dist/server/tunnel-proxy-router.js +110 -0
  95. package/dist/server/tunnel-proxy-router.js.map +1 -0
  96. package/dist/server/tunnel-proxy-router.test.d.ts +1 -0
  97. package/dist/server/tunnel-proxy-router.test.js +229 -0
  98. package/dist/server/tunnel-proxy-router.test.js.map +1 -0
  99. package/dist/server/viewsDevServer.d.ts +14 -0
  100. package/dist/server/viewsDevServer.js +45 -0
  101. package/dist/server/viewsDevServer.js.map +1 -0
  102. package/dist/test/utils.d.ts +13 -21
  103. package/dist/test/utils.js +42 -37
  104. package/dist/test/utils.js.map +1 -1
  105. package/dist/test/view.test.d.ts +1 -0
  106. package/dist/test/view.test.js +523 -0
  107. package/dist/test/view.test.js.map +1 -0
  108. package/dist/version.d.ts +1 -0
  109. package/dist/version.js +3 -0
  110. package/dist/version.js.map +1 -0
  111. package/dist/web/bridges/apps-sdk/adaptor.d.ts +6 -4
  112. package/dist/web/bridges/apps-sdk/adaptor.js +37 -16
  113. package/dist/web/bridges/apps-sdk/adaptor.js.map +1 -1
  114. package/dist/web/bridges/apps-sdk/bridge.js.map +1 -1
  115. package/dist/web/bridges/apps-sdk/index.js.map +1 -1
  116. package/dist/web/bridges/apps-sdk/types.d.ts +16 -6
  117. package/dist/web/bridges/apps-sdk/types.js.map +1 -1
  118. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.js.map +1 -1
  119. package/dist/web/bridges/get-adaptor.js.map +1 -1
  120. package/dist/web/bridges/index.js.map +1 -1
  121. package/dist/web/bridges/mcp-app/adaptor.d.ts +20 -8
  122. package/dist/web/bridges/mcp-app/adaptor.js +135 -62
  123. package/dist/web/bridges/mcp-app/adaptor.js.map +1 -1
  124. package/dist/web/bridges/mcp-app/bridge.d.ts +13 -30
  125. package/dist/web/bridges/mcp-app/bridge.js +43 -201
  126. package/dist/web/bridges/mcp-app/bridge.js.map +1 -1
  127. package/dist/web/bridges/mcp-app/index.js.map +1 -1
  128. package/dist/web/bridges/mcp-app/types.js.map +1 -1
  129. package/dist/web/bridges/mcp-app/use-mcp-app-context.d.ts +5 -3
  130. package/dist/web/bridges/mcp-app/use-mcp-app-context.js +2 -2
  131. package/dist/web/bridges/mcp-app/use-mcp-app-context.js.map +1 -1
  132. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js +1 -41
  133. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js.map +1 -1
  134. package/dist/web/bridges/types.d.ts +19 -10
  135. package/dist/web/bridges/types.js.map +1 -1
  136. package/dist/web/bridges/use-host-context.js.map +1 -1
  137. package/dist/web/components/modal-provider.js +3 -5
  138. package/dist/web/components/modal-provider.js.map +1 -1
  139. package/dist/web/create-store.js +17 -3
  140. package/dist/web/create-store.js.map +1 -1
  141. package/dist/web/create-store.test.js +17 -17
  142. package/dist/web/create-store.test.js.map +1 -1
  143. package/dist/web/data-llm.d.ts +1 -1
  144. package/dist/web/data-llm.js +3 -3
  145. package/dist/web/data-llm.js.map +1 -1
  146. package/dist/web/data-llm.test.js +23 -22
  147. package/dist/web/data-llm.test.js.map +1 -1
  148. package/dist/web/generate-helpers.d.ts +20 -18
  149. package/dist/web/generate-helpers.js +20 -18
  150. package/dist/web/generate-helpers.js.map +1 -1
  151. package/dist/web/generate-helpers.test-d.js +26 -26
  152. package/dist/web/generate-helpers.test-d.js.map +1 -1
  153. package/dist/web/generate-helpers.test.js.map +1 -1
  154. package/dist/web/helpers/state.d.ts +2 -2
  155. package/dist/web/helpers/state.js +11 -11
  156. package/dist/web/helpers/state.js.map +1 -1
  157. package/dist/web/helpers/state.test.js +9 -9
  158. package/dist/web/helpers/state.test.js.map +1 -1
  159. package/dist/web/hooks/index.d.ts +1 -1
  160. package/dist/web/hooks/index.js +1 -1
  161. package/dist/web/hooks/index.js.map +1 -1
  162. package/dist/web/hooks/test/utils.js +4 -0
  163. package/dist/web/hooks/test/utils.js.map +1 -1
  164. package/dist/web/hooks/use-call-tool.js.map +1 -1
  165. package/dist/web/hooks/use-call-tool.test-d.js.map +1 -1
  166. package/dist/web/hooks/use-call-tool.test.js +0 -4
  167. package/dist/web/hooks/use-call-tool.test.js.map +1 -1
  168. package/dist/web/hooks/use-display-mode.js.map +1 -1
  169. package/dist/web/hooks/use-display-mode.test-d.js.map +1 -1
  170. package/dist/web/hooks/use-display-mode.test.js.map +1 -1
  171. package/dist/web/hooks/use-files.d.ts +2 -1
  172. package/dist/web/hooks/use-files.js +1 -0
  173. package/dist/web/hooks/use-files.js.map +1 -1
  174. package/dist/web/hooks/use-files.test.js +22 -2
  175. package/dist/web/hooks/use-files.test.js.map +1 -1
  176. package/dist/web/hooks/use-layout.js.map +1 -1
  177. package/dist/web/hooks/use-layout.test.js +3 -3
  178. package/dist/web/hooks/use-layout.test.js.map +1 -1
  179. package/dist/web/hooks/use-open-external.js.map +1 -1
  180. package/dist/web/hooks/use-open-external.test.js +15 -10
  181. package/dist/web/hooks/use-open-external.test.js.map +1 -1
  182. package/dist/web/hooks/use-request-modal.d.ts +1 -1
  183. package/dist/web/hooks/use-request-modal.js +4 -4
  184. package/dist/web/hooks/use-request-modal.js.map +1 -1
  185. package/dist/web/hooks/use-request-modal.test.js +5 -1
  186. package/dist/web/hooks/use-request-modal.test.js.map +1 -1
  187. package/dist/web/hooks/use-send-follow-up-message.d.ts +2 -1
  188. package/dist/web/hooks/use-send-follow-up-message.js +2 -2
  189. package/dist/web/hooks/use-send-follow-up-message.js.map +1 -1
  190. package/dist/web/hooks/use-set-open-in-app-url.js.map +1 -1
  191. package/dist/web/hooks/use-set-open-in-app-url.test.js.map +1 -1
  192. package/dist/web/hooks/use-tool-info.js.map +1 -1
  193. package/dist/web/hooks/use-tool-info.test-d.js.map +1 -1
  194. package/dist/web/hooks/use-tool-info.test.js +1 -1
  195. package/dist/web/hooks/use-tool-info.test.js.map +1 -1
  196. package/dist/web/hooks/use-user.js +18 -2
  197. package/dist/web/hooks/use-user.js.map +1 -1
  198. package/dist/web/hooks/use-user.test.js +29 -1
  199. package/dist/web/hooks/use-user.test.js.map +1 -1
  200. package/dist/web/hooks/use-view-state.d.ts +4 -0
  201. package/dist/web/hooks/use-view-state.js +32 -0
  202. package/dist/web/hooks/use-view-state.js.map +1 -0
  203. package/dist/web/hooks/use-view-state.test.d.ts +1 -0
  204. package/dist/web/hooks/use-view-state.test.js +177 -0
  205. package/dist/web/hooks/use-view-state.test.js.map +1 -0
  206. package/dist/web/index.d.ts +1 -2
  207. package/dist/web/index.js +1 -2
  208. package/dist/web/index.js.map +1 -1
  209. package/dist/web/mount-view.d.ts +1 -0
  210. package/dist/web/{mount-widget.js → mount-view.js} +2 -2
  211. package/dist/web/mount-view.js.map +1 -0
  212. package/dist/web/plugin/data-llm.test.js.map +1 -1
  213. package/dist/web/plugin/plugin.d.ts +4 -1
  214. package/dist/web/plugin/plugin.js +127 -25
  215. package/dist/web/plugin/plugin.js.map +1 -1
  216. package/dist/web/plugin/scan-views.d.ts +16 -0
  217. package/dist/web/plugin/scan-views.js +88 -0
  218. package/dist/web/plugin/scan-views.js.map +1 -0
  219. package/dist/web/plugin/scan-views.test.d.ts +1 -0
  220. package/dist/web/plugin/scan-views.test.js +99 -0
  221. package/dist/web/plugin/scan-views.test.js.map +1 -0
  222. package/dist/web/plugin/transform-data-llm.js +1 -1
  223. package/dist/web/plugin/transform-data-llm.js.map +1 -1
  224. package/dist/web/plugin/transform-data-llm.test.js.map +1 -1
  225. package/dist/web/plugin/validate-view.d.ts +1 -0
  226. package/dist/web/plugin/validate-view.js +9 -0
  227. package/dist/web/plugin/validate-view.js.map +1 -0
  228. package/dist/web/plugin/validate-view.test.d.ts +1 -0
  229. package/dist/web/plugin/validate-view.test.js +24 -0
  230. package/dist/web/plugin/validate-view.test.js.map +1 -0
  231. package/dist/web/proxy.js.map +1 -1
  232. package/dist/web/types.js.map +1 -1
  233. package/package.json +31 -21
  234. package/tsconfig.base.json +5 -0
  235. package/dist/server/const.d.ts +0 -1
  236. package/dist/server/const.js +0 -2
  237. package/dist/server/const.js.map +0 -1
  238. package/dist/server/templates/development.hbs +0 -67
  239. package/dist/server/templates/production.hbs +0 -6
  240. package/dist/server/widgetsDevServer.d.ts +0 -12
  241. package/dist/server/widgetsDevServer.js +0 -63
  242. package/dist/server/widgetsDevServer.js.map +0 -1
  243. package/dist/test/widget.test.js +0 -261
  244. package/dist/test/widget.test.js.map +0 -1
  245. package/dist/web/hooks/use-widget-state.d.ts +0 -4
  246. package/dist/web/hooks/use-widget-state.js +0 -32
  247. package/dist/web/hooks/use-widget-state.js.map +0 -1
  248. package/dist/web/hooks/use-widget-state.test.js +0 -64
  249. package/dist/web/hooks/use-widget-state.test.js.map +0 -1
  250. package/dist/web/mount-widget.d.ts +0 -1
  251. package/dist/web/mount-widget.js.map +0 -1
  252. package/dist/web/plugin/validate-widget.d.ts +0 -5
  253. package/dist/web/plugin/validate-widget.js +0 -27
  254. package/dist/web/plugin/validate-widget.js.map +0 -1
  255. package/dist/web/plugin/validate-widget.test.js +0 -42
  256. package/dist/web/plugin/validate-widget.test.js.map +0 -1
  257. /package/dist/{test/widget.test.d.ts → cli/tunnel-control-server.test.d.ts} +0 -0
  258. /package/dist/{web/hooks/use-widget-state.test.d.ts → cli/tunnel-handler.test.d.ts} +0 -0
  259. /package/dist/{web/plugin/validate-widget.test.d.ts → cli/tunnel.test.d.ts} +0 -0
@@ -0,0 +1,88 @@
1
+ import { globSync, mkdirSync, readFileSync, writeFileSync } from "node:fs";
2
+ import { basename, dirname, isAbsolute, join, parse, resolve } from "node:path";
3
+ import { hasDefaultExport } from "./validate-view.js";
4
+ export function scanViewsSync(viewsDir) {
5
+ const flatPattern = resolve(viewsDir, "*.{tsx,jsx}");
6
+ const dirPattern = resolve(viewsDir, "*/index.{tsx,jsx}");
7
+ const flatFiles = globSync(flatPattern).map((file) => ({
8
+ name: parse(file).name,
9
+ filePath: file,
10
+ }));
11
+ const dirFiles = globSync(dirPattern).map((file) => ({
12
+ name: basename(dirname(file)),
13
+ filePath: file,
14
+ }));
15
+ // A barrel file like `views/foo/index.tsx` (no default export) must not
16
+ // falsely collide with a sibling `views/foo.tsx`. Drop top-level `index`
17
+ // before splitting valid vs invalid.
18
+ const candidates = [...flatFiles, ...dirFiles].filter((v) => v.name !== "index");
19
+ const valid = [];
20
+ const invalid = [];
21
+ for (const candidate of candidates) {
22
+ const code = readFileSync(candidate.filePath, "utf-8");
23
+ if (hasDefaultExport(code, candidate.filePath)) {
24
+ valid.push(candidate);
25
+ }
26
+ else {
27
+ invalid.push({
28
+ filePath: candidate.filePath,
29
+ });
30
+ }
31
+ }
32
+ return { valid, invalid };
33
+ }
34
+ export function assertUniqueViewNames(views) {
35
+ const nameMap = new Map();
36
+ for (const view of views) {
37
+ const paths = nameMap.get(view.name) ?? [];
38
+ paths.push(view.filePath);
39
+ nameMap.set(view.name, paths);
40
+ }
41
+ for (const [name, paths] of nameMap) {
42
+ if (paths.length > 1) {
43
+ throw new Error(`skybridge: duplicate view name "${name}" resolved from:\n - ${paths.join("\n - ")}\nRename one of the files to avoid the conflict.`);
44
+ }
45
+ }
46
+ }
47
+ export function discoverViewsSync(viewsDir) {
48
+ const { valid } = scanViewsSync(viewsDir);
49
+ assertUniqueViewNames(valid);
50
+ return valid;
51
+ }
52
+ export function generateViewsDts(views) {
53
+ const entries = views.map((v) => ` "${v.name}": true;`).join("\n");
54
+ return [
55
+ "export {};",
56
+ "",
57
+ 'declare module "skybridge/server" {',
58
+ " interface ViewNameRegistry {",
59
+ entries,
60
+ " }",
61
+ "}",
62
+ "",
63
+ ].join("\n");
64
+ }
65
+ export function writeViewsDts(projectRoot, views) {
66
+ const dir = join(projectRoot, ".skybridge");
67
+ mkdirSync(dir, { recursive: true });
68
+ const filePath = join(dir, "views.d.ts");
69
+ const content = generateViewsDts(views);
70
+ try {
71
+ const existing = readFileSync(filePath, "utf-8");
72
+ if (existing === content) {
73
+ return;
74
+ }
75
+ }
76
+ catch {
77
+ // File doesn't exist yet
78
+ }
79
+ writeFileSync(filePath, content, "utf-8");
80
+ }
81
+ export function scanAndWriteViewsDts(projectRoot, viewsDir) {
82
+ const root = projectRoot ?? process.cwd();
83
+ const rawDir = viewsDir ?? "src/views";
84
+ const resolvedDir = isAbsolute(rawDir) ? rawDir : resolve(root, rawDir);
85
+ const views = discoverViewsSync(resolvedDir);
86
+ writeViewsDts(root, views);
87
+ }
88
+ //# sourceMappingURL=scan-views.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scan-views.js","sourceRoot":"","sources":["../../../src/web/plugin/scan-views.ts"],"names":[],"mappings":"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","sourcesContent":["import { globSync, mkdirSync, readFileSync, writeFileSync } from \"node:fs\";\nimport { basename, dirname, isAbsolute, join, parse, resolve } from \"node:path\";\nimport { hasDefaultExport } from \"./validate-view.js\";\n\nexport interface DiscoveredView {\n name: string;\n filePath: string;\n}\n\nexport interface InvalidView {\n filePath: string;\n}\n\nexport function scanViewsSync(viewsDir: string): {\n valid: DiscoveredView[];\n invalid: InvalidView[];\n} {\n const flatPattern = resolve(viewsDir, \"*.{tsx,jsx}\");\n const dirPattern = resolve(viewsDir, \"*/index.{tsx,jsx}\");\n\n const flatFiles = globSync(flatPattern).map((file) => ({\n name: parse(file).name,\n filePath: file,\n }));\n\n const dirFiles = globSync(dirPattern).map((file) => ({\n name: basename(dirname(file)),\n filePath: file,\n }));\n\n // A barrel file like `views/foo/index.tsx` (no default export) must not\n // falsely collide with a sibling `views/foo.tsx`. Drop top-level `index`\n // before splitting valid vs invalid.\n const candidates = [...flatFiles, ...dirFiles].filter(\n (v) => v.name !== \"index\",\n );\n\n const valid: DiscoveredView[] = [];\n const invalid: InvalidView[] = [];\n for (const candidate of candidates) {\n const code = readFileSync(candidate.filePath, \"utf-8\");\n if (hasDefaultExport(code, candidate.filePath)) {\n valid.push(candidate);\n } else {\n invalid.push({\n filePath: candidate.filePath,\n });\n }\n }\n\n return { valid, invalid };\n}\n\nexport function assertUniqueViewNames(views: DiscoveredView[]): void {\n const nameMap = new Map<string, string[]>();\n for (const view of views) {\n const paths = nameMap.get(view.name) ?? [];\n paths.push(view.filePath);\n nameMap.set(view.name, paths);\n }\n\n for (const [name, paths] of nameMap) {\n if (paths.length > 1) {\n throw new Error(\n `skybridge: duplicate view name \"${name}\" resolved from:\\n - ${paths.join(\"\\n - \")}\\nRename one of the files to avoid the conflict.`,\n );\n }\n }\n}\n\nexport function discoverViewsSync(viewsDir: string): DiscoveredView[] {\n const { valid } = scanViewsSync(viewsDir);\n assertUniqueViewNames(valid);\n return valid;\n}\n\nexport function generateViewsDts(views: DiscoveredView[]): string {\n const entries = views.map((v) => ` \"${v.name}\": true;`).join(\"\\n\");\n return [\n \"export {};\",\n \"\",\n 'declare module \"skybridge/server\" {',\n \" interface ViewNameRegistry {\",\n entries,\n \" }\",\n \"}\",\n \"\",\n ].join(\"\\n\");\n}\n\nexport function writeViewsDts(\n projectRoot: string,\n views: DiscoveredView[],\n): void {\n const dir = join(projectRoot, \".skybridge\");\n mkdirSync(dir, { recursive: true });\n\n const filePath = join(dir, \"views.d.ts\");\n const content = generateViewsDts(views);\n\n try {\n const existing = readFileSync(filePath, \"utf-8\");\n if (existing === content) {\n return;\n }\n } catch {\n // File doesn't exist yet\n }\n\n writeFileSync(filePath, content, \"utf-8\");\n}\n\nexport function scanAndWriteViewsDts(\n projectRoot?: string,\n viewsDir?: string,\n): void {\n const root = projectRoot ?? process.cwd();\n const rawDir = viewsDir ?? \"src/views\";\n const resolvedDir = isAbsolute(rawDir) ? rawDir : resolve(root, rawDir);\n\n const views = discoverViewsSync(resolvedDir);\n writeViewsDts(root, views);\n}\n"]}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,99 @@
1
+ import { mkdirSync, mkdtempSync, readFileSync, rmSync, statSync, writeFileSync, } from "node:fs";
2
+ import { tmpdir } from "node:os";
3
+ import { join } from "node:path";
4
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
5
+ import { discoverViewsSync, scanAndWriteViewsDts, scanViewsSync, writeViewsDts, } from "./scan-views.js";
6
+ const DEFAULT_EXPORT = "export default function V() { return null; }";
7
+ describe("discoverViewsSync", () => {
8
+ let root;
9
+ let viewsDir;
10
+ beforeEach(() => {
11
+ root = mkdtempSync(join(tmpdir(), "skybridge-scan-"));
12
+ viewsDir = join(root, "views");
13
+ mkdirSync(viewsDir, { recursive: true });
14
+ });
15
+ afterEach(() => {
16
+ rmSync(root, { recursive: true, force: true });
17
+ });
18
+ it("picks up flat and dir-index views", () => {
19
+ writeFileSync(join(viewsDir, "a.tsx"), DEFAULT_EXPORT);
20
+ mkdirSync(join(viewsDir, "my-view"));
21
+ writeFileSync(join(viewsDir, "my-view/index.tsx"), DEFAULT_EXPORT);
22
+ expect(discoverViewsSync(viewsDir)
23
+ .map((v) => v.name)
24
+ .sort()).toEqual(["a", "my-view"]);
25
+ });
26
+ it("throws on duplicate view names (flat + dir-index collision)", () => {
27
+ writeFileSync(join(viewsDir, "dup.tsx"), DEFAULT_EXPORT);
28
+ mkdirSync(join(viewsDir, "dup"));
29
+ writeFileSync(join(viewsDir, "dup/index.tsx"), DEFAULT_EXPORT);
30
+ expect(() => discoverViewsSync(viewsDir)).toThrow(/duplicate view name "dup"/);
31
+ });
32
+ });
33
+ describe("scanViewsSync", () => {
34
+ let root;
35
+ let viewsDir;
36
+ beforeEach(() => {
37
+ root = mkdtempSync(join(tmpdir(), "skybridge-scan-views-"));
38
+ viewsDir = join(root, "views");
39
+ mkdirSync(viewsDir, { recursive: true });
40
+ });
41
+ afterEach(() => {
42
+ rmSync(root, { recursive: true, force: true });
43
+ });
44
+ it("returns valid and invalid views from a flat layout", () => {
45
+ writeFileSync(join(viewsDir, "ok.tsx"), DEFAULT_EXPORT);
46
+ writeFileSync(join(viewsDir, "broken.tsx"), "export const Foo = () => null;");
47
+ const { valid, invalid } = scanViewsSync(viewsDir);
48
+ expect(valid.map((v) => v.name).sort()).toEqual(["ok"]);
49
+ expect(invalid).toEqual([{ filePath: join(viewsDir, "broken.tsx") }]);
50
+ });
51
+ it("flags an index file in a view dir that lacks a default export", () => {
52
+ mkdirSync(join(viewsDir, "broken"));
53
+ writeFileSync(join(viewsDir, "broken/index.tsx"), "export const Foo = () => null;");
54
+ const { valid, invalid } = scanViewsSync(viewsDir);
55
+ expect(valid).toEqual([]);
56
+ expect(invalid).toEqual([{ filePath: join(viewsDir, "broken/index.tsx") }]);
57
+ });
58
+ it("ignores top-level index.tsx (treated as a barrel, not a view)", () => {
59
+ writeFileSync(join(viewsDir, "index.tsx"), "export const Foo = () => null;");
60
+ const { valid, invalid } = scanViewsSync(viewsDir);
61
+ expect(valid).toEqual([]);
62
+ expect(invalid).toEqual([]);
63
+ });
64
+ });
65
+ describe("writeViewsDts", () => {
66
+ let root;
67
+ beforeEach(() => {
68
+ root = mkdtempSync(join(tmpdir(), "skybridge-dts-"));
69
+ });
70
+ afterEach(() => {
71
+ rmSync(root, { recursive: true, force: true });
72
+ });
73
+ it("is a no-op when content is unchanged", () => {
74
+ const views = [{ name: "a", filePath: "/a.tsx" }];
75
+ writeViewsDts(root, views);
76
+ const dtsPath = join(root, ".skybridge", "views.d.ts");
77
+ const firstMtime = statSync(dtsPath).mtimeMs;
78
+ writeViewsDts(root, views);
79
+ expect(statSync(dtsPath).mtimeMs).toBe(firstMtime);
80
+ });
81
+ });
82
+ describe("scanAndWriteViewsDts", () => {
83
+ let root;
84
+ beforeEach(() => {
85
+ root = mkdtempSync(join(tmpdir(), "skybridge-scan-dts-"));
86
+ mkdirSync(join(root, "src/views"), { recursive: true });
87
+ writeFileSync(join(root, "src/views/hello.tsx"), DEFAULT_EXPORT);
88
+ });
89
+ afterEach(() => {
90
+ rmSync(root, { recursive: true, force: true });
91
+ });
92
+ it("writes a views.d.ts that augments skybridge/server with discovered view names", () => {
93
+ scanAndWriteViewsDts(root);
94
+ const content = readFileSync(join(root, ".skybridge/views.d.ts"), "utf-8");
95
+ expect(content).toContain('declare module "skybridge/server"');
96
+ expect(content).toContain('"hello": true;');
97
+ });
98
+ });
99
+ //# sourceMappingURL=scan-views.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scan-views.test.js","sourceRoot":"","sources":["../../../src/web/plugin/scan-views.test.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n mkdirSync,\n mkdtempSync,\n readFileSync,\n rmSync,\n statSync,\n writeFileSync,\n} from \"node:fs\";\nimport { tmpdir } from \"node:os\";\nimport { join } from \"node:path\";\nimport { afterEach, beforeEach, describe, expect, it } from \"vitest\";\nimport {\n discoverViewsSync,\n scanAndWriteViewsDts,\n scanViewsSync,\n writeViewsDts,\n} from \"./scan-views.js\";\n\nconst DEFAULT_EXPORT = \"export default function V() { return null; }\";\n\ndescribe(\"discoverViewsSync\", () => {\n let root: string;\n let viewsDir: string;\n\n beforeEach(() => {\n root = mkdtempSync(join(tmpdir(), \"skybridge-scan-\"));\n viewsDir = join(root, \"views\");\n mkdirSync(viewsDir, { recursive: true });\n });\n\n afterEach(() => {\n rmSync(root, { recursive: true, force: true });\n });\n\n it(\"picks up flat and dir-index views\", () => {\n writeFileSync(join(viewsDir, \"a.tsx\"), DEFAULT_EXPORT);\n mkdirSync(join(viewsDir, \"my-view\"));\n writeFileSync(join(viewsDir, \"my-view/index.tsx\"), DEFAULT_EXPORT);\n\n expect(\n discoverViewsSync(viewsDir)\n .map((v) => v.name)\n .sort(),\n ).toEqual([\"a\", \"my-view\"]);\n });\n\n it(\"throws on duplicate view names (flat + dir-index collision)\", () => {\n writeFileSync(join(viewsDir, \"dup.tsx\"), DEFAULT_EXPORT);\n mkdirSync(join(viewsDir, \"dup\"));\n writeFileSync(join(viewsDir, \"dup/index.tsx\"), DEFAULT_EXPORT);\n\n expect(() => discoverViewsSync(viewsDir)).toThrow(\n /duplicate view name \"dup\"/,\n );\n });\n});\n\ndescribe(\"scanViewsSync\", () => {\n let root: string;\n let viewsDir: string;\n\n beforeEach(() => {\n root = mkdtempSync(join(tmpdir(), \"skybridge-scan-views-\"));\n viewsDir = join(root, \"views\");\n mkdirSync(viewsDir, { recursive: true });\n });\n\n afterEach(() => {\n rmSync(root, { recursive: true, force: true });\n });\n\n it(\"returns valid and invalid views from a flat layout\", () => {\n writeFileSync(join(viewsDir, \"ok.tsx\"), DEFAULT_EXPORT);\n writeFileSync(\n join(viewsDir, \"broken.tsx\"),\n \"export const Foo = () => null;\",\n );\n\n const { valid, invalid } = scanViewsSync(viewsDir);\n\n expect(valid.map((v) => v.name).sort()).toEqual([\"ok\"]);\n expect(invalid).toEqual([{ filePath: join(viewsDir, \"broken.tsx\") }]);\n });\n\n it(\"flags an index file in a view dir that lacks a default export\", () => {\n mkdirSync(join(viewsDir, \"broken\"));\n writeFileSync(\n join(viewsDir, \"broken/index.tsx\"),\n \"export const Foo = () => null;\",\n );\n\n const { valid, invalid } = scanViewsSync(viewsDir);\n\n expect(valid).toEqual([]);\n expect(invalid).toEqual([{ filePath: join(viewsDir, \"broken/index.tsx\") }]);\n });\n\n it(\"ignores top-level index.tsx (treated as a barrel, not a view)\", () => {\n writeFileSync(\n join(viewsDir, \"index.tsx\"),\n \"export const Foo = () => null;\",\n );\n\n const { valid, invalid } = scanViewsSync(viewsDir);\n\n expect(valid).toEqual([]);\n expect(invalid).toEqual([]);\n });\n});\n\ndescribe(\"writeViewsDts\", () => {\n let root: string;\n\n beforeEach(() => {\n root = mkdtempSync(join(tmpdir(), \"skybridge-dts-\"));\n });\n\n afterEach(() => {\n rmSync(root, { recursive: true, force: true });\n });\n\n it(\"is a no-op when content is unchanged\", () => {\n const views = [{ name: \"a\", filePath: \"/a.tsx\" }];\n writeViewsDts(root, views);\n\n const dtsPath = join(root, \".skybridge\", \"views.d.ts\");\n const firstMtime = statSync(dtsPath).mtimeMs;\n\n writeViewsDts(root, views);\n expect(statSync(dtsPath).mtimeMs).toBe(firstMtime);\n });\n});\n\ndescribe(\"scanAndWriteViewsDts\", () => {\n let root: string;\n\n beforeEach(() => {\n root = mkdtempSync(join(tmpdir(), \"skybridge-scan-dts-\"));\n mkdirSync(join(root, \"src/views\"), { recursive: true });\n writeFileSync(join(root, \"src/views/hello.tsx\"), DEFAULT_EXPORT);\n });\n\n afterEach(() => {\n rmSync(root, { recursive: true, force: true });\n });\n\n it(\"writes a views.d.ts that augments skybridge/server with discovered view names\", () => {\n scanAndWriteViewsDts(root);\n\n const content = readFileSync(join(root, \".skybridge/views.d.ts\"), \"utf-8\");\n expect(content).toContain('declare module \"skybridge/server\"');\n expect(content).toContain('\"hello\": true;');\n });\n});\n"]}
@@ -85,7 +85,7 @@ export const transform = async (code, id) => {
85
85
  sourceFileName: id,
86
86
  };
87
87
  const result = transformSync(code, babelOptions);
88
- if (!result || !result.code) {
88
+ if (!result?.code) {
89
89
  return null;
90
90
  }
91
91
  return {
@@ -1 +1 @@
1
- {"version":3,"file":"transform-data-llm.js","sourceRoot":"","sources":["../../../src/web/plugin/transform-data-llm.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"transform-data-llm.js","sourceRoot":"","sources":["../../../src/web/plugin/transform-data-llm.ts"],"names":[],"mappings":"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","sourcesContent":["import type { PluginObj, TransformOptions, types } from \"@babel/core\";\n\nconst LLM_IMPORT_SOURCE = \"skybridge/web\";\n\ninterface State {\n hasDataLLMImport?: boolean;\n needsDataLLMImport?: boolean;\n}\n\nfunction createBabelPlugin(t: typeof types): PluginObj<State> {\n return {\n name: \"data-llm-babel\",\n\n visitor: {\n Program: {\n enter(path, state) {\n state.hasDataLLMImport = false;\n state.needsDataLLMImport = false;\n\n for (const node of path.node.body) {\n if (!t.isImportDeclaration(node)) {\n continue;\n }\n if (node.source.value !== LLM_IMPORT_SOURCE) {\n continue;\n }\n\n const hasSpecifier = node.specifiers.some(\n (s) =>\n t.isImportSpecifier(s) &&\n t.isIdentifier(s.imported, { name: \"DataLLM\" }),\n );\n\n if (hasSpecifier) {\n state.hasDataLLMImport = true;\n break;\n }\n }\n },\n\n exit(path, state) {\n if (state.needsDataLLMImport && !state.hasDataLLMImport) {\n const importDecl = t.importDeclaration(\n [\n t.importSpecifier(\n t.identifier(\"DataLLM\"),\n t.identifier(\"DataLLM\"),\n ),\n ],\n t.stringLiteral(LLM_IMPORT_SOURCE),\n );\n\n path.node.body.unshift(importDecl);\n }\n },\n },\n\n JSXElement(path, state) {\n const opening = path.node.openingElement;\n const attributes = opening.attributes;\n\n const llmAttributeIndex = attributes.findIndex(\n (attribute) =>\n t.isJSXAttribute(attribute) &&\n t.isJSXIdentifier(attribute.name, { name: \"data-llm\" }),\n );\n\n if (llmAttributeIndex === -1) {\n return;\n }\n\n const llmAttribute = attributes[\n llmAttributeIndex\n ] as types.JSXAttribute;\n\n let contentExpression: types.Expression;\n\n if (t.isStringLiteral(llmAttribute.value)) {\n contentExpression = llmAttribute.value;\n } else if (t.isJSXExpressionContainer(llmAttribute.value)) {\n contentExpression = llmAttribute.value.expression as types.Expression;\n } else {\n return;\n }\n\n const contentAttr = t.jsxAttribute(\n t.jsxIdentifier(\"content\"),\n t.isStringLiteral(contentExpression)\n ? contentExpression\n : t.jsxExpressionContainer(contentExpression),\n );\n\n const filteredAttributes = attributes.filter(\n (_, index) => index !== llmAttributeIndex,\n );\n const newOpening = t.jsxOpeningElement(\n opening.name,\n filteredAttributes,\n opening.selfClosing,\n );\n\n const elementWithoutLlm = t.jsxElement(\n newOpening,\n path.node.closingElement,\n path.node.children,\n path.node.selfClosing,\n );\n\n const llmOpening = t.jsxOpeningElement(t.jsxIdentifier(\"DataLLM\"), [\n contentAttr,\n ]);\n const llmClosing = t.jsxClosingElement(t.jsxIdentifier(\"DataLLM\"));\n\n const wrapped = t.jsxElement(\n llmOpening,\n llmClosing,\n [elementWithoutLlm],\n false,\n );\n\n state.needsDataLLMImport = true;\n path.replaceWith(wrapped);\n },\n },\n };\n}\n\nexport const transform = async (code: string, id: string) => {\n if (!/\\.(jsx|tsx)$/.test(id)) {\n return null;\n }\n\n if (id.includes(\"node_modules\")) {\n return null;\n }\n\n // Dynamic import to ensure @babel/core is only loaded in Node.js context\n const { types: t, transformSync } = await import(\"@babel/core\");\n\n const babelOptions: TransformOptions = {\n plugins: [createBabelPlugin(t)],\n parserOpts: {\n plugins: [\"jsx\", \"typescript\"],\n },\n filename: id,\n sourceFileName: id,\n };\n\n const result = transformSync(code, babelOptions);\n\n if (!result?.code) {\n return null;\n }\n\n return {\n code: result.code,\n map: result.map || null,\n };\n};\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"transform-data-llm.test.js","sourceRoot":"","sources":["../../../src/web/plugin/transform-data-llm.test.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"transform-data-llm.test.js","sourceRoot":"","sources":["../../../src/web/plugin/transform-data-llm.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { describe, expect, it } from \"vitest\";\nimport { transform } from \"./transform-data-llm.js\";\n\ndescribe(\"data-llm plugin\", () => {\n it(\"should transform JSX element with data-llm string attribute\", async () => {\n const code = `\n function Component() {\n return <div data-llm=\"Test description\">Content</div>;\n }\n `;\n\n const result = await transform(code, \"test.tsx\");\n\n expect(result).not.toBeNull();\n expect(result?.code).toContain(\"DataLLM\");\n expect(result?.code).toContain('content=\"Test description\"');\n expect(result?.code).not.toContain(\"data-llm\");\n });\n\n it(\"should transform JSX element with data-llm expression attribute\", async () => {\n const code = `\n function Component() {\n const desc = \"Dynamic description\";\n return <div data-llm={desc}>Content</div>;\n }\n `;\n\n const result = await transform(code, \"test.tsx\");\n\n expect(result).not.toBeNull();\n expect(result?.code).toContain(\"DataLLM\");\n expect(result?.code).toContain(\"content={desc}\");\n expect(result?.code).not.toContain(\"data-llm\");\n });\n\n it(\"should add import for DataLLM when not present\", async () => {\n const code = `\n function Component() {\n return <div data-llm=\"Test\">Content</div>;\n }\n `;\n\n const result = await transform(code, \"test.tsx\");\n\n expect(result).not.toBeNull();\n expect(result?.code).toContain('import { DataLLM } from \"skybridge/web\"');\n });\n\n it(\"should handle DataLLM imports correctly\", async () => {\n // No duplicate import\n const codeWithImport = `\n import { DataLLM } from \"skybridge/web\";\n function Component() {\n return <div data-llm=\"Test\">Content</div>;\n }\n `;\n const result1 = await transform(codeWithImport, \"test.tsx\");\n expect(\n result1?.code.match(/import.*DataLLM.*from.*skybridge\\/web/g),\n ).toHaveLength(1);\n\n // Preserve other imports and add missing DataLLM\n const codeWithOthers = `\n import React from \"react\";\n import { useState } from \"react\";\n function Component() {\n return <div data-llm=\"Test\">Content</div>;\n }\n `;\n const result2 = await transform(codeWithOthers, \"test.tsx\");\n expect(result2?.code).toContain('import React from \"react\"');\n expect(result2?.code).toContain('import { useState } from \"react\"');\n expect(result2?.code).toContain('import { DataLLM } from \"skybridge/web\"');\n });\n\n it(\"should handle complex JSX with multiple data-llm attributes\", async () => {\n const code = `\n function Component() {\n return (\n <div>\n <section data-llm=\"Section 1\">\n <p>Content 1</p>\n </section>\n <section data-llm=\"Section 2\">\n <p>Content 2</p>\n </section>\n </div>\n );\n }\n `;\n\n const result = await transform(code, \"test.tsx\");\n\n expect(result).toMatchSnapshot();\n });\n});\n"]}
@@ -0,0 +1 @@
1
+ export declare function hasDefaultExport(code: string, _filePath?: string): boolean;
@@ -0,0 +1,9 @@
1
+ function stripComments(code) {
2
+ return code.replace(/\/\/.*$/gm, "").replace(/\/\*[\s\S]*?\*\//g, "");
3
+ }
4
+ export function hasDefaultExport(code, _filePath) {
5
+ const stripped = stripComments(code);
6
+ return (/export\s+default\s/.test(stripped) ||
7
+ /export\s*\{[^}]*\bas\s+default\b[^}]*}/.test(stripped));
8
+ }
9
+ //# sourceMappingURL=validate-view.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate-view.js","sourceRoot":"","sources":["../../../src/web/plugin/validate-view.ts"],"names":[],"mappings":"AAAA,SAAS,aAAa,CAAC,IAAY;IACjC,OAAO,IAAI,CAAC,OAAO,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,mBAAmB,EAAE,EAAE,CAAC,CAAC;AACxE,CAAC;AAED,MAAM,UAAU,gBAAgB,CAAC,IAAY,EAAE,SAAkB;IAC/D,MAAM,QAAQ,GAAG,aAAa,CAAC,IAAI,CAAC,CAAC;IACrC,OAAO,CACL,oBAAoB,CAAC,IAAI,CAAC,QAAQ,CAAC;QACnC,wCAAwC,CAAC,IAAI,CAAC,QAAQ,CAAC,CACxD,CAAC;AACJ,CAAC","sourcesContent":["function stripComments(code: string): string {\n return code.replace(/\\/\\/.*$/gm, \"\").replace(/\\/\\*[\\s\\S]*?\\*\\//g, \"\");\n}\n\nexport function hasDefaultExport(code: string, _filePath?: string): boolean {\n const stripped = stripComments(code);\n return (\n /export\\s+default\\s/.test(stripped) ||\n /export\\s*\\{[^}]*\\bas\\s+default\\b[^}]*}/.test(stripped)\n );\n}\n"]}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,24 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import { hasDefaultExport } from "./validate-view.js";
3
+ describe("hasDefaultExport", () => {
4
+ it("detects export default declaration", () => {
5
+ expect(hasDefaultExport("export default MyView;")).toBe(true);
6
+ });
7
+ it("detects export default function", () => {
8
+ expect(hasDefaultExport("export default function MyView() {}")).toBe(true);
9
+ });
10
+ it("detects re-export as default", () => {
11
+ expect(hasDefaultExport("export { Foo as default };")).toBe(true);
12
+ });
13
+ it("returns false when no default export", () => {
14
+ expect(hasDefaultExport("export const Foo = 1;")).toBe(false);
15
+ });
16
+ it("ignores commented-out default exports", () => {
17
+ const code = `
18
+ // export default MyView;
19
+ /* export default MyView; */
20
+ `;
21
+ expect(hasDefaultExport(code)).toBe(false);
22
+ });
23
+ });
24
+ //# sourceMappingURL=validate-view.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validate-view.test.js","sourceRoot":"","sources":["../../../src/web/plugin/validate-view.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAE,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;AAEtD,QAAQ,CAAC,kBAAkB,EAAE,GAAG,EAAE;IAChC,EAAE,CAAC,oCAAoC,EAAE,GAAG,EAAE;QAC5C,MAAM,CAAC,gBAAgB,CAAC,wBAAwB,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAChE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,iCAAiC,EAAE,GAAG,EAAE;QACzC,MAAM,CAAC,gBAAgB,CAAC,qCAAqC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC7E,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8BAA8B,EAAE,GAAG,EAAE;QACtC,MAAM,CAAC,gBAAgB,CAAC,4BAA4B,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACpE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,sCAAsC,EAAE,GAAG,EAAE;QAC9C,MAAM,CAAC,gBAAgB,CAAC,uBAAuB,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAChE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,uCAAuC,EAAE,GAAG,EAAE;QAC/C,MAAM,IAAI,GAAG;;;KAGZ,CAAC;QACF,MAAM,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC7C,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC","sourcesContent":["import { describe, expect, it } from \"vitest\";\nimport { hasDefaultExport } from \"./validate-view.js\";\n\ndescribe(\"hasDefaultExport\", () => {\n it(\"detects export default declaration\", () => {\n expect(hasDefaultExport(\"export default MyView;\")).toBe(true);\n });\n\n it(\"detects export default function\", () => {\n expect(hasDefaultExport(\"export default function MyView() {}\")).toBe(true);\n });\n\n it(\"detects re-export as default\", () => {\n expect(hasDefaultExport(\"export { Foo as default };\")).toBe(true);\n });\n\n it(\"returns false when no default export\", () => {\n expect(hasDefaultExport(\"export const Foo = 1;\")).toBe(false);\n });\n\n it(\"ignores commented-out default exports\", () => {\n const code = `\n // export default MyView;\n /* export default MyView; */\n `;\n expect(hasDefaultExport(code)).toBe(false);\n });\n});\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"proxy.js","sourceRoot":"","sources":["../../src/web/proxy.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"proxy.js","sourceRoot":"","sources":["../../src/web/proxy.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,GAAG;IACb,KAAK,EAAE,SAAS;IAChB,IAAI,EAAE,SAAS;IACf,OAAO,EAAE,SAAS;IAClB,KAAK,EAAE,SAAS;CACR,CAAC;AAEX,MAAM,UAAU,yBAAyB;IACvC,IAAI,OAAO,MAAM,KAAK,WAAW,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC;QACpD,OAAO;IACT,CAAC;IAED,MAAM,UAAU,GAAG,MAAM,CAAC,wBAAwB,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;IACrE,IAAI,UAAU,EAAE,YAAY,KAAK,KAAK,IAAI,UAAU,EAAE,QAAQ,KAAK,KAAK,EAAE,CAAC;QACzE,OAAO,CAAC,IAAI,CACV,2FAA2F,CAC5F,CAAC;QACF,OAAO;IACT,CAAC;IAED,MAAM,cAAc,GAAG,MAAM,CAAC,MAAM,CAAC;IAErC,MAAM,OAAO,GAAwC;QACnD,GAAG,CAAC,MAAM,EAAE,IAAI,EAAE,QAAQ;YACxB,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,MAAM,EAAE,IAAI,EAAE,QAAQ,CAAC,CAAC;YAElD,IAAI,OAAO,KAAK,KAAK,UAAU,EAAE,CAAC;gBAChC,OAAO,KAAK,CAAC;YACf,CAAC;YAED,OAAO,CAAC,GAAG,IAAe,EAAE,EAAE;gBAC5B,MAAM,UAAU,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC;gBAEhC,OAAO,CAAC,KAAK,CACX,yBAAyB,UAAU,EAAE,EACrC,UAAU,MAAM,CAAC,KAAK,uBAAuB,EAC7C,UAAU,MAAM,CAAC,IAAI,uBAAuB,EAC5C,UAAU,MAAM,CAAC,OAAO,EAAE,CAC3B,CAAC;gBACF,OAAO,CAAC,GAAG,CAAC,WAAW,EAAE,UAAU,MAAM,CAAC,IAAI,EAAE,EAAE,IAAI,CAAC,CAAC;gBAExD,MAAM,MAAM,GAAG,KAAK,CAAC,KAAK,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;gBAEzC,IAAI,MAAM,IAAI,OAAO,MAAM,CAAC,IAAI,KAAK,UAAU,EAAE,CAAC;oBAChD,OAAO,MAAM,CAAC,IAAI,CAChB,CAAC,QAAiB,EAAE,EAAE;wBACpB,OAAO,CAAC,GAAG,CACT,eAAe,EACf,UAAU,MAAM,CAAC,OAAO,EAAE,EAC1B,QAAQ,CACT,CAAC;wBACF,OAAO,CAAC,QAAQ,EAAE,CAAC;wBACnB,OAAO,QAAQ,CAAC;oBAClB,CAAC,EACD,CAAC,KAAc,EAAE,EAAE;wBACjB,OAAO,CAAC,KAAK,CAAC,eAAe,EAAE,UAAU,MAAM,CAAC,KAAK,EAAE,EAAE,KAAK,CAAC,CAAC;wBAChE,OAAO,CAAC,QAAQ,EAAE,CAAC;wBACnB,MAAM,KAAK,CAAC;oBACd,CAAC,CACF,CAAC;gBACJ,CAAC;gBAED,OAAO,CAAC,GAAG,CAAC,eAAe,EAAE,UAAU,MAAM,CAAC,OAAO,EAAE,EAAE,MAAM,CAAC,CAAC;gBACjE,OAAO,CAAC,QAAQ,EAAE,CAAC;gBAEnB,OAAO,MAAM,CAAC;YAChB,CAAC,CAAC;QACJ,CAAC;QAED,GAAG,CAAC,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,QAAQ;YAC/B,OAAO,CAAC,GAAG,CACT,yBAAyB,MAAM,CAAC,IAAI,CAAC,EAAE,EACvC,UAAU,MAAM,CAAC,KAAK,EAAE,EACxB,UAAU,MAAM,CAAC,IAAI,EAAE,EACvB,UAAU,MAAM,CAAC,OAAO,qBAAqB,EAC7C,GAAG,EACH,KAAK,CACN,CAAC;YAEF,OAAO,OAAO,CAAC,GAAG,CAAC,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,QAAQ,CAAC,CAAC;QACpD,CAAC;KACF,CAAC;IAEF,MAAM,CAAC,MAAM,GAAG,IAAI,KAAK,CAAC,cAAc,EAAE,OAAO,CAAC,CAAC;IAEnD,OAAO,CAAC,GAAG,CACT,8DAA8D,EAC9D,UAAU,MAAM,CAAC,KAAK,EAAE,EACxB,UAAU,MAAM,CAAC,IAAI,EAAE,CACxB,CAAC;AACJ,CAAC","sourcesContent":["const colors = {\n brand: \"#6366f1\",\n info: \"#22223b\",\n success: \"#22c55e\",\n error: \"#ef4444\",\n} as const;\n\nexport function installOpenAILoggingProxy() {\n if (typeof window === \"undefined\" || !window.openai) {\n return;\n }\n\n const descriptor = Object.getOwnPropertyDescriptor(window, \"openai\");\n if (descriptor?.configurable === false || descriptor?.writable === false) {\n console.warn(\n \"[openai-proxy] window.openai is not configurable or writable, skipping proxy installation\",\n );\n return;\n }\n\n const originalOpenAI = window.openai;\n\n const handler: ProxyHandler<typeof originalOpenAI> = {\n get(target, prop, receiver) {\n const value = Reflect.get(target, prop, receiver);\n\n if (typeof value !== \"function\") {\n return value;\n }\n\n return (...args: unknown[]) => {\n const methodName = String(prop);\n\n console.group(\n `%c[openai] %cmethod %c${methodName}`,\n `color: ${colors.brand}; font-weight: normal`,\n `color: ${colors.info}; font-weight: normal`,\n `color: ${colors.success}`,\n );\n console.log(\"%c← args:\", `color: ${colors.info}`, args);\n\n const result = value.apply(target, args);\n\n if (result && typeof result.then === \"function\") {\n return result.then(\n (resolved: unknown) => {\n console.log(\n \"%c→ resolved:\",\n `color: ${colors.success}`,\n resolved,\n );\n console.groupEnd();\n return resolved;\n },\n (error: unknown) => {\n console.error(\"%c→ rejected:\", `color: ${colors.error}`, error);\n console.groupEnd();\n throw error;\n },\n );\n }\n\n console.log(\"%c→ returned:\", `color: ${colors.success}`, result);\n console.groupEnd();\n\n return result;\n };\n },\n\n set(target, prop, value, receiver) {\n console.log(\n `%c[openai] %cupdate %c${String(prop)}`,\n `color: ${colors.brand}`,\n `color: ${colors.info}`,\n `color: ${colors.success}; font-weight: bold`,\n \"←\",\n value,\n );\n\n return Reflect.set(target, prop, value, receiver);\n },\n };\n\n window.openai = new Proxy(originalOpenAI, handler);\n\n console.log(\n \"%c[openai-proxy] %cInstalled logging proxy for window.openai\",\n `color: ${colors.brand}`,\n `color: ${colors.info}`,\n );\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/web/types.ts"],"names":[],"mappings":"AAAA,OAAO,OAAO,CAAC"}
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/web/types.ts"],"names":[],"mappings":"AAAA,OAAO,OAAO,CAAC","sourcesContent":["import \"react\";\n\ndeclare module \"react\" {\n interface HTMLAttributes<T> {\n \"data-llm\"?: string;\n }\n}\n\nexport type UnknownObject = Record<string, unknown>;\n\nexport type Prettify<T> = { [K in keyof T]: T[K] } & {};\nexport type Objectify<T> = T & UnknownObject;\n\ntype RequiredKeys<T> = {\n [K in keyof T]-?: Record<string, never> extends Pick<T, K> ? never : K;\n}[keyof T];\nexport type HasRequiredKeys<T> = RequiredKeys<T> extends never ? false : true;\n"]}
package/package.json CHANGED
@@ -1,12 +1,15 @@
1
1
  {
2
2
  "name": "skybridge",
3
- "version": "0.0.0-dev.f89b570",
3
+ "version": "0.0.0-dev.f8d2f8c",
4
4
  "description": "Skybridge is a framework for building ChatGPT and MCP Apps",
5
5
  "repository": {
6
6
  "type": "git",
7
7
  "url": "https://github.com/alpic-ai/skybridge.git"
8
8
  },
9
9
  "type": "module",
10
+ "engines": {
11
+ "node": ">=22.0.0"
12
+ },
10
13
  "files": [
11
14
  "dist",
12
15
  "README.md",
@@ -22,6 +25,10 @@
22
25
  "./web": {
23
26
  "types": "./dist/web/index.d.ts",
24
27
  "default": "./dist/web/index.js"
28
+ },
29
+ "./vite": {
30
+ "types": "./dist/web/plugin/plugin.d.ts",
31
+ "default": "./dist/web/plugin/plugin.js"
25
32
  }
26
33
  },
27
34
  "keywords": [
@@ -33,46 +40,50 @@
33
40
  "author": "Frédéric Barthelet",
34
41
  "license": "ISC",
35
42
  "peerDependencies": {
36
- "@modelcontextprotocol/sdk": ">=1.0.0",
37
- "@skybridge/devtools": ">=0.26.1",
43
+ "@modelcontextprotocol/sdk": ">=1.27.0",
44
+ "@skybridge/devtools": ">=0.35.14 <1.0.0",
38
45
  "nodemon": ">=3.0.0",
39
46
  "react": ">=18.0.0",
40
47
  "react-dom": ">=18.0.0",
41
- "vite": "^7.3.1"
48
+ "vite": ">=7.3.1"
42
49
  },
43
50
  "dependencies": {
44
51
  "@babel/core": "^7.29.0",
45
- "@oclif/core": "^4.8.3",
52
+ "@modelcontextprotocol/ext-apps": "^1.3.2",
53
+ "@oclif/core": "^4.10.3",
46
54
  "ci-info": "^4.4.0",
47
55
  "cors": "^2.8.6",
56
+ "cross-spawn": "^7.0.6",
48
57
  "dequal": "^2.0.3",
49
58
  "es-toolkit": "^1.45.1",
50
59
  "express": "^5.2.1",
51
- "handlebars": "^4.7.8",
52
- "ink": "^6.8.0",
53
- "posthog-node": "^5.28.0",
60
+ "handlebars": "^4.7.9",
61
+ "ink": "^7.0.0",
62
+ "open": "^11.0.0",
63
+ "posthog-node": "^5.28.9",
54
64
  "superjson": "^2.2.6",
55
- "zustand": "^5.0.11"
65
+ "zustand": "^5.0.12"
56
66
  },
57
67
  "devDependencies": {
58
- "@modelcontextprotocol/ext-apps": "^1.2.0",
59
- "@modelcontextprotocol/sdk": "^1.27.1",
68
+ "@modelcontextprotocol/sdk": "^1.29.0",
60
69
  "@testing-library/dom": "^10.4.1",
61
70
  "@testing-library/react": "^16.3.2",
62
71
  "@total-typescript/tsconfig": "^1.0.4",
63
72
  "@types/babel__core": "^7.20.5",
64
73
  "@types/cors": "^2.8.19",
74
+ "@types/cross-spawn": "^6.0.6",
65
75
  "@types/express": "^5.0.6",
66
- "@types/jsdom": "^28.0.0",
76
+ "@types/jsdom": "^28.0.1",
67
77
  "@types/node": "^24.12.0",
68
78
  "@types/react": "^19.2.14",
69
79
  "@types/react-dom": "^19.2.3",
70
- "@vitest/ui": "^4.0.18",
71
- "jsdom": "^28.1.0",
80
+ "@vitest/ui": "^4.1.4",
81
+ "jsdom": "^29.0.1",
72
82
  "shx": "^0.4.0",
73
83
  "ts-node": "^10.9.2",
74
- "typescript": "^5.9.3",
75
- "vitest": "^4.0.18",
84
+ "typescript": "^6.0.2",
85
+ "vite": "^8.0.0",
86
+ "vitest": "^4.1.4",
76
87
  "zod": "^4.3.6"
77
88
  },
78
89
  "bin": {
@@ -89,12 +100,11 @@
89
100
  }
90
101
  },
91
102
  "scripts": {
92
- "build": "shx rm -rf dist && tsc && pnpm run build:templates",
93
- "build:templates": "cp -r src/server/templates dist/server/",
103
+ "build": "shx rm -rf dist && pnpm run build:templates && tsc",
104
+ "build:templates": "node scripts/precompile-templates.mjs",
94
105
  "format": "biome check --write --error-on-warnings",
95
- "test": "pnpm run test:unit && pnpm run test:type && pnpm run test:format",
96
- "test:unit": "vitest run",
97
- "test:type": "tsc --noEmit",
106
+ "test": "pnpm run test:unit && pnpm run test:format",
107
+ "test:unit": "pnpm run build:templates && vitest run",
98
108
  "test:format": "biome ci"
99
109
  }
100
110
  }
@@ -6,6 +6,8 @@
6
6
  "verbatimModuleSyntax": true,
7
7
 
8
8
  // Output options
9
+ "rootDir": "${configDir}/src",
10
+ "outDir": "${configDir}/dist",
9
11
  "noEmit": false,
10
12
  "incremental": true,
11
13
  "sourceMap": true,
@@ -15,6 +17,9 @@
15
17
  "lib": ["ES2022", "DOM", "DOM.Iterable"],
16
18
  "jsx": "react-jsx",
17
19
 
20
+ // Type declarations
21
+ "types": ["node", "vite/client"],
22
+
18
23
  // Strictness and checking options
19
24
  "strict": true,
20
25
  "skipLibCheck": true,
@@ -1 +0,0 @@
1
- export declare const DEFAULT_HMR_PORT = 24678;
@@ -1,2 +0,0 @@
1
- export const DEFAULT_HMR_PORT = 24678;
2
- //# sourceMappingURL=const.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"const.js","sourceRoot":"","sources":["../../src/server/const.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,gBAAgB,GAAG,KAAK,CAAC"}
@@ -1,67 +0,0 @@
1
- <script type="module">window.skybridge = { hostType: "{{hostType}}", serverUrl: "{{serverUrl}}" };</script>
2
- <script type="module">
3
- import { injectIntoGlobalHook } from "{{serverUrl}}/assets/@react-refresh";
4
- injectIntoGlobalHook(window); window.$RefreshReg$ = () => {};
5
- window.$RefreshSig$ = () => (type) => type;
6
- window.__vite_plugin_react_preamble_installed__ = true;
7
- </script>
8
- <script type="module" src="{{serverUrl}}/@vite/client"></script>
9
- {{#if useLocalNetworkAccess}}
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
- {{/if}}
64
- <div id="root"></div>
65
- <script type="module" id="dev-widget-entry">
66
- import('{{serverUrl}}/src/widgets/{{widgetName}}');
67
- </script>
@@ -1,6 +0,0 @@
1
- <script type="module">window.skybridge = { hostType: "{{hostType}}", serverUrl: "{{serverUrl}}" };</script>
2
- <div id="root"></div>
3
- <script type="module">
4
- import('{{serverUrl}}/assets/{{widgetFile}}');
5
- </script>
6
- <link rel="stylesheet" crossorigin href="{{serverUrl}}/assets/{{styleFile}}" />
@@ -1,12 +0,0 @@
1
- import { type Router } from "express";
2
- /**
3
- * Install Vite dev server
4
- * This router MUST be installed at the application root, like so:
5
- *
6
- * const app = express();
7
- *
8
- * if (env.NODE_ENV !== "production") {
9
- * app.use(await widgetsRouter());
10
- * }
11
- */
12
- export declare const widgetsDevServer: () => Promise<Router>;