create-skybridge 0.0.0-dev.c9794a7 → 0.0.0-dev.c9a9356

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 (77) hide show
  1. package/dist/index.js +318 -212
  2. package/package.json +8 -7
  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/{template → templates/blank}/_gitignore +2 -1
  8. package/templates/blank/node_modules/.bin/alpic +21 -0
  9. package/templates/blank/node_modules/.bin/sb +21 -0
  10. package/templates/blank/node_modules/.bin/skybridge +21 -0
  11. package/{template → templates/blank}/node_modules/.bin/tsc +2 -2
  12. package/{template → templates/blank}/node_modules/.bin/tsserver +2 -2
  13. package/templates/blank/node_modules/.bin/vite +21 -0
  14. package/templates/blank/package.json +28 -0
  15. package/templates/blank/src/helpers.ts +4 -0
  16. package/templates/blank/src/server.ts +21 -0
  17. package/templates/blank/src/vite-manifest.d.ts +4 -0
  18. package/templates/blank/tsconfig.json +5 -0
  19. package/templates/blank/vite.config.ts +6 -0
  20. package/templates/demo/.dockerignore +4 -0
  21. package/templates/demo/AGENTS.md +1 -0
  22. package/templates/demo/Dockerfile +53 -0
  23. package/templates/demo/README.md +95 -0
  24. package/templates/demo/_gitignore +6 -0
  25. package/templates/demo/alpic.json +3 -0
  26. package/templates/demo/node_modules/.bin/alpic +21 -0
  27. package/templates/demo/node_modules/.bin/sb +21 -0
  28. package/templates/demo/node_modules/.bin/skybridge +21 -0
  29. package/templates/demo/node_modules/.bin/tsc +21 -0
  30. package/templates/demo/node_modules/.bin/tsserver +21 -0
  31. package/templates/demo/node_modules/.bin/vite +21 -0
  32. package/{template → templates/demo}/package.json +15 -7
  33. package/templates/demo/src/helpers.ts +4 -0
  34. package/templates/demo/src/index.css +59 -0
  35. package/templates/demo/src/server.ts +99 -0
  36. package/templates/demo/src/views/components/doc-link.tsx +22 -0
  37. package/templates/demo/src/views/components/doc.tsx +21 -0
  38. package/templates/demo/src/views/components/nav.tsx +31 -0
  39. package/templates/demo/src/views/components/progress.tsx +35 -0
  40. package/templates/demo/src/views/components/steps/outro.tsx +68 -0
  41. package/templates/demo/src/views/components/steps/state.tsx +47 -0
  42. package/templates/demo/src/views/components/steps/tool-call.tsx +53 -0
  43. package/templates/demo/src/views/components/steps/tool-output.tsx +40 -0
  44. package/templates/demo/src/views/images/mascot/beret.png +0 -0
  45. package/templates/demo/src/views/images/mascot/chapka.png +0 -0
  46. package/templates/demo/src/views/images/mascot/cowboy-hat.png +0 -0
  47. package/templates/demo/src/views/images/mascot/fez.png +0 -0
  48. package/templates/demo/src/views/images/mascot/jester-hat.png +0 -0
  49. package/templates/demo/src/views/images/mascot/mitre.png +0 -0
  50. package/templates/demo/src/views/images/mascot/non-la.png +0 -0
  51. package/templates/demo/src/views/images/mascot/original.png +0 -0
  52. package/templates/demo/src/views/images/mascot/propeller-beanie.png +0 -0
  53. package/templates/demo/src/views/images/mascot/ski-mask.png +0 -0
  54. package/templates/demo/src/views/images/mascot/sombrero.png +0 -0
  55. package/templates/demo/src/views/images/mascot/top-hat.png +0 -0
  56. package/templates/demo/src/views/images/mascot/viking-helmet.png +0 -0
  57. package/templates/demo/src/views/onboarding.tsx +63 -0
  58. package/templates/demo/src/views/use-mascot.ts +60 -0
  59. package/templates/demo/src/vite-manifest.d.ts +4 -0
  60. package/templates/demo/tsconfig.json +11 -0
  61. package/templates/demo/vite.config.ts +14 -0
  62. package/dist/index.test.d.ts +0 -1
  63. package/dist/index.test.js +0 -33
  64. package/template/AGENTS.md +0 -1
  65. package/template/README.md +0 -97
  66. package/template/node_modules/.bin/alpic +0 -21
  67. package/template/node_modules/.bin/sb +0 -21
  68. package/template/node_modules/.bin/skybridge +0 -21
  69. package/template/node_modules/.bin/vite +0 -21
  70. package/template/server/src/index.ts +0 -62
  71. package/template/tsconfig.json +0 -9
  72. package/template/web/src/helpers.ts +0 -4
  73. package/template/web/src/index.css +0 -154
  74. package/template/web/src/widgets/magic-8-ball.tsx +0 -27
  75. package/template/web/vite.config.ts +0 -15
  76. /package/{template → templates/blank}/alpic.json +0 -0
  77. /package/{template → templates/demo}/node_modules/.bin/tsx +0 -0
package/dist/index.js CHANGED
@@ -1,251 +1,371 @@
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";
6
- import { downloadTemplate } from "giget";
5
+ import spawn from "cross-spawn";
7
6
  import mri from "mri";
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
14
16
 
15
- Options:
16
- -h, --help show this help message
17
- --repo <uri> use a git repository instead of the built-in template
18
- --overwrite remove existing files in target directory
19
- --immediate install dependencies and start development server
17
+ Arguments:
18
+ path Where the project will be created. Prompted when omitted.
20
19
 
21
- Repository URI formats:
22
- github:user/repo
23
- gitlab:user/repo/subdirectory
24
- bitbucket:user/repo#branch
20
+ Options:
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
25
28
 
26
- Examples:
27
- create-skybridge my-app
28
- create-skybridge my-app --repo github:alpic-ai/skybridge/examples/ecom-carousel
29
- create-skybridge . --overwrite --immediate
30
- `;
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
+ };
31
61
  export async function init(args = process.argv.slice(2)) {
32
62
  const argv = mri(args, {
33
- boolean: ["help", "overwrite", "immediate"],
34
- string: ["repo"],
63
+ boolean: ["help", "blank", "overwrite", "skip-skills", "start", "yes"],
64
+ string: ["pm"],
35
65
  alias: { h: "help" },
36
66
  });
37
- const argTargetDir = argv._[0]
38
- ? sanitizeTargetDir(String(argv._[0]))
39
- : undefined;
40
- const argRepo = argv.repo;
41
- const argOverwrite = argv.overwrite;
42
- const argImmediate = argv.immediate;
43
- const help = argv.help;
44
- if (help) {
45
- console.log(helpMessage);
67
+ if (argv.help) {
68
+ console.log(HELP_MESSAGE);
46
69
  return;
47
70
  }
48
- const interactive = process.stdin.isTTY;
49
- const cancel = () => prompts.cancel("Operation cancelled");
50
- // 1. Get project name and target dir
51
- 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
52
83
  if (!targetDir) {
53
- if (interactive) {
54
- const projectName = await prompts.text({
55
- message: "Project name:",
56
- defaultValue: defaultProjectName,
57
- placeholder: defaultProjectName,
58
- validate: (value) => {
59
- return !value || sanitizeTargetDir(value).length > 0
60
- ? undefined
61
- : "Invalid project name";
62
- },
63
- });
64
- if (prompts.isCancel(projectName)) {
65
- return cancel();
66
- }
67
- targetDir = sanitizeTargetDir(projectName);
68
- }
69
- else {
70
- 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();
71
94
  }
95
+ targetDir = sanitizeTargetDir(choice);
72
96
  }
73
- // 2. Handle directory if exist and not empty
97
+ // 2. Existing-directory handling
74
98
  if (fs.existsSync(targetDir) && !isEmpty(targetDir)) {
75
- let overwrite = argOverwrite ? "yes" : undefined;
76
- if (!overwrite) {
77
- if (interactive) {
78
- const res = await prompts.select({
79
- message: (targetDir === "."
80
- ? "Current directory"
81
- : `Target directory "${targetDir}"`) +
82
- ` is not empty. Please choose how to proceed:`,
83
- options: [
84
- {
85
- label: "Cancel operation",
86
- value: "no",
87
- },
88
- {
89
- label: "Remove existing files and continue",
90
- value: "yes",
91
- },
92
- ],
93
- });
94
- if (prompts.isCancel(res)) {
95
- return cancel();
96
- }
97
- overwrite = res;
98
- }
99
- else {
100
- 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();
101
113
  }
114
+ Spinner.start(`Cleaning up ${targetDir}`);
115
+ emptyDir(targetDir);
116
+ Spinner.stop(`Cleaned up ${targetDir}`);
102
117
  }
103
- switch (overwrite) {
104
- case "yes":
105
- emptyDir(targetDir);
106
- break;
107
- case "no":
108
- prompts.log.error("Target directory is not empty.");
109
- process.exit(1);
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();
110
146
  }
147
+ template = choice;
111
148
  }
112
- const root = path.join(process.cwd(), targetDir);
113
- // 3. Download from repo or copy template
149
+ // 4. Copy template
150
+ const root = path.resolve(targetDir);
151
+ Spinner.start(`Copying ${template} template`);
114
152
  try {
115
- if (argRepo) {
116
- prompts.log.step(`Downloading ${argRepo}...`);
117
- await downloadTemplate(argRepo, { dir: root });
118
- prompts.log.success(`Project created in ${root}`);
119
- }
120
- else {
121
- prompts.log.step(`Copying template...`);
122
- const templateDir = fileURLToPath(new URL("../template", import.meta.url));
123
- // Copy template to target directory
124
- fs.cpSync(templateDir, root, {
125
- recursive: true,
126
- filter: (src) => [".npmrc"].every((file) => !src.endsWith(file)),
127
- });
128
- // Rename _gitignore to .gitignore
129
- fs.renameSync(path.join(root, "_gitignore"), path.join(root, ".gitignore"));
130
- prompts.log.success(`Project created in ${root}`);
153
+ const templateDir = fileURLToPath(new URL(`../templates/${template}`, import.meta.url));
154
+ fs.cpSync(templateDir, root, {
155
+ recursive: true,
156
+ filter: (src) => [".npmrc"].every((file) => !src.endsWith(file)),
157
+ });
158
+ const gitignoreSource = path.join(root, "_gitignore");
159
+ if (fs.existsSync(gitignoreSource)) {
160
+ fs.renameSync(gitignoreSource, path.join(root, ".gitignore"));
131
161
  }
162
+ Spinner.stop(`Copied ${template} template`);
132
163
  }
133
164
  catch (error) {
134
- prompts.log.error("Failed to create project from template");
135
- console.error(error);
136
- process.exit(1);
137
- }
138
- // Update project name in package.json
139
- const pkgPath = path.join(root, "package.json");
140
- if (!fs.existsSync(pkgPath)) {
141
- prompts.log.error("No package.json found in project");
142
- process.exit(1);
165
+ Spinner.error("Failed to copy template");
166
+ abort(String(error));
143
167
  }
168
+ // 5. Set package.json name to the project dir basename
144
169
  try {
145
- const pkg = JSON.parse(fs.readFileSync(pkgPath, "utf-8"));
146
- pkg.name = path.basename(root);
147
- fs.writeFileSync(pkgPath, `${JSON.stringify(pkg, null, 2)}\n`);
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`);
148
174
  }
149
175
  catch (error) {
150
- prompts.log.error("Failed to update project name in package.json");
151
- console.error(error);
152
- process.exit(1);
176
+ abort("Failed to update project name in package.json.", String(error));
153
177
  }
154
- const userAgent = process.env.npm_config_user_agent;
155
- const pkgManager = userAgent?.split(" ")[0]?.split("/")[0] || "npm";
156
- // 4. Ask about skills installation
157
- if (interactive) {
158
- const skillsResult = await prompts.confirm({
159
- message: "Install the coding agents skills? (recommended)",
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,
189
+ });
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)",
160
226
  initialValue: true,
161
227
  });
162
- if (prompts.isCancel(skillsResult)) {
228
+ if (prompts.isCancel(choice)) {
163
229
  return cancel();
164
230
  }
165
- if (skillsResult) {
166
- run([
167
- ...getPkgExecCmd(pkgManager, "skills"),
168
- "add",
169
- "alpic-ai/skybridge",
170
- "-s",
171
- "chatgpt-app-builder",
172
- ], {
173
- stdio: "inherit",
174
- cwd: targetDir,
175
- });
176
- }
231
+ installSkills = choice;
177
232
  }
178
- // 5. Ask about immediate installation
179
- let immediate = argImmediate;
180
- if (immediate === undefined) {
181
- if (interactive) {
182
- const immediateResult = await prompts.confirm({
183
- message: `Install with ${pkgManager} and start now?`,
184
- });
185
- if (prompts.isCancel(immediateResult)) {
186
- return cancel();
187
- }
188
- immediate = immediateResult;
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`);
189
251
  }
190
252
  else {
191
- 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`.");
192
256
  }
193
257
  }
194
- const installCmd = [pkgManager, "install"];
195
- const runCmd = [pkgManager];
196
- switch (pkgManager) {
197
- case "yarn":
198
- case "pnpm":
258
+ // 7. Package manager — autodetect, prompt only if detection fails (interactive)
259
+ if (!pm) {
260
+ pm = detectPackageManager() || "npm";
261
+ }
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);
320
+ }
321
+ function abort(...lines) {
322
+ for (const line of lines) {
323
+ prompts.log.error(line);
324
+ }
325
+ prompts.outro("Aborted");
326
+ process.exit(1);
327
+ }
328
+ function parsePackageManager(value) {
329
+ switch (value) {
199
330
  case "bun":
200
- break;
331
+ return "bun";
201
332
  case "deno":
202
- runCmd.push("task");
203
- break;
333
+ return "deno";
334
+ case "npm":
335
+ return "npm";
336
+ case "pnpm":
337
+ return "pnpm";
338
+ case "yarn":
339
+ return "yarn";
204
340
  default:
205
- runCmd.push("run");
206
- }
207
- runCmd.push("dev");
208
- if (!immediate) {
209
- prompts.outro(`Done! Next steps:
210
- cd ${targetDir}
211
- ${installCmd.join(" ")}
212
- ${runCmd.join(" ")}
213
- `);
214
- return;
341
+ return undefined;
215
342
  }
216
- prompts.log.step(`Installing dependencies with ${pkgManager}...`);
217
- run(installCmd, {
218
- stdio: "inherit",
219
- cwd: root,
220
- });
221
- prompts.log.step("Starting dev server...");
222
- run(runCmd, {
223
- stdio: "inherit",
224
- cwd: root,
225
- });
226
343
  }
227
- function run([command, ...args], options) {
228
- const { status, error } = spawnSync(command, args, options);
229
- if (status != null && status > 0) {
230
- process.exit(status);
344
+ function detectPackageManager() {
345
+ const userAgent = process.env.npm_config_user_agent;
346
+ if (!userAgent) {
347
+ return undefined;
231
348
  }
232
- if (error) {
233
- console.error(`\n${command} ${args.join(" ")} error!`);
234
- console.error(error);
235
- process.exit(1);
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];
236
362
  }
237
363
  }
364
+ function scriptCommand(pm, script) {
365
+ return [pm, ...scriptArgs(pm, script)].join(" ");
366
+ }
238
367
  function sanitizeTargetDir(targetDir) {
239
- return (targetDir
240
- .trim()
241
- // Only keep alphanumeric, dash, underscore, dot, @, /
242
- .replace(/[^a-zA-Z0-9\-_.@/]/g, "")
243
- // Prevent path traversal
244
- .replace(/\.\./g, "")
245
- // Collapse multiple slashes
246
- .replace(/\/+/g, "/")
247
- // Remove leading/trailing slashes
248
- .replace(/^\/+|\/+$/g, ""));
368
+ return targetDir.trim().replace(/\/+$/g, "");
249
369
  }
250
370
  // Skip user's SPEC.md and IDE/agent preferences (.idea, .claude, etc.)
251
371
  function isSkippedEntry(entry) {
@@ -267,17 +387,3 @@ function emptyDir(dir) {
267
387
  fs.rmSync(path.join(dir, entry.name), { recursive: true, force: true });
268
388
  }
269
389
  }
270
- function getPkgExecCmd(pkgManager, cmd) {
271
- switch (pkgManager) {
272
- case "yarn":
273
- return ["yarn", "dlx", cmd];
274
- case "pnpm":
275
- return ["pnpm", "dlx", cmd];
276
- case "bun":
277
- return ["bunx", cmd];
278
- case "deno":
279
- return ["deno", "run", "-A", `npm:${cmd}`];
280
- default:
281
- return ["npx", cmd];
282
- }
283
- }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "create-skybridge",
3
- "version": "0.0.0-dev.c9794a7",
3
+ "version": "0.0.0-dev.c9a9356",
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
20
  "@clack/prompts": "^1.1.0",
21
- "giget": "^3.1.2",
21
+ "cross-spawn": "^7.0.6",
22
22
  "mri": "^1.2.0"
23
23
  },
24
24
  "devDependencies": {
25
- "typescript": "^5.9.3",
26
- "vitest": "^4.1.0"
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.