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.
- package/dist/cli/header.js +1 -1
- package/dist/cli/header.js.map +1 -1
- package/dist/cli/use-nodemon.js +2 -2
- package/dist/cli/use-nodemon.js.map +1 -1
- package/dist/cli/use-tunnel.d.ts +12 -5
- package/dist/cli/use-tunnel.js +34 -29
- package/dist/cli/use-tunnel.js.map +1 -1
- package/dist/cli/use-typescript-check.d.ts +1 -0
- package/dist/cli/use-typescript-check.js +41 -6
- package/dist/cli/use-typescript-check.js.map +1 -1
- package/dist/commands/build.js +28 -7
- package/dist/commands/build.js.map +1 -1
- package/dist/commands/dev.d.ts +1 -0
- package/dist/commands/dev.js +7 -2
- package/dist/commands/dev.js.map +1 -1
- package/dist/commands/start.js +7 -10
- package/dist/commands/start.js.map +1 -1
- package/dist/server/content-helpers.d.ts +27 -0
- package/dist/server/content-helpers.js +46 -0
- package/dist/server/content-helpers.js.map +1 -0
- package/dist/server/content-helpers.test.js +70 -0
- package/dist/server/content-helpers.test.js.map +1 -0
- package/dist/server/express.js +3 -3
- package/dist/server/express.js.map +1 -1
- package/dist/server/express.test.js +39 -2
- package/dist/server/express.test.js.map +1 -1
- package/dist/server/index.d.ts +4 -3
- package/dist/server/index.js +3 -2
- package/dist/server/index.js.map +1 -1
- package/dist/server/middleware.test.js +12 -9
- package/dist/server/middleware.test.js.map +1 -1
- package/dist/server/server.d.ts +95 -72
- package/dist/server/server.js +195 -73
- package/dist/server/server.js.map +1 -1
- package/dist/server/templateHelper.d.ts +5 -5
- package/dist/server/templates/development.hbs +2 -2
- package/dist/server/templates/production.hbs +1 -1
- package/dist/server/viewsDevServer.d.ts +14 -0
- package/dist/server/viewsDevServer.js +45 -0
- package/dist/server/viewsDevServer.js.map +1 -0
- package/dist/test/utils.d.ts +13 -21
- package/dist/test/utils.js +42 -37
- package/dist/test/utils.js.map +1 -1
- package/dist/test/view.test.js +523 -0
- package/dist/test/view.test.js.map +1 -0
- package/dist/web/bridges/apps-sdk/adaptor.d.ts +4 -2
- package/dist/web/bridges/apps-sdk/adaptor.js +24 -12
- package/dist/web/bridges/apps-sdk/adaptor.js.map +1 -1
- package/dist/web/bridges/apps-sdk/types.d.ts +10 -5
- package/dist/web/bridges/apps-sdk/types.js.map +1 -1
- package/dist/web/bridges/mcp-app/adaptor.d.ts +6 -0
- package/dist/web/bridges/mcp-app/adaptor.js +6 -0
- package/dist/web/bridges/mcp-app/adaptor.js.map +1 -1
- package/dist/web/bridges/types.d.ts +9 -3
- package/dist/web/generate-helpers.d.ts +20 -18
- package/dist/web/generate-helpers.js +20 -18
- package/dist/web/generate-helpers.js.map +1 -1
- package/dist/web/generate-helpers.test-d.js +20 -20
- package/dist/web/generate-helpers.test-d.js.map +1 -1
- package/dist/web/hooks/use-files.d.ts +2 -1
- package/dist/web/hooks/use-files.js +1 -0
- package/dist/web/hooks/use-files.js.map +1 -1
- package/dist/web/hooks/use-files.test.js +22 -2
- package/dist/web/hooks/use-files.test.js.map +1 -1
- package/dist/web/index.d.ts +1 -0
- package/dist/web/index.js.map +1 -1
- package/dist/web/plugin/plugin.d.ts +4 -1
- package/dist/web/plugin/plugin.js +112 -25
- package/dist/web/plugin/plugin.js.map +1 -1
- package/dist/web/plugin/scan-views.d.ts +8 -0
- package/dist/web/plugin/scan-views.js +70 -0
- package/dist/web/plugin/scan-views.js.map +1 -0
- package/dist/web/plugin/scan-views.test.d.ts +1 -0
- package/dist/web/plugin/scan-views.test.js +67 -0
- package/dist/web/plugin/scan-views.test.js.map +1 -0
- package/dist/web/plugin/validate-view.d.ts +1 -0
- package/dist/web/plugin/validate-view.js +9 -0
- package/dist/web/plugin/validate-view.js.map +1 -0
- package/dist/web/plugin/validate-view.test.d.ts +1 -0
- package/dist/web/plugin/validate-view.test.js +24 -0
- package/dist/web/plugin/validate-view.test.js.map +1 -0
- package/package.json +6 -3
- package/tsconfig.base.json +2 -0
- package/dist/server/widgetsDevServer.d.ts +0 -13
- package/dist/server/widgetsDevServer.js +0 -52
- package/dist/server/widgetsDevServer.js.map +0 -1
- package/dist/test/widget.test.js +0 -303
- package/dist/test/widget.test.js.map +0 -1
- package/dist/web/plugin/validate-widget.d.ts +0 -5
- package/dist/web/plugin/validate-widget.js +0 -27
- package/dist/web/plugin/validate-widget.js.map +0 -1
- package/dist/web/plugin/validate-widget.test.js +0 -42
- package/dist/web/plugin/validate-widget.test.js.map +0 -1
- /package/dist/{test/widget.test.d.ts → server/content-helpers.test.d.ts} +0 -0
- /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 {
|
|
3
|
-
|
|
4
|
-
const
|
|
5
|
-
|
|
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
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
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
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
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":"
|
|
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.
|
|
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.
|
|
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": "^
|
|
56
|
+
"ink": "^7.0.0",
|
|
54
57
|
"posthog-node": "^5.28.9",
|
|
55
58
|
"superjson": "^2.2.6",
|
|
56
59
|
"zustand": "^5.0.12"
|
package/tsconfig.base.json
CHANGED
|
@@ -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"}
|