create-skybridge 0.0.0-dev.bad05f4 → 0.0.0-dev.baf2661

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/index.js CHANGED
@@ -127,6 +127,8 @@ export async function init(args = process.argv.slice(2)) {
127
127
  });
128
128
  // Rename _gitignore to .gitignore
129
129
  fs.renameSync(path.join(root, "_gitignore"), path.join(root, ".gitignore"));
130
+ // Rename _dockerignore to .dockerignore
131
+ fs.renameSync(path.join(root, "_dockerignore"), path.join(root, ".dockerignore"));
130
132
  prompts.log.success(`Project created in ${root}`);
131
133
  }
132
134
  }
@@ -18,6 +18,8 @@ describe("create-skybridge", () => {
18
18
  const name = `../../${tempDirName}//project$`;
19
19
  await init([name]);
20
20
  await fs.access(path.join(process.cwd(), tempDirName, "project", ".gitignore"));
21
+ await fs.access(path.join(process.cwd(), tempDirName, "project", ".dockerignore"));
22
+ await fs.access(path.join(process.cwd(), tempDirName, "project", "Dockerfile"));
21
23
  expect(fs.access(path.join(process.cwd(), tempDirName, "project", ".npmrc"))).rejects.toThrowError();
22
24
  });
23
25
  it("should download template from repo", { timeout: 10000 }, async () => {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "create-skybridge",
3
- "version": "0.0.0-dev.bad05f4",
3
+ "version": "0.0.0-dev.baf2661",
4
4
  "type": "module",
5
5
  "license": "MIT",
6
6
  "author": "Alpic",
@@ -27,9 +27,8 @@
27
27
  },
28
28
  "scripts": {
29
29
  "build": "tsc",
30
- "test": "pnpm run test:unit && pnpm run test:type && pnpm run test:format",
30
+ "test": "pnpm run test:unit && pnpm run test:format",
31
31
  "test:unit": "vitest run",
32
- "test:type": "tsc --noEmit",
33
32
  "format": "biome check --write --error-on-warnings",
34
33
  "test:format": "biome ci"
35
34
  }
@@ -0,0 +1,35 @@
1
+ # syntax=docker/dockerfile:1
2
+
3
+ # Multi-stage Dockerfile for a Skybridge MCP server.
4
+ #
5
+ # Defaults to npm. If you scaffolded with yarn, pnpm, or bun, swap the
6
+ # install/build/prune commands and the lockfile in the COPY line below.
7
+
8
+ # ---- Build stage ----
9
+ FROM node:24-slim AS build
10
+ WORKDIR /app
11
+
12
+ COPY package.json package-lock.json ./
13
+ RUN npm ci
14
+
15
+ COPY . .
16
+ RUN npm run build
17
+ RUN npm prune --omit=dev
18
+
19
+ # ---- Runtime stage ----
20
+ FROM node:24-slim AS runtime
21
+ WORKDIR /app
22
+ ENV NODE_ENV=production
23
+
24
+ USER node
25
+
26
+ COPY --from=build --chown=node:node /app/node_modules ./node_modules
27
+ COPY --from=build --chown=node:node /app/dist ./dist
28
+ COPY --from=build --chown=node:node /app/package.json ./package.json
29
+
30
+ EXPOSE 3000
31
+
32
+ # Run the built server directly rather than via `npm start` / `skybridge start`.
33
+ # Each wrapper adds a process layer that can swallow SIGTERM, which makes
34
+ # graceful shutdowns time out on platforms like Cloud Run, Fly, and k8s.
35
+ CMD ["node", "dist/server.js"]
@@ -44,34 +44,31 @@ This command starts:
44
44
  #### 3. Project structure
45
45
 
46
46
  ```
47
- ├── server/
48
- └── src/
49
- └── index.ts # Server entry point
50
- ├── web/
51
- │ ├── src/
52
- │ ├── widgets/ # React components (one per widget)
53
- │ │ ├── helpers.ts # Shared utilities
54
- │ │ └── index.css # Global styles
55
- │ └── vite.config.ts
47
+ ├── src/
48
+ ├── server.ts # Server entry point
49
+ ├── views/ # React components (one per view)
50
+ ├── components/ # Shared UI components
51
+ │ ├── helpers.ts # Shared utilities
52
+ └── index.css # Global styles
53
+ ├── vite.config.ts
56
54
  ├── alpic.json # Deployment config
57
- ├── nodemon.json # Dev server config
58
55
  └── package.json
59
56
  ```
60
57
 
61
- ### Create your first widget
58
+ ### Create your first view
62
59
 
63
- #### 1. Add a new widget
60
+ #### 1. Add a new view
64
61
 
65
- - Register a widget in `server/src/server.ts` with a unique name (e.g., `my-widget`) using [`registerWidget`](https://docs.skybridge.tech/api-reference/register-widget)
66
- - Create a matching React component at `web/src/widgets/my-widget.tsx`. **The file name must match the widget name exactly**.
62
+ - Register a tool in `src/server.ts` with a unique name (e.g., `my-view`) using [`registerTool`](https://docs.skybridge.tech/api-reference/register-tool) and a `view` config.
63
+ - Create a matching React component at `src/views/my-view.tsx`. **The file name must match the view name exactly**.
67
64
 
68
- #### 2. Edit widgets with Hot Module Replacement (HMR)
65
+ #### 2. Edit views with Hot Module Replacement (HMR)
69
66
 
70
- Edit and save components in `web/src/widgets/` — changes will appear instantly inside your App.
67
+ Edit and save components in `src/views/` — changes will appear instantly inside your App.
71
68
 
72
69
  #### 3. Edit server code
73
70
 
74
- Modify files in `server/` and refresh the connection with your testing MCP Client to see the changes.
71
+ Modify files in `src/` and refresh the connection with your testing MCP Client to see the changes.
75
72
 
76
73
  ### Testing your App
77
74
 
@@ -0,0 +1,4 @@
1
+ node_modules
2
+ .git
3
+ dist
4
+ .env*
@@ -2,4 +2,5 @@ node_modules/
2
2
  dist/
3
3
  .env*
4
4
  .DS_store
5
- *.tsbuildinfo
5
+ *.tsbuildinfo
6
+ .skybridge/
@@ -10,9 +10,9 @@ case `uname` in
10
10
  esac
11
11
 
12
12
  if [ -z "$NODE_PATH" ]; then
13
- export NODE_PATH="/home/runner/work/skybridge/skybridge/node_modules/.pnpm/skybridge@0.35.14_@modelcontextprotocol+sdk@1.29.0_zod@4.3.6__@skybridge+devtools@0.35._9ddb2dfa56e61c7e1e71871404a50ada/node_modules/skybridge/node_modules:/home/runner/work/skybridge/skybridge/node_modules/.pnpm/skybridge@0.35.14_@modelcontextprotocol+sdk@1.29.0_zod@4.3.6__@skybridge+devtools@0.35._9ddb2dfa56e61c7e1e71871404a50ada/node_modules:/home/runner/work/skybridge/skybridge/node_modules/.pnpm/node_modules"
13
+ export NODE_PATH="/home/runner/work/skybridge/skybridge/node_modules/.pnpm/node_modules"
14
14
  else
15
- export NODE_PATH="/home/runner/work/skybridge/skybridge/node_modules/.pnpm/skybridge@0.35.14_@modelcontextprotocol+sdk@1.29.0_zod@4.3.6__@skybridge+devtools@0.35._9ddb2dfa56e61c7e1e71871404a50ada/node_modules/skybridge/node_modules:/home/runner/work/skybridge/skybridge/node_modules/.pnpm/skybridge@0.35.14_@modelcontextprotocol+sdk@1.29.0_zod@4.3.6__@skybridge+devtools@0.35._9ddb2dfa56e61c7e1e71871404a50ada/node_modules:/home/runner/work/skybridge/skybridge/node_modules/.pnpm/node_modules:$NODE_PATH"
15
+ export NODE_PATH="/home/runner/work/skybridge/skybridge/node_modules/.pnpm/node_modules:$NODE_PATH"
16
16
  fi
17
17
  if [ -x "$basedir/node" ]; then
18
18
  exec "$basedir/node" "$basedir/../skybridge/bin/run.js" "$@"
@@ -10,9 +10,9 @@ case `uname` in
10
10
  esac
11
11
 
12
12
  if [ -z "$NODE_PATH" ]; then
13
- export NODE_PATH="/home/runner/work/skybridge/skybridge/node_modules/.pnpm/skybridge@0.35.14_@modelcontextprotocol+sdk@1.29.0_zod@4.3.6__@skybridge+devtools@0.35._9ddb2dfa56e61c7e1e71871404a50ada/node_modules/skybridge/node_modules:/home/runner/work/skybridge/skybridge/node_modules/.pnpm/skybridge@0.35.14_@modelcontextprotocol+sdk@1.29.0_zod@4.3.6__@skybridge+devtools@0.35._9ddb2dfa56e61c7e1e71871404a50ada/node_modules:/home/runner/work/skybridge/skybridge/node_modules/.pnpm/node_modules"
13
+ export NODE_PATH="/home/runner/work/skybridge/skybridge/node_modules/.pnpm/node_modules"
14
14
  else
15
- export NODE_PATH="/home/runner/work/skybridge/skybridge/node_modules/.pnpm/skybridge@0.35.14_@modelcontextprotocol+sdk@1.29.0_zod@4.3.6__@skybridge+devtools@0.35._9ddb2dfa56e61c7e1e71871404a50ada/node_modules/skybridge/node_modules:/home/runner/work/skybridge/skybridge/node_modules/.pnpm/skybridge@0.35.14_@modelcontextprotocol+sdk@1.29.0_zod@4.3.6__@skybridge+devtools@0.35._9ddb2dfa56e61c7e1e71871404a50ada/node_modules:/home/runner/work/skybridge/skybridge/node_modules/.pnpm/node_modules:$NODE_PATH"
15
+ export NODE_PATH="/home/runner/work/skybridge/skybridge/node_modules/.pnpm/node_modules:$NODE_PATH"
16
16
  fi
17
17
  if [ -x "$basedir/node" ]; then
18
18
  exec "$basedir/node" "$basedir/../skybridge/bin/run.js" "$@"
@@ -6,6 +6,7 @@
6
6
  "type": "module",
7
7
  "scripts": {
8
8
  "dev": "skybridge dev",
9
+ "dev:tunnel": "skybridge dev --tunnel",
9
10
  "build": "skybridge build",
10
11
  "start": "skybridge start",
11
12
  "deploy": "alpic deploy"
@@ -14,14 +15,14 @@
14
15
  "@modelcontextprotocol/sdk": "^1.29.0",
15
16
  "react": "^19.2.4",
16
17
  "react-dom": "^19.2.4",
17
- "skybridge": ">=0.35.14 <1.0.0",
18
+ "skybridge": ">=0.0.0-dev.baf2661",
18
19
  "vite": "^8.0.3",
19
20
  "zod": "^4.3.6"
20
21
  },
21
22
  "devDependencies": {
22
23
  "@skybridge/devtools": ">=0.35.14 <1.0.0",
23
- "@types/react": "^19.2.14",
24
24
  "@types/node": "^24.12.0",
25
+ "@types/react": "^19.2.14",
25
26
  "@types/react-dom": "^19.2.3",
26
27
  "@vitejs/plugin-react": "^6.0.1",
27
28
  "alpic": "^1.104.1",
@@ -0,0 +1,22 @@
1
+ export function Ball({
2
+ question,
3
+ answer,
4
+ }: {
5
+ question?: string;
6
+ answer?: string;
7
+ }) {
8
+ return (
9
+ <div className="container">
10
+ <div className="ball">
11
+ {answer ? (
12
+ <>
13
+ <div className="question">{question}</div>
14
+ <div className="answer">{answer}</div>
15
+ </>
16
+ ) : (
17
+ <div className="question">Shaking...</div>
18
+ )}
19
+ </div>
20
+ </div>
21
+ );
22
+ }
@@ -0,0 +1,4 @@
1
+ import { generateHelpers } from "skybridge/web";
2
+ import type { AppType } from "./server.js";
3
+
4
+ export const { useToolInfo, useCallTool } = generateHelpers<AppType>();
@@ -25,27 +25,26 @@ const server = new McpServer(
25
25
  version: "0.0.1",
26
26
  },
27
27
  { capabilities: {} },
28
- ).registerWidget(
29
- "magic-8-ball",
30
- {
31
- description: "Magic 8 Ball",
32
- },
28
+ ).registerTool(
33
29
  {
30
+ name: "magic-8-ball",
34
31
  description: "For fortune-telling or seeking advice.",
35
32
  inputSchema: {
36
33
  question: z.string().describe("The user question."),
37
34
  },
35
+ view: {
36
+ component: "magic-8-ball",
37
+ description: "Magic 8 Ball",
38
+ },
38
39
  },
39
40
  async ({ question }) => {
40
41
  try {
41
- // deterministic answer
42
42
  const hash = question
43
43
  .split("")
44
44
  .reduce((acc, char) => acc + char.charCodeAt(0), 0);
45
45
  const answer = Answers[hash % Answers.length];
46
46
  return {
47
47
  structuredContent: { answer },
48
- content: [],
49
48
  isError: false,
50
49
  };
51
50
  } catch (error) {
@@ -0,0 +1,10 @@
1
+ import "@/index.css";
2
+
3
+ import { Ball } from "../components/ball.js";
4
+ import { useToolInfo } from "../helpers.js";
5
+
6
+ export default function Magic8Ball() {
7
+ const { input, output } = useToolInfo<"magic-8-ball">();
8
+
9
+ return <Ball question={input?.question} answer={output?.answer} />;
10
+ }
@@ -2,12 +2,10 @@
2
2
  "extends": "skybridge/tsconfig",
3
3
 
4
4
  "compilerOptions": {
5
- "outDir": "dist",
6
- "types": ["node", "vite/client"],
7
5
  "paths": {
8
- "@/*": ["./web/src/*"]
6
+ "@/*": ["./src/*"]
9
7
  }
10
8
  },
11
9
 
12
- "include": ["server/src", "web/src"]
10
+ "include": ["src", ".skybridge/**/*.d.ts"]
13
11
  }
@@ -0,0 +1,13 @@
1
+ import path from "node:path";
2
+ import react from "@vitejs/plugin-react";
3
+ import { skybridge } from "skybridge/vite";
4
+ import { defineConfig } from "vite";
5
+
6
+ export default defineConfig({
7
+ plugins: [skybridge(), react()],
8
+ resolve: {
9
+ alias: {
10
+ "@": path.resolve(__dirname, "./src"),
11
+ },
12
+ },
13
+ });
@@ -1,4 +0,0 @@
1
- import { generateHelpers } from "skybridge/web";
2
- import type { AppType } from "../../server/src/index.js";
3
-
4
- export const { useToolInfo } = generateHelpers<AppType>();
@@ -1,27 +0,0 @@
1
- import "@/index.css";
2
-
3
- import { mountWidget } from "skybridge/web";
4
- import { useToolInfo } from "../helpers.js";
5
-
6
- function Magic8Ball() {
7
- const { input, output } = useToolInfo<"magic-8-ball">();
8
-
9
- return (
10
- <div className="container">
11
- <div className="ball">
12
- {output ? (
13
- <>
14
- <div className="question">{input.question}</div>
15
- <div className="answer">{output.answer}</div>
16
- </>
17
- ) : (
18
- <div className="question">Shaking...</div>
19
- )}
20
- </div>
21
- </div>
22
- );
23
- }
24
-
25
- export default Magic8Ball;
26
-
27
- mountWidget(<Magic8Ball />);
@@ -1,15 +0,0 @@
1
- import path from "node:path";
2
- import react from "@vitejs/plugin-react";
3
- import { skybridge } from "skybridge/web";
4
- import { defineConfig, type PluginOption } from "vite";
5
-
6
- // https://vite.dev/config/
7
- export default defineConfig({
8
- plugins: [skybridge() as PluginOption, react()],
9
- root: __dirname,
10
- resolve: {
11
- alias: {
12
- "@": path.resolve(__dirname, "./src"),
13
- },
14
- },
15
- });
File without changes