skybridge 0.0.0-dev.f381d90 → 0.0.0-dev.f391982

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 (95) hide show
  1. package/dist/cli/header.js +1 -1
  2. package/dist/cli/header.js.map +1 -1
  3. package/dist/cli/use-nodemon.js +2 -2
  4. package/dist/cli/use-nodemon.js.map +1 -1
  5. package/dist/cli/use-tunnel.d.ts +12 -5
  6. package/dist/cli/use-tunnel.js +34 -29
  7. package/dist/cli/use-tunnel.js.map +1 -1
  8. package/dist/cli/use-typescript-check.d.ts +1 -0
  9. package/dist/cli/use-typescript-check.js +41 -6
  10. package/dist/cli/use-typescript-check.js.map +1 -1
  11. package/dist/commands/build.js +28 -7
  12. package/dist/commands/build.js.map +1 -1
  13. package/dist/commands/dev.d.ts +1 -0
  14. package/dist/commands/dev.js +7 -2
  15. package/dist/commands/dev.js.map +1 -1
  16. package/dist/commands/start.js +7 -10
  17. package/dist/commands/start.js.map +1 -1
  18. package/dist/server/content-helpers.d.ts +27 -0
  19. package/dist/server/content-helpers.js +46 -0
  20. package/dist/server/content-helpers.js.map +1 -0
  21. package/dist/server/content-helpers.test.js +70 -0
  22. package/dist/server/content-helpers.test.js.map +1 -0
  23. package/dist/server/express.js +3 -3
  24. package/dist/server/express.js.map +1 -1
  25. package/dist/server/express.test.js +39 -2
  26. package/dist/server/express.test.js.map +1 -1
  27. package/dist/server/index.d.ts +4 -3
  28. package/dist/server/index.js +3 -2
  29. package/dist/server/index.js.map +1 -1
  30. package/dist/server/middleware.test.js +12 -9
  31. package/dist/server/middleware.test.js.map +1 -1
  32. package/dist/server/server.d.ts +95 -72
  33. package/dist/server/server.js +195 -73
  34. package/dist/server/server.js.map +1 -1
  35. package/dist/server/templateHelper.d.ts +5 -5
  36. package/dist/server/templates/development.hbs +2 -2
  37. package/dist/server/templates/production.hbs +1 -1
  38. package/dist/server/viewsDevServer.d.ts +14 -0
  39. package/dist/server/viewsDevServer.js +45 -0
  40. package/dist/server/viewsDevServer.js.map +1 -0
  41. package/dist/test/utils.d.ts +13 -21
  42. package/dist/test/utils.js +42 -37
  43. package/dist/test/utils.js.map +1 -1
  44. package/dist/test/view.test.js +523 -0
  45. package/dist/test/view.test.js.map +1 -0
  46. package/dist/web/bridges/apps-sdk/adaptor.d.ts +4 -2
  47. package/dist/web/bridges/apps-sdk/adaptor.js +24 -12
  48. package/dist/web/bridges/apps-sdk/adaptor.js.map +1 -1
  49. package/dist/web/bridges/apps-sdk/types.d.ts +10 -5
  50. package/dist/web/bridges/apps-sdk/types.js.map +1 -1
  51. package/dist/web/bridges/mcp-app/adaptor.d.ts +6 -0
  52. package/dist/web/bridges/mcp-app/adaptor.js +6 -0
  53. package/dist/web/bridges/mcp-app/adaptor.js.map +1 -1
  54. package/dist/web/bridges/types.d.ts +9 -3
  55. package/dist/web/generate-helpers.d.ts +20 -18
  56. package/dist/web/generate-helpers.js +20 -18
  57. package/dist/web/generate-helpers.js.map +1 -1
  58. package/dist/web/generate-helpers.test-d.js +20 -20
  59. package/dist/web/generate-helpers.test-d.js.map +1 -1
  60. package/dist/web/hooks/use-files.d.ts +2 -1
  61. package/dist/web/hooks/use-files.js +1 -0
  62. package/dist/web/hooks/use-files.js.map +1 -1
  63. package/dist/web/hooks/use-files.test.js +22 -2
  64. package/dist/web/hooks/use-files.test.js.map +1 -1
  65. package/dist/web/index.d.ts +1 -0
  66. package/dist/web/index.js.map +1 -1
  67. package/dist/web/plugin/plugin.d.ts +4 -1
  68. package/dist/web/plugin/plugin.js +112 -25
  69. package/dist/web/plugin/plugin.js.map +1 -1
  70. package/dist/web/plugin/scan-views.d.ts +8 -0
  71. package/dist/web/plugin/scan-views.js +70 -0
  72. package/dist/web/plugin/scan-views.js.map +1 -0
  73. package/dist/web/plugin/scan-views.test.d.ts +1 -0
  74. package/dist/web/plugin/scan-views.test.js +67 -0
  75. package/dist/web/plugin/scan-views.test.js.map +1 -0
  76. package/dist/web/plugin/validate-view.d.ts +1 -0
  77. package/dist/web/plugin/validate-view.js +9 -0
  78. package/dist/web/plugin/validate-view.js.map +1 -0
  79. package/dist/web/plugin/validate-view.test.d.ts +1 -0
  80. package/dist/web/plugin/validate-view.test.js +24 -0
  81. package/dist/web/plugin/validate-view.test.js.map +1 -0
  82. package/package.json +6 -3
  83. package/tsconfig.base.json +2 -0
  84. package/dist/server/widgetsDevServer.d.ts +0 -13
  85. package/dist/server/widgetsDevServer.js +0 -52
  86. package/dist/server/widgetsDevServer.js.map +0 -1
  87. package/dist/test/widget.test.js +0 -303
  88. package/dist/test/widget.test.js.map +0 -1
  89. package/dist/web/plugin/validate-widget.d.ts +0 -5
  90. package/dist/web/plugin/validate-widget.js +0 -27
  91. package/dist/web/plugin/validate-widget.js.map +0 -1
  92. package/dist/web/plugin/validate-widget.test.js +0 -42
  93. package/dist/web/plugin/validate-widget.test.js.map +0 -1
  94. /package/dist/{test/widget.test.d.ts → server/content-helpers.test.d.ts} +0 -0
  95. /package/dist/{web/plugin/validate-widget.test.d.ts → test/view.test.d.ts} +0 -0
@@ -1,29 +1,51 @@
1
+ import { isAbsolute, resolve } from "node:path";
2
+ import { discoverViewsSync, writeViewsDts, } from "./scan-views.js";
1
3
  import { transform as dataLlmTransform } from "./transform-data-llm.js";
2
- import { validateWidget } from "./validate-widget.js";
3
- // Matches widget entry files (e.g. src/widgets/foo.tsx, src/widgets/foo/index.tsx) with optional Vite query strings
4
- const WIDGET_ENTRY_RE = /\/src\/widgets\/(?:[^/]+\.(?:jsx|tsx)|[^/]+\/index\.tsx)(?:\?.*)?$/;
5
- export function skybridge() {
4
+ import { hasDefaultExport } from "./validate-view.js";
5
+ const VIRTUAL_PREFIX = "/_skybridge/view/";
6
+ const VIRTUAL_MODULE_PREFIX = "\0skybridge:view:";
7
+ function buildVirtualEntry(viewFilePath) {
8
+ const normalized = viewFilePath.replace(/\\/g, "/");
9
+ return [
10
+ `import { mountWidget } from "skybridge/web";`,
11
+ `import Component from "${normalized}";`,
12
+ `import { createElement } from "react";`,
13
+ `mountWidget(createElement(Component));`,
14
+ ].join("\n");
15
+ }
16
+ function getViewEntryPattern(viewsDir) {
17
+ const escaped = viewsDir.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
18
+ return new RegExp(`${escaped}\\/(?:[^/]+\\.(?:jsx|tsx)|[^/]+\\/index\\.(?:tsx|jsx))(?:\\?.*)?$`);
19
+ }
20
+ export function skybridge(options) {
21
+ const rawViewsDir = options?.viewsDir ?? "src/views";
22
+ let resolvedViewsDir;
23
+ let projectRoot;
24
+ let viewMap = new Map();
25
+ let viewEntryPattern;
6
26
  return {
7
27
  name: "skybridge",
8
- async config(config) {
9
- // Dynamic imports to ensure Node modules are only loaded in Node.js context
10
- const { globSync } = await import("node:fs");
11
- const { basename, dirname, parse, resolve } = await import("node:path");
12
- const projectRoot = config.root || process.cwd();
13
- const flatWidgetPattern = resolve(projectRoot, "src/widgets/*.{js,ts,jsx,tsx,html}");
14
- const dirWidgetPattern = resolve(projectRoot, "src/widgets/*/index.tsx");
15
- const flatWidgets = globSync(flatWidgetPattern).map((file) => {
16
- const name = parse(file).name;
17
- return [name, file];
18
- });
19
- const dirWidgets = globSync(dirWidgetPattern).map((file) => {
20
- const name = basename(dirname(file));
21
- return [name, file];
22
- });
23
- const input = Object.fromEntries([...flatWidgets, ...dirWidgets]);
28
+ enforce: "pre",
29
+ // Read by `skybridge build` to resolve viewsDir before `tsc -b` runs.
30
+ api: { viewsDir: rawViewsDir },
31
+ config(config) {
32
+ projectRoot = config.root || process.cwd();
33
+ resolvedViewsDir = isAbsolute(rawViewsDir)
34
+ ? rawViewsDir
35
+ : resolve(projectRoot, rawViewsDir);
36
+ viewEntryPattern = getViewEntryPattern(resolvedViewsDir);
37
+ const views = discoverViewsSync(resolvedViewsDir);
38
+ viewMap = new Map(views.map((v) => [v.name, v]));
39
+ writeViewsDts(projectRoot, views);
40
+ const input = {};
41
+ for (const view of views) {
42
+ input[view.name] = `${VIRTUAL_PREFIX}${view.name}`;
43
+ }
24
44
  return {
25
45
  base: "/assets",
26
46
  build: {
47
+ outDir: "dist/assets",
48
+ emptyOutDir: true,
27
49
  manifest: true,
28
50
  minify: true,
29
51
  cssCodeSplit: false,
@@ -31,6 +53,27 @@ export function skybridge() {
31
53
  input,
32
54
  },
33
55
  },
56
+ // Pre-bundle view deps at startup so the first tool invocation
57
+ // doesn't hit Vite's on-demand re-optimization path (which sends
58
+ // `full-reload` over HMR — in our iframe flow the parent host
59
+ // can't honour a reload, and the view silently never mounts).
60
+ optimizeDeps: {
61
+ // Scan view files so transitive user deps (zod, tailwind, etc.)
62
+ // get pre-bundled at startup.
63
+ entries: [
64
+ `${resolvedViewsDir}/*.{tsx,jsx}`,
65
+ `${resolvedViewsDir}/*/index.{tsx,jsx}`,
66
+ ],
67
+ // Framework deps imported by the synthesized virtual entry.
68
+ // The scanner can't see the virtual module source, so we must
69
+ // list these explicitly.
70
+ include: [
71
+ "react",
72
+ "react-dom/client",
73
+ "react/jsx-runtime",
74
+ "skybridge/web",
75
+ ],
76
+ },
34
77
  experimental: {
35
78
  renderBuiltUrl: (filename) => {
36
79
  return {
@@ -40,12 +83,56 @@ export function skybridge() {
40
83
  },
41
84
  };
42
85
  },
43
- enforce: "pre",
44
- async transform(code, id) {
45
- if (WIDGET_ENTRY_RE.test(id)) {
46
- for (const warning of validateWidget(code, id)) {
47
- this.warn(warning.message);
86
+ resolveId(id) {
87
+ if (id.startsWith(VIRTUAL_PREFIX)) {
88
+ const name = id.slice(VIRTUAL_PREFIX.length);
89
+ if (viewMap.has(name)) {
90
+ return `${VIRTUAL_MODULE_PREFIX}${name}`;
91
+ }
92
+ }
93
+ return null;
94
+ },
95
+ load(id) {
96
+ if (id.startsWith(VIRTUAL_MODULE_PREFIX)) {
97
+ const name = id.slice(VIRTUAL_MODULE_PREFIX.length);
98
+ const view = viewMap.get(name);
99
+ if (view) {
100
+ return buildVirtualEntry(view.filePath);
101
+ }
102
+ }
103
+ return null;
104
+ },
105
+ configureServer(server) {
106
+ if (!resolvedViewsDir) {
107
+ const root = server.config.root || process.cwd();
108
+ resolvedViewsDir = isAbsolute(rawViewsDir)
109
+ ? rawViewsDir
110
+ : resolve(root, rawViewsDir);
111
+ projectRoot = root;
112
+ viewEntryPattern = getViewEntryPattern(resolvedViewsDir);
113
+ }
114
+ server.watcher.add(resolvedViewsDir);
115
+ const rescan = () => {
116
+ try {
117
+ const views = discoverViewsSync(resolvedViewsDir);
118
+ viewMap = new Map(views.map((v) => [v.name, v]));
119
+ writeViewsDts(projectRoot, views);
48
120
  }
121
+ catch (err) {
122
+ // discoverViewsSync throws on duplicate view names. Catch so
123
+ // chokidar's listener chain doesn't surface it as unhandled and
124
+ // crash the dev server — previous viewMap stays active until
125
+ // the user fixes the conflict.
126
+ const message = err instanceof Error ? err.message : String(err);
127
+ server.config.logger.error(`[skybridge] view rescan failed: ${message}`);
128
+ }
129
+ };
130
+ server.watcher.on("add", rescan);
131
+ server.watcher.on("unlink", rescan);
132
+ },
133
+ async transform(code, id) {
134
+ if (viewEntryPattern?.test(id) && !hasDefaultExport(code, id)) {
135
+ this.warn(`View file "${id.split("/").pop()}" is missing a default export.`);
49
136
  }
50
137
  return await dataLlmTransform(code, id);
51
138
  },
@@ -1 +1 @@
1
- {"version":3,"file":"plugin.js","sourceRoot":"","sources":["../../../src/web/plugin/plugin.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,SAAS,IAAI,gBAAgB,EAAE,MAAM,yBAAyB,CAAC;AACxE,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AAEtD,oHAAoH;AACpH,MAAM,eAAe,GACnB,oEAAoE,CAAC;AAEvE,MAAM,UAAU,SAAS;IACvB,OAAO;QACL,IAAI,EAAE,WAAW;QAEjB,KAAK,CAAC,MAAM,CAAC,MAAM;YACjB,4EAA4E;YAC5E,MAAM,EAAE,QAAQ,EAAE,GAAG,MAAM,MAAM,CAAC,SAAS,CAAC,CAAC;YAC7C,MAAM,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,GAAG,MAAM,MAAM,CAAC,WAAW,CAAC,CAAC;YAExE,MAAM,WAAW,GAAG,MAAM,CAAC,IAAI,IAAI,OAAO,CAAC,GAAG,EAAE,CAAC;YACjD,MAAM,iBAAiB,GAAG,OAAO,CAC/B,WAAW,EACX,oCAAoC,CACrC,CAAC;YACF,MAAM,gBAAgB,GAAG,OAAO,CAAC,WAAW,EAAE,yBAAyB,CAAC,CAAC;YAEzE,MAAM,WAAW,GAAG,QAAQ,CAAC,iBAAiB,CAAC,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE;gBAC3D,MAAM,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC;gBAC9B,OAAO,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;YACtB,CAAC,CAAC,CAAC;YACH,MAAM,UAAU,GAAG,QAAQ,CAAC,gBAAgB,CAAC,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE;gBACzD,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC;gBACrC,OAAO,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;YACtB,CAAC,CAAC,CAAC;YACH,MAAM,KAAK,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC,GAAG,WAAW,EAAE,GAAG,UAAU,CAAC,CAAC,CAAC;YAElE,OAAO;gBACL,IAAI,EAAE,SAAS;gBACf,KAAK,EAAE;oBACL,QAAQ,EAAE,IAAI;oBACd,MAAM,EAAE,IAAI;oBACZ,YAAY,EAAE,KAAK;oBACnB,aAAa,EAAE;wBACb,KAAK;qBACN;iBACF;gBACD,YAAY,EAAE;oBACZ,cAAc,EAAE,CAAC,QAAQ,EAAE,EAAE;wBAC3B,OAAO;4BACL,OAAO,EAAE,yCAAyC,QAAQ,GAAG;yBAC9D,CAAC;oBACJ,CAAC;iBACF;aACF,CAAC;QACJ,CAAC;QACD,OAAO,EAAE,KAAK;QACd,KAAK,CAAC,SAAS,CAAC,IAAI,EAAE,EAAE;YACtB,IAAI,eAAe,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC;gBAC7B,KAAK,MAAM,OAAO,IAAI,cAAc,CAAC,IAAI,EAAE,EAAE,CAAC,EAAE,CAAC;oBAC/C,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;gBAC7B,CAAC;YACH,CAAC;YAED,OAAO,MAAM,gBAAgB,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;QAC1C,CAAC;KACF,CAAC;AACJ,CAAC"}
1
+ {"version":3,"file":"plugin.js","sourceRoot":"","sources":["../../../src/web/plugin/plugin.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ export interface DiscoveredView {
2
+ name: string;
3
+ filePath: string;
4
+ }
5
+ export declare function discoverViewsSync(viewsDir: string): DiscoveredView[];
6
+ export declare function generateViewsDts(views: DiscoveredView[]): string;
7
+ export declare function writeViewsDts(projectRoot: string, views: DiscoveredView[]): void;
8
+ export declare function scanAndWriteViewsDts(projectRoot?: string, viewsDir?: string): void;
@@ -0,0 +1,70 @@
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 discoverViewsSync(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
+ // Filter first, then check duplicates — so a barrel file like
16
+ // `views/foo/index.tsx` (no default export) doesn't falsely collide with
17
+ // a sibling view at `views/foo.tsx`.
18
+ const views = [...flatFiles, ...dirFiles]
19
+ .filter((v) => v.name !== "index")
20
+ .filter((v) => hasDefaultExport(readFileSync(v.filePath, "utf-8"), v.filePath));
21
+ const nameMap = new Map();
22
+ for (const view of views) {
23
+ const paths = nameMap.get(view.name) ?? [];
24
+ paths.push(view.filePath);
25
+ nameMap.set(view.name, paths);
26
+ }
27
+ for (const [name, paths] of nameMap) {
28
+ if (paths.length > 1) {
29
+ throw new Error(`skybridge: duplicate view name "${name}" resolved from:\n - ${paths.join("\n - ")}\nRename one of the files to avoid the conflict.`);
30
+ }
31
+ }
32
+ return views;
33
+ }
34
+ export function generateViewsDts(views) {
35
+ const entries = views.map((v) => ` "${v.name}": true;`).join("\n");
36
+ return [
37
+ "export {};",
38
+ "",
39
+ 'declare module "skybridge/server" {',
40
+ " interface ViewNameRegistry {",
41
+ entries,
42
+ " }",
43
+ "}",
44
+ "",
45
+ ].join("\n");
46
+ }
47
+ export function writeViewsDts(projectRoot, views) {
48
+ const dir = join(projectRoot, ".skybridge");
49
+ mkdirSync(dir, { recursive: true });
50
+ const filePath = join(dir, "views.d.ts");
51
+ const content = generateViewsDts(views);
52
+ try {
53
+ const existing = readFileSync(filePath, "utf-8");
54
+ if (existing === content) {
55
+ return;
56
+ }
57
+ }
58
+ catch {
59
+ // File doesn't exist yet
60
+ }
61
+ writeFileSync(filePath, content, "utf-8");
62
+ }
63
+ export function scanAndWriteViewsDts(projectRoot, viewsDir) {
64
+ const root = projectRoot ?? process.cwd();
65
+ const rawDir = viewsDir ?? "src/views";
66
+ const resolvedDir = isAbsolute(rawDir) ? rawDir : resolve(root, rawDir);
67
+ const views = discoverViewsSync(resolvedDir);
68
+ writeViewsDts(root, views);
69
+ }
70
+ //# 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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"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,67 @@
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, 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("writeViewsDts", () => {
34
+ let root;
35
+ beforeEach(() => {
36
+ root = mkdtempSync(join(tmpdir(), "skybridge-dts-"));
37
+ });
38
+ afterEach(() => {
39
+ rmSync(root, { recursive: true, force: true });
40
+ });
41
+ it("is a no-op when content is unchanged", () => {
42
+ const views = [{ name: "a", filePath: "/a.tsx" }];
43
+ writeViewsDts(root, views);
44
+ const dtsPath = join(root, ".skybridge", "views.d.ts");
45
+ const firstMtime = statSync(dtsPath).mtimeMs;
46
+ writeViewsDts(root, views);
47
+ expect(statSync(dtsPath).mtimeMs).toBe(firstMtime);
48
+ });
49
+ });
50
+ describe("scanAndWriteViewsDts", () => {
51
+ let root;
52
+ beforeEach(() => {
53
+ root = mkdtempSync(join(tmpdir(), "skybridge-scan-dts-"));
54
+ mkdirSync(join(root, "src/views"), { recursive: true });
55
+ writeFileSync(join(root, "src/views/hello.tsx"), DEFAULT_EXPORT);
56
+ });
57
+ afterEach(() => {
58
+ rmSync(root, { recursive: true, force: true });
59
+ });
60
+ it("writes a views.d.ts that augments skybridge/server with discovered view names", () => {
61
+ scanAndWriteViewsDts(root);
62
+ const content = readFileSync(join(root, ".skybridge/views.d.ts"), "utf-8");
63
+ expect(content).toContain('declare module "skybridge/server"');
64
+ expect(content).toContain('"hello": true;');
65
+ });
66
+ });
67
+ //# 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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"}
@@ -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"}
@@ -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"}
package/package.json CHANGED
@@ -1,12 +1,15 @@
1
1
  {
2
2
  "name": "skybridge",
3
- "version": "0.0.0-dev.f381d90",
3
+ "version": "0.0.0-dev.f391982",
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",
@@ -33,7 +36,7 @@
33
36
  "author": "Frédéric Barthelet",
34
37
  "license": "ISC",
35
38
  "peerDependencies": {
36
- "@modelcontextprotocol/sdk": ">=1.0.0",
39
+ "@modelcontextprotocol/sdk": ">=1.27.0",
37
40
  "@skybridge/devtools": ">=0.35.14 <1.0.0",
38
41
  "nodemon": ">=3.0.0",
39
42
  "react": ">=18.0.0",
@@ -50,7 +53,7 @@
50
53
  "es-toolkit": "^1.45.1",
51
54
  "express": "^5.2.1",
52
55
  "handlebars": "^4.7.9",
53
- "ink": "^6.8.0",
56
+ "ink": "^7.0.0",
54
57
  "posthog-node": "^5.28.9",
55
58
  "superjson": "^2.2.6",
56
59
  "zustand": "^5.0.12"
@@ -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,
@@ -1,13 +0,0 @@
1
- import type http from "node:http";
2
- import { type Router } from "express";
3
- /**
4
- * Install Vite dev server
5
- * This router MUST be installed at the application root, like so:
6
- *
7
- * const app = express();
8
- *
9
- * if (env.NODE_ENV !== "production") {
10
- * app.use(await widgetsRouter());
11
- * }
12
- */
13
- export declare const widgetsDevServer: (httpServer: http.Server) => Promise<Router>;
@@ -1,52 +0,0 @@
1
- import { existsSync } from "node:fs";
2
- import path from "node:path";
3
- import cors from "cors";
4
- import express, {} from "express";
5
- import { assetBaseUrlTransformPlugin } from "./asset-base-url-transform-plugin.js";
6
- /**
7
- * Install Vite dev server
8
- * This router MUST be installed at the application root, like so:
9
- *
10
- * const app = express();
11
- *
12
- * if (env.NODE_ENV !== "production") {
13
- * app.use(await widgetsRouter());
14
- * }
15
- */
16
- export const widgetsDevServer = async (httpServer) => {
17
- const router = express.Router();
18
- const { createServer, searchForWorkspaceRoot, loadConfigFromFile } = await import("vite");
19
- // Since 0.16.0, the template is a single package that does not rely on workspace.
20
- // It means that, when starting the server, the working dir is the template root
21
- // hence we don't need to walk up the tree to find the workspace, which does not exist anymore.
22
- let webAppRoot = path.join(process.cwd(), "web");
23
- // fallback to the old behavior for backward compatibility
24
- const hasWebAppRoot = existsSync(webAppRoot);
25
- if (!hasWebAppRoot) {
26
- const workspaceRoot = searchForWorkspaceRoot(process.cwd());
27
- webAppRoot = path.join(workspaceRoot, "web");
28
- }
29
- const configResult = await loadConfigFromFile({ command: "serve", mode: "development" }, path.join(webAppRoot, "vite.config.ts"), webAppRoot);
30
- const { build, preview, plugins: userPlugins = [], ...devConfig } = configResult?.config || {};
31
- const vite = await createServer({
32
- ...devConfig,
33
- configFile: false, // Keep this to prevent vite from trying to resolve path in the target config file
34
- appType: "custom",
35
- server: {
36
- allowedHosts: true,
37
- middlewareMode: true,
38
- hmr: {
39
- server: httpServer,
40
- },
41
- },
42
- root: webAppRoot,
43
- optimizeDeps: {
44
- include: ["react", "react-dom/client"],
45
- },
46
- plugins: [...userPlugins, assetBaseUrlTransformPlugin()],
47
- });
48
- router.use(cors());
49
- router.use("/", vite.middlewares);
50
- return router;
51
- };
52
- //# sourceMappingURL=widgetsDevServer.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"widgetsDevServer.js","sourceRoot":"","sources":["../../src/server/widgetsDevServer.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AAErC,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,OAAO,EAAE,EAAe,MAAM,SAAS,CAAC;AAC/C,OAAO,EAAE,2BAA2B,EAAE,MAAM,sCAAsC,CAAC;AAEnF;;;;;;;;;GASG;AACH,MAAM,CAAC,MAAM,gBAAgB,GAAG,KAAK,EACnC,UAAuB,EACN,EAAE;IACnB,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IAEhC,MAAM,EAAE,YAAY,EAAE,sBAAsB,EAAE,kBAAkB,EAAE,GAChE,MAAM,MAAM,CAAC,MAAM,CAAC,CAAC;IAEvB,kFAAkF;IAClF,gFAAgF;IAChF,+FAA+F;IAC/F,IAAI,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,KAAK,CAAC,CAAC;IAEjD,0DAA0D;IAC1D,MAAM,aAAa,GAAG,UAAU,CAAC,UAAU,CAAC,CAAC;IAC7C,IAAI,CAAC,aAAa,EAAE,CAAC;QACnB,MAAM,aAAa,GAAG,sBAAsB,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;QAC5D,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,KAAK,CAAC,CAAC;IAC/C,CAAC;IAED,MAAM,YAAY,GAAG,MAAM,kBAAkB,CAC3C,EAAE,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,aAAa,EAAE,EACzC,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,gBAAgB,CAAC,EACvC,UAAU,CACX,CAAC;IAEF,MAAM,EACJ,KAAK,EACL,OAAO,EACP,OAAO,EAAE,WAAW,GAAG,EAAE,EACzB,GAAG,SAAS,EACb,GAAG,YAAY,EAAE,MAAM,IAAI,EAAE,CAAC;IAE/B,MAAM,IAAI,GAAG,MAAM,YAAY,CAAC;QAC9B,GAAG,SAAS;QACZ,UAAU,EAAE,KAAK,EAAE,kFAAkF;QACrG,OAAO,EAAE,QAAQ;QACjB,MAAM,EAAE;YACN,YAAY,EAAE,IAAI;YAClB,cAAc,EAAE,IAAI;YACpB,GAAG,EAAE;gBACH,MAAM,EAAE,UAAU;aACnB;SACF;QACD,IAAI,EAAE,UAAU;QAChB,YAAY,EAAE;YACZ,OAAO,EAAE,CAAC,OAAO,EAAE,kBAAkB,CAAC;SACvC;QACD,OAAO,EAAE,CAAC,GAAG,WAAW,EAAE,2BAA2B,EAAE,CAAC;KACzD,CAAC,CAAC;IAEH,MAAM,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC;IACnB,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;IAElC,OAAO,MAAM,CAAC;AAChB,CAAC,CAAC"}