create-skybridge 0.0.0-dev.ba902f0 → 0.0.0-dev.baa9b4b

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 (83) hide show
  1. package/dist/index.js +339 -167
  2. package/package.json +9 -8
  3. package/templates/blank/.dockerignore +4 -0
  4. package/templates/blank/AGENTS.md +1 -0
  5. package/templates/blank/Dockerfile +53 -0
  6. package/templates/blank/README.md +92 -0
  7. package/templates/blank/_gitignore +6 -0
  8. package/templates/blank/alpic.json +3 -0
  9. package/templates/blank/node_modules/.bin/alpic +21 -0
  10. package/templates/blank/node_modules/.bin/sb +21 -0
  11. package/templates/blank/node_modules/.bin/skybridge +21 -0
  12. package/templates/blank/node_modules/.bin/tsc +21 -0
  13. package/templates/blank/node_modules/.bin/tsserver +21 -0
  14. package/templates/blank/node_modules/.bin/vite +21 -0
  15. package/templates/blank/package.json +28 -0
  16. package/templates/blank/src/helpers.ts +4 -0
  17. package/templates/blank/src/server.ts +21 -0
  18. package/templates/blank/src/vite-manifest.d.ts +4 -0
  19. package/templates/blank/tsconfig.json +5 -0
  20. package/templates/blank/vite.config.ts +6 -0
  21. package/templates/demo/.dockerignore +4 -0
  22. package/templates/demo/AGENTS.md +1 -0
  23. package/templates/demo/Dockerfile +53 -0
  24. package/templates/demo/README.md +95 -0
  25. package/templates/demo/_gitignore +6 -0
  26. package/templates/demo/alpic.json +3 -0
  27. package/templates/demo/node_modules/.bin/alpic +21 -0
  28. package/templates/demo/node_modules/.bin/sb +21 -0
  29. package/templates/demo/node_modules/.bin/skybridge +21 -0
  30. package/templates/demo/node_modules/.bin/tsc +21 -0
  31. package/templates/demo/node_modules/.bin/tsserver +21 -0
  32. package/templates/demo/node_modules/.bin/tsx +21 -0
  33. package/templates/demo/node_modules/.bin/vite +21 -0
  34. package/templates/demo/package.json +41 -0
  35. package/templates/demo/src/helpers.ts +4 -0
  36. package/templates/demo/src/index.css +59 -0
  37. package/templates/demo/src/server.ts +99 -0
  38. package/templates/demo/src/views/components/doc-link.tsx +22 -0
  39. package/templates/demo/src/views/components/doc.tsx +21 -0
  40. package/templates/demo/src/views/components/nav.tsx +31 -0
  41. package/templates/demo/src/views/components/progress.tsx +35 -0
  42. package/templates/demo/src/views/components/steps/outro.tsx +68 -0
  43. package/templates/demo/src/views/components/steps/state.tsx +47 -0
  44. package/templates/demo/src/views/components/steps/tool-call.tsx +53 -0
  45. package/templates/demo/src/views/components/steps/tool-output.tsx +40 -0
  46. package/templates/demo/src/views/images/mascot/beret.png +0 -0
  47. package/templates/demo/src/views/images/mascot/chapka.png +0 -0
  48. package/templates/demo/src/views/images/mascot/cowboy-hat.png +0 -0
  49. package/templates/demo/src/views/images/mascot/fez.png +0 -0
  50. package/templates/demo/src/views/images/mascot/jester-hat.png +0 -0
  51. package/templates/demo/src/views/images/mascot/mitre.png +0 -0
  52. package/templates/demo/src/views/images/mascot/non-la.png +0 -0
  53. package/templates/demo/src/views/images/mascot/original.png +0 -0
  54. package/templates/demo/src/views/images/mascot/propeller-beanie.png +0 -0
  55. package/templates/demo/src/views/images/mascot/ski-mask.png +0 -0
  56. package/templates/demo/src/views/images/mascot/sombrero.png +0 -0
  57. package/templates/demo/src/views/images/mascot/top-hat.png +0 -0
  58. package/templates/demo/src/views/images/mascot/viking-helmet.png +0 -0
  59. package/templates/demo/src/views/onboarding.tsx +63 -0
  60. package/templates/demo/src/views/use-mascot.ts +60 -0
  61. package/templates/demo/src/vite-manifest.d.ts +4 -0
  62. package/templates/demo/tsconfig.json +11 -0
  63. package/{template/web → templates/demo}/vite.config.ts +3 -4
  64. package/dist/index.test.d.ts +0 -1
  65. package/dist/index.test.js +0 -23
  66. package/template/README.md +0 -66
  67. package/template/_gitignore +0 -4
  68. package/template/alpic.json +0 -4
  69. package/template/package.json +0 -21
  70. package/template/pnpm-workspace.yaml +0 -8
  71. package/template/server/nodemon.json +0 -5
  72. package/template/server/package.json +0 -33
  73. package/template/server/src/index.ts +0 -36
  74. package/template/server/src/middleware.ts +0 -54
  75. package/template/server/src/server.ts +0 -66
  76. package/template/server/tsconfig.json +0 -17
  77. package/template/web/package.json +0 -24
  78. package/template/web/src/helpers.ts +0 -4
  79. package/template/web/src/index.css +0 -30
  80. package/template/web/src/widgets/magic-8-ball.tsx +0 -24
  81. package/template/web/tsconfig.app.json +0 -34
  82. package/template/web/tsconfig.json +0 -13
  83. package/template/web/tsconfig.node.json +0 -26
package/dist/index.js CHANGED
@@ -1,217 +1,389 @@
1
- import { spawnSync } from "node:child_process";
2
1
  import fs from "node:fs";
3
2
  import path from "node:path";
4
3
  import { fileURLToPath } from "node:url";
5
4
  import * as prompts from "@clack/prompts";
5
+ import spawn from "cross-spawn";
6
6
  import mri from "mri";
7
- const minimumPnpmVersion = 10;
8
- const defaultProjectName = "skybridge-project";
9
- // prettier-ignore
10
- const helpMessage = `\
11
- Usage: create-skybridge [OPTION]... [DIRECTORY]
7
+ const OUTPUT_TAIL_LINES = 10;
8
+ const DEFAULT_PROJECT_NAME = "skybridge-project";
9
+ const PACKAGE_MANAGERS = ["bun", "deno", "npm", "pnpm", "yarn"];
10
+ const TEMPLATES = ["demo", "blank"];
11
+ const pkg = JSON.parse(fs.readFileSync(fileURLToPath(new URL("../package.json", import.meta.url)), "utf-8"));
12
+ const version = pkg.version;
13
+ const HELP_MESSAGE = `Usage: skybridge create [path] [options]
12
14
 
13
- Create a new Skybridge project by copying the starter template.
15
+ Skybridge v${version} - the fullstack framework for building MCP Apps
16
+
17
+ Arguments:
18
+ path Where the project will be created. Prompted when omitted.
14
19
 
15
20
  Options:
16
- -h, --help show this help message
17
- --overwrite remove existing files in target directory
18
- --immediate install dependencies and start development server
21
+ --blank scaffold a minimal project without demo tools and views
22
+ --overwrite remove existing files if target directory is not empty
23
+ --pm <choice> package manager to use (choices: ${PACKAGE_MANAGERS.join(", ")}. default to npm when none is provided or infered)
24
+ --skip-skills skip installing coding agent skills
25
+ --start start dev server
26
+ --yes skip prompts and use default values for unprovided options
27
+ --help display this help message
19
28
 
20
- Examples:
21
- create-skybridge my-app
22
- create-skybridge . --overwrite --immediate
23
- `;
29
+ Non-interactive usage:
30
+ Mandatory: path argument and --yes option
31
+ Example: skybridge create my-app --yes`;
32
+ const isTTY = process.stdout.isTTY;
33
+ const _spinner = prompts.spinner();
34
+ const Spinner = {
35
+ start(msg) {
36
+ if (!isTTY) {
37
+ prompts.log.info(msg);
38
+ }
39
+ else {
40
+ _spinner.clear();
41
+ _spinner.start(msg);
42
+ }
43
+ },
44
+ stop(msg) {
45
+ if (!isTTY) {
46
+ prompts.log.success(msg);
47
+ }
48
+ else {
49
+ _spinner.stop(msg);
50
+ }
51
+ },
52
+ error(msg) {
53
+ if (!isTTY) {
54
+ prompts.log.error(msg);
55
+ }
56
+ else {
57
+ _spinner.error(msg);
58
+ }
59
+ },
60
+ };
24
61
  export async function init(args = process.argv.slice(2)) {
25
62
  const argv = mri(args, {
26
- boolean: ["help", "overwrite", "immediate"],
63
+ boolean: ["help", "blank", "overwrite", "skip-skills", "start", "yes"],
64
+ string: ["pm"],
27
65
  alias: { h: "help" },
28
66
  });
29
- const argTargetDir = argv._[0]
30
- ? sanitizeTargetDir(String(argv._[0]))
31
- : undefined;
32
- const argOverwrite = argv.overwrite;
33
- const argImmediate = argv.immediate;
34
- const help = argv.help;
35
- if (help) {
36
- console.log(helpMessage);
67
+ if (argv.help) {
68
+ console.log(HELP_MESSAGE);
37
69
  return;
38
70
  }
39
- const interactive = process.stdin.isTTY;
40
- const cancel = () => prompts.cancel("Operation cancelled");
41
- // 1. Get project name and target dir
42
- let targetDir = argTargetDir;
71
+ const { yes } = argv;
72
+ let targetDir = argv._[0] ? sanitizeTargetDir(String(argv._[0])) : undefined;
73
+ if (yes && !targetDir) {
74
+ abort("The target directory is required in non-interactive mode.", "Example: skybridge create my-app --yes");
75
+ }
76
+ let pm = parsePackageManager(argv.pm || "");
77
+ if (argv.pm && !pm) {
78
+ abort(`Invalid --pm value "${argv.pm}". Expected one of: ${PACKAGE_MANAGERS.join(", ")}.`);
79
+ }
80
+ console.log(); // cosmetic line break
81
+ prompts.intro(`\x1b[1;36m⛰ Welcome to Skybridge v${version} \x1b[22m- the fullstack framework for building MCP Apps\x1b[0m`);
82
+ // 1. Target directory
43
83
  if (!targetDir) {
44
- if (interactive) {
45
- const projectName = await prompts.text({
46
- message: "Project name:",
47
- defaultValue: defaultProjectName,
48
- placeholder: defaultProjectName,
49
- validate: (value) => {
50
- return value.length === 0 || sanitizeTargetDir(value).length > 0
51
- ? undefined
52
- : "Invalid project name";
53
- },
54
- });
55
- if (prompts.isCancel(projectName)) {
56
- return cancel();
57
- }
58
- targetDir = sanitizeTargetDir(projectName);
59
- }
60
- else {
61
- targetDir = defaultProjectName;
84
+ const choice = await prompts.text({
85
+ message: "Project directory:",
86
+ placeholder: DEFAULT_PROJECT_NAME,
87
+ defaultValue: DEFAULT_PROJECT_NAME,
88
+ validate: (value) => !value || sanitizeTargetDir(value).length > 0
89
+ ? undefined
90
+ : "Invalid project name",
91
+ });
92
+ if (prompts.isCancel(choice)) {
93
+ return cancel();
62
94
  }
95
+ targetDir = sanitizeTargetDir(choice);
63
96
  }
64
- // 2. Handle directory if exist and not empty
97
+ // 2. Existing-directory handling
65
98
  if (fs.existsSync(targetDir) && !isEmpty(targetDir)) {
66
- let overwrite = argOverwrite ? "yes" : undefined;
67
- if (!overwrite) {
68
- if (interactive) {
69
- const res = await prompts.select({
70
- message: (targetDir === "."
71
- ? "Current directory"
72
- : `Target directory "${targetDir}"`) +
73
- ` is not empty. Please choose how to proceed:`,
74
- options: [
75
- {
76
- label: "Cancel operation",
77
- value: "no",
78
- },
79
- {
80
- label: "Remove existing files and continue",
81
- value: "yes",
82
- },
83
- ],
84
- });
85
- if (prompts.isCancel(res)) {
86
- return cancel();
87
- }
88
- overwrite = res;
89
- }
90
- else {
91
- overwrite = "no";
99
+ if (argv.overwrite) {
100
+ emptyDir(targetDir);
101
+ }
102
+ else if (yes) {
103
+ prompts.log.error(`Target directory "${targetDir}" is not empty. Use --overwrite to remove existing files.`);
104
+ process.exit(1);
105
+ }
106
+ else {
107
+ const ok = await prompts.confirm({
108
+ message: `Target directory "${targetDir}" is not empty. Remove existing files?`,
109
+ initialValue: true,
110
+ });
111
+ if (prompts.isCancel(ok) || !ok) {
112
+ return cancel();
92
113
  }
114
+ Spinner.start(`Cleaning up ${targetDir}`);
115
+ emptyDir(targetDir);
116
+ Spinner.stop(`Cleaned up ${targetDir}`);
93
117
  }
94
- switch (overwrite) {
95
- case "yes":
96
- emptyDir(targetDir);
97
- break;
98
- case "no":
99
- cancel();
100
- return;
118
+ }
119
+ // 3. Template
120
+ let template;
121
+ if (argv.blank) {
122
+ template = "blank";
123
+ }
124
+ else if (yes) {
125
+ template = "demo";
126
+ }
127
+ else {
128
+ const choice = await prompts.select({
129
+ message: "Choose a template:",
130
+ options: [
131
+ {
132
+ value: "demo",
133
+ label: "demo",
134
+ hint: "starter code with tools and UI",
135
+ },
136
+ {
137
+ value: "blank",
138
+ label: "blank",
139
+ hint: "minimal boilerplate without tools",
140
+ },
141
+ ],
142
+ initialValue: "demo",
143
+ });
144
+ if (prompts.isCancel(choice)) {
145
+ return cancel();
101
146
  }
147
+ template = choice;
102
148
  }
103
- const root = path.join(process.cwd(), targetDir);
104
- // 3. Copy the repository
105
- prompts.log.step(`Copying template...`);
149
+ // 4. Copy template
150
+ const root = path.resolve(targetDir);
151
+ Spinner.start(`Copying ${template} template`);
106
152
  try {
107
- const templateDir = fileURLToPath(new URL("../template", import.meta.url));
108
- // Copy template to target directory
153
+ const templateDir = fileURLToPath(new URL(`../templates/${template}`, import.meta.url));
109
154
  fs.cpSync(templateDir, root, {
110
155
  recursive: true,
111
- filter: (src) => !src.endsWith(".npmrc"),
156
+ filter: (src) => [".npmrc"].every((file) => !src.endsWith(file)),
112
157
  });
113
- // Rename _gitignore to .gitignore
114
- fs.renameSync(path.join(root, "_gitignore"), path.join(root, ".gitignore"));
115
- // Update project name in package.json
116
- const name = path.basename(root);
117
- for (const dir of ["", "server", "web"]) {
118
- const pkgPath = path.join(root, dir, "package.json");
119
- const pkg = fs.readFileSync(pkgPath, "utf-8");
120
- const fixed = pkg.replace(/apps-sdk-template/g, name);
121
- fs.writeFileSync(pkgPath, fixed);
122
- }
123
- prompts.log.success(`Project created in ${root}`);
158
+ const gitignoreSource = path.join(root, "_gitignore");
159
+ if (fs.existsSync(gitignoreSource)) {
160
+ fs.renameSync(gitignoreSource, path.join(root, ".gitignore"));
161
+ }
162
+ Spinner.stop(`Copied ${template} template`);
163
+ }
164
+ catch (error) {
165
+ Spinner.error("Failed to copy template");
166
+ abort(String(error));
167
+ }
168
+ // 5. Set package.json name to the project dir basename
169
+ try {
170
+ const pkgPath = path.join(root, "package.json");
171
+ const projectPkg = JSON.parse(fs.readFileSync(pkgPath, "utf-8"));
172
+ projectPkg.name = path.basename(root);
173
+ fs.writeFileSync(pkgPath, `${JSON.stringify(projectPkg, null, 2)}\n`);
124
174
  }
125
175
  catch (error) {
126
- prompts.log.error("Failed to copy repository");
127
- console.error(error);
128
- process.exit(1);
129
- }
130
- // 4. Ask about immediate installation
131
- let immediate = argImmediate;
132
- if (immediate === undefined) {
133
- if (interactive) {
134
- const immediateResult = await prompts.confirm({
135
- message: `Install with pnpm and start now?`,
176
+ abort("Failed to update project name in package.json.", String(error));
177
+ }
178
+ // Async spawn wrapper so a spinner can keep animating during the subprocess
179
+ // (cross-spawn.sync would block the event loop). Captures stdout/stderr to
180
+ // `output` when stdio is "pipe", trimmed to the last OUTPUT_TAIL_LINES lines
181
+ // install errors land at the tail, so we keep that and prefix with an
182
+ // ellipsis when content gets dropped.
183
+ function spawnAsync(command, args) {
184
+ return new Promise((resolve) => {
185
+ let raw = "";
186
+ const child = spawn(command, args, {
187
+ stdio: ["ignore", "pipe", "pipe"],
188
+ cwd: root,
136
189
  });
137
- if (prompts.isCancel(immediateResult)) {
138
- return cancel();
139
- }
140
- immediate = immediateResult;
190
+ child.stdout?.on("data", (chunk) => {
191
+ raw += chunk.toString();
192
+ });
193
+ child.stderr?.on("data", (chunk) => {
194
+ raw += chunk.toString();
195
+ });
196
+ const done = (status) => {
197
+ const tail = [];
198
+ for (const part of raw.split("\n").reverse()) {
199
+ const line = part.trim();
200
+ if (!line) {
201
+ continue;
202
+ }
203
+ if (tail.length >= OUTPUT_TAIL_LINES) {
204
+ tail.push(`… (truncated, showing last ${OUTPUT_TAIL_LINES} lines)`);
205
+ break;
206
+ }
207
+ tail.push(line);
208
+ }
209
+ resolve({ status, output: tail.reverse().join("\n") });
210
+ };
211
+ child.on("close", done);
212
+ child.on("error", () => done(1));
213
+ });
214
+ }
215
+ // 6. Skills install (single Y/n prompt)
216
+ let installSkills;
217
+ if (argv["skip-skills"]) {
218
+ installSkills = false;
219
+ }
220
+ else if (yes) {
221
+ installSkills = true;
222
+ }
223
+ else {
224
+ const choice = await prompts.confirm({
225
+ message: "Install coding agent skills? (recommended)",
226
+ initialValue: true,
227
+ });
228
+ if (prompts.isCancel(choice)) {
229
+ return cancel();
230
+ }
231
+ installSkills = choice;
232
+ }
233
+ if (installSkills) {
234
+ Spinner.start("Installing coding agent skills");
235
+ const { status, output } = await spawnAsync("npx", [
236
+ "--yes",
237
+ "skills",
238
+ "add",
239
+ "alpic-ai/skybridge",
240
+ "--skill",
241
+ "chatgpt-app-builder",
242
+ "--agent",
243
+ "universal",
244
+ "claude-code",
245
+ "--copy", // something the symlink fails for some reason
246
+ "--yes",
247
+ ]);
248
+ // skills cli always returns 0 so we look for the success message
249
+ if (status === 0 && output.includes("Done!")) {
250
+ Spinner.stop(`Installed coding agent skills`);
141
251
  }
142
252
  else {
143
- immediate = false;
253
+ Spinner.error(`Failed to install coding agent skills:
254
+ \x1b[2m${output}\x1b[0m`);
255
+ prompts.log.error("Try manually: `npx skills add alpic-ai/skybridge`.");
144
256
  }
145
257
  }
146
- const installCmd = ["pnpm", "install"];
147
- const runCmd = ["pnpm", "dev"];
148
- if (!immediate) {
149
- prompts.outro(`Done! Next steps:
150
- cd ${targetDir}
151
- ${installCmd.join(" ")}
152
- ${runCmd.join(" ")}
153
- `);
154
- return;
258
+ // 7. Package manager — autodetect, prompt only if detection fails (interactive)
259
+ if (!pm) {
260
+ pm = detectPackageManager() || "npm";
155
261
  }
156
- // check if pnpm is installed
157
- const result = spawnSync("pnpm", ["--version"], { encoding: "utf-8" });
158
- if (result.error || result.status !== 0) {
159
- console.error("Error: pnpm is not installed. Please install pnpm first.");
160
- process.exit(1);
161
- }
162
- // check if pnpm major is greater or equal to the one set in package.json packageManager, which should do the trick
163
- const version = result.stdout.trim();
164
- const major = Number(version.split(".")[0]);
165
- if (Number.isNaN(major) || major < minimumPnpmVersion) {
166
- console.error(`Error: pnpm version ${version} is too old. Minimum required version is ${minimumPnpmVersion}.`);
167
- process.exit(1);
168
- }
169
- prompts.log.step(`Installing dependencies with pnpm...`);
170
- run(installCmd, {
171
- stdio: "inherit",
172
- cwd: root,
173
- });
174
- prompts.log.step("Starting dev server...");
175
- run(runCmd, {
176
- stdio: "inherit",
177
- cwd: root,
178
- });
262
+ if (!yes) {
263
+ const choice = await prompts.select({
264
+ message: "Choose a package manager:",
265
+ options: PACKAGE_MANAGERS.map((value) => ({ value })),
266
+ initialValue: pm,
267
+ });
268
+ if (prompts.isCancel(choice)) {
269
+ return cancel();
270
+ }
271
+ pm = choice;
272
+ }
273
+ // 8. Always install dependencies
274
+ Spinner.start(`Installing dependencies with ${pm}`);
275
+ const { status, output } = await spawnAsync(pm, ["install"]);
276
+ if (status === 0) {
277
+ Spinner.stop(`Installed dependencies with ${pm}`);
278
+ }
279
+ else {
280
+ Spinner.error(`Dependency installation failed:
281
+ \x1b[2m${output}\x1b[0m`);
282
+ abort(`Try manually: cd ${targetDir} && ${pm} install`);
283
+ }
284
+ // 9. Start dev server?
285
+ let start = false;
286
+ if (argv.start) {
287
+ start = true;
288
+ }
289
+ else if (!yes) {
290
+ const choice = await prompts.confirm({
291
+ message: "Start dev server now?",
292
+ initialValue: true,
293
+ });
294
+ if (prompts.isCancel(choice)) {
295
+ return cancel();
296
+ }
297
+ start = choice;
298
+ }
299
+ if (start) {
300
+ prompts.outro(`Starting dev server in ${targetDir}…`);
301
+ const devResult = spawn.sync(pm, scriptArgs(pm, "dev"), {
302
+ stdio: "inherit",
303
+ cwd: root,
304
+ });
305
+ process.exit(devResult.status ?? 0);
306
+ }
307
+ prompts.log.success("All set! Next steps:");
308
+ prompts.log.info(`Start:
309
+ cd ${targetDir}
310
+ ${scriptCommand(pm, "dev")}`);
311
+ prompts.log.info(`Deploy:
312
+ ${scriptCommand(pm, "deploy")}`);
313
+ prompts.outro(`🛟 Need help?
314
+ Chat: https://discord.alpic.ai
315
+ Docs: https://docs.skybridge.tech`);
316
+ }
317
+ function cancel() {
318
+ prompts.cancel("Operation cancelled");
319
+ process.exit(0);
179
320
  }
180
- function run([command, ...args], options) {
181
- const { status, error } = spawnSync(command, args, options);
182
- if (status != null && status > 0) {
183
- process.exit(status);
321
+ function abort(...lines) {
322
+ for (const line of lines) {
323
+ prompts.log.error(line);
184
324
  }
185
- if (error) {
186
- console.error(`\n${command} ${args.join(" ")} error!`);
187
- console.error(error);
188
- process.exit(1);
325
+ prompts.outro("Aborted");
326
+ process.exit(1);
327
+ }
328
+ function parsePackageManager(value) {
329
+ switch (value) {
330
+ case "bun":
331
+ return "bun";
332
+ case "deno":
333
+ return "deno";
334
+ case "npm":
335
+ return "npm";
336
+ case "pnpm":
337
+ return "pnpm";
338
+ case "yarn":
339
+ return "yarn";
340
+ default:
341
+ return undefined;
342
+ }
343
+ }
344
+ function detectPackageManager() {
345
+ const userAgent = process.env.npm_config_user_agent;
346
+ if (!userAgent) {
347
+ return undefined;
189
348
  }
349
+ const name = userAgent.split(" ")[0]?.split("/")[0];
350
+ return parsePackageManager(name);
351
+ }
352
+ function scriptArgs(pm, script) {
353
+ switch (pm) {
354
+ case "yarn":
355
+ case "pnpm":
356
+ case "bun":
357
+ return [script];
358
+ case "deno":
359
+ return ["task", script];
360
+ case "npm":
361
+ return ["run", script];
362
+ }
363
+ }
364
+ function scriptCommand(pm, script) {
365
+ return [pm, ...scriptArgs(pm, script)].join(" ");
190
366
  }
191
367
  function sanitizeTargetDir(targetDir) {
192
- return (targetDir
193
- .trim()
194
- // Only keep alphanumeric, dash, underscore, dot, @, /
195
- .replace(/[^a-zA-Z0-9\-_.@/]/g, "")
196
- // Prevent path traversal
197
- .replace(/\.\./g, "")
198
- // Collapse multiple slashes
199
- .replace(/\/+/g, "/")
200
- // Remove leading/trailing slashes
201
- .replace(/^\/+|\/+$/g, ""));
368
+ return targetDir.trim().replace(/\/+$/g, "");
369
+ }
370
+ // Skip user's SPEC.md and IDE/agent preferences (.idea, .claude, etc.)
371
+ function isSkippedEntry(entry) {
372
+ return ((entry.name.startsWith(".") && entry.isDirectory()) ||
373
+ entry.name === "SPEC.md");
202
374
  }
203
- function isEmpty(path) {
204
- const files = fs.readdirSync(path);
205
- return files.length === 0 || (files.length === 1 && files[0] === ".git");
375
+ function isEmpty(dirPath) {
376
+ const entries = fs.readdirSync(dirPath, { withFileTypes: true });
377
+ return entries.every(isSkippedEntry);
206
378
  }
207
379
  function emptyDir(dir) {
208
380
  if (!fs.existsSync(dir)) {
209
381
  return;
210
382
  }
211
- for (const file of fs.readdirSync(dir)) {
212
- if (file === ".git") {
383
+ for (const entry of fs.readdirSync(dir, { withFileTypes: true })) {
384
+ if (isSkippedEntry(entry)) {
213
385
  continue;
214
386
  }
215
- fs.rmSync(path.resolve(dir, file), { recursive: true, force: true });
387
+ fs.rmSync(path.join(dir, entry.name), { recursive: true, force: true });
216
388
  }
217
389
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "create-skybridge",
3
- "version": "0.0.0-dev.ba902f0",
3
+ "version": "0.0.0-dev.baa9b4b",
4
4
  "type": "module",
5
5
  "license": "MIT",
6
6
  "author": "Alpic",
@@ -14,22 +14,23 @@
14
14
  "files": [
15
15
  "index.js",
16
16
  "dist",
17
- "template"
17
+ "templates"
18
18
  ],
19
19
  "dependencies": {
20
- "@clack/prompts": "^0.11.0",
20
+ "@clack/prompts": "^1.1.0",
21
+ "cross-spawn": "^7.0.6",
21
22
  "mri": "^1.2.0"
22
23
  },
23
24
  "devDependencies": {
24
- "@types/node": "^25.0.3",
25
- "typescript": "^5.9.3",
26
- "vitest": "^2.1.9"
25
+ "@types/cross-spawn": "^6.0.6",
26
+ "typescript": "^6.0.2",
27
+ "vitest": "^4.1.4"
27
28
  },
28
29
  "scripts": {
29
30
  "build": "tsc",
30
- "test": "pnpm run test:unit && pnpm run test:type && pnpm run test:format",
31
+ "test": "pnpm run test:unit && pnpm run test:format",
31
32
  "test:unit": "vitest run",
32
- "test:type": "tsc --noEmit",
33
+ "format": "biome check --write --error-on-warnings",
33
34
  "test:format": "biome ci"
34
35
  }
35
36
  }
@@ -0,0 +1,4 @@
1
+ node_modules
2
+ .git
3
+ dist
4
+ .env*
@@ -0,0 +1 @@
1
+ This is a ChatGPT/MCP app built with Skybridge. ALWAYS use the `chatgpt-app-builder` skill when planning or updating the codebase.
@@ -0,0 +1,53 @@
1
+ # syntax=docker/dockerfile:1
2
+
3
+ # Dockerfile for a Skybridge MCP server.
4
+ #
5
+ # Detects npm, yarn, or pnpm from the lockfile in your project.
6
+ # (For bun or deno, adapt the install/build/prune commands below.)
7
+
8
+ # Build stage: install deps, compile the app, then prune dev deps.
9
+ FROM node:24-slim AS build
10
+ WORKDIR /app
11
+
12
+ COPY package.json yarn.lock* package-lock.json* pnpm-lock.yaml* .npmrc* ./
13
+ RUN --mount=type=cache,target=/root/.npm \
14
+ --mount=type=cache,target=/usr/local/share/.cache/yarn \
15
+ --mount=type=cache,target=/root/.local/share/pnpm/store \
16
+ if [ -f package-lock.json ]; then \
17
+ npm ci; \
18
+ elif [ -f yarn.lock ]; then \
19
+ corepack enable yarn && yarn install --frozen-lockfile; \
20
+ elif [ -f pnpm-lock.yaml ]; then \
21
+ corepack enable pnpm && pnpm install --frozen-lockfile; \
22
+ else \
23
+ echo "No lockfile found." && exit 1; \
24
+ fi
25
+
26
+ ENV NODE_ENV=production
27
+
28
+ COPY . .
29
+ RUN if [ -f package-lock.json ]; then \
30
+ npm run build && npm prune --omit=dev; \
31
+ elif [ -f yarn.lock ]; then \
32
+ corepack enable yarn && yarn build && yarn install --frozen-lockfile --production=true; \
33
+ elif [ -f pnpm-lock.yaml ]; then \
34
+ corepack enable pnpm && pnpm build && pnpm prune --prod; \
35
+ fi
36
+
37
+ # Runtime stage: copy built artifacts and prod deps, run as non-root.
38
+ FROM node:24-slim AS runtime
39
+ WORKDIR /app
40
+ ENV NODE_ENV=production
41
+
42
+ USER node
43
+
44
+ COPY --from=build --chown=node:node /app/node_modules ./node_modules
45
+ COPY --from=build --chown=node:node /app/dist ./dist
46
+ COPY --from=build --chown=node:node /app/package.json ./package.json
47
+
48
+ EXPOSE 3000
49
+
50
+ # Run the built server directly rather than via `npm start` / `skybridge start`.
51
+ # Each wrapper adds a process layer that can swallow SIGTERM, which makes
52
+ # graceful shutdowns time out on platforms like Cloud Run, Fly, and k8s.
53
+ CMD ["node", "dist/server.js"]