skybridge 0.0.0-dev.f6bca06 → 0.0.0-dev.f727bfa

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 (224) hide show
  1. package/README.md +24 -16
  2. package/dist/cli/detect-port.d.ts +18 -0
  3. package/dist/cli/detect-port.js +61 -0
  4. package/dist/cli/detect-port.js.map +1 -0
  5. package/dist/cli/header.js +1 -1
  6. package/dist/cli/header.js.map +1 -1
  7. package/dist/cli/tunnel-control-server.d.ts +9 -0
  8. package/dist/cli/tunnel-control-server.js +31 -0
  9. package/dist/cli/tunnel-control-server.js.map +1 -0
  10. package/dist/cli/tunnel-control-server.test.js +39 -0
  11. package/dist/cli/tunnel-control-server.test.js.map +1 -0
  12. package/dist/cli/tunnel-handler.d.ts +3 -0
  13. package/dist/cli/tunnel-handler.js +48 -0
  14. package/dist/cli/tunnel-handler.js.map +1 -0
  15. package/dist/cli/tunnel-handler.test.js +105 -0
  16. package/dist/cli/tunnel-handler.test.js.map +1 -0
  17. package/dist/cli/tunnel.d.ts +57 -0
  18. package/dist/cli/tunnel.js +154 -0
  19. package/dist/cli/tunnel.js.map +1 -0
  20. package/dist/cli/tunnel.test.d.ts +1 -0
  21. package/dist/cli/tunnel.test.js +190 -0
  22. package/dist/cli/tunnel.test.js.map +1 -0
  23. package/dist/cli/types.d.ts +5 -0
  24. package/dist/cli/types.js +2 -0
  25. package/dist/cli/types.js.map +1 -0
  26. package/dist/cli/use-messages.d.ts +3 -0
  27. package/dist/cli/use-messages.js +11 -0
  28. package/dist/cli/use-messages.js.map +1 -0
  29. package/dist/cli/use-nodemon.d.ts +2 -6
  30. package/dist/cli/use-nodemon.js +19 -16
  31. package/dist/cli/use-nodemon.js.map +1 -1
  32. package/dist/cli/use-tunnel.d.ts +14 -0
  33. package/dist/cli/use-tunnel.js +131 -0
  34. package/dist/cli/use-tunnel.js.map +1 -0
  35. package/dist/cli/use-typescript-check.d.ts +1 -0
  36. package/dist/cli/use-typescript-check.js +41 -6
  37. package/dist/cli/use-typescript-check.js.map +1 -1
  38. package/dist/commands/build.js +28 -7
  39. package/dist/commands/build.js.map +1 -1
  40. package/dist/commands/dev.d.ts +3 -1
  41. package/dist/commands/dev.js +50 -8
  42. package/dist/commands/dev.js.map +1 -1
  43. package/dist/commands/start.d.ts +3 -1
  44. package/dist/commands/start.js +30 -9
  45. package/dist/commands/start.js.map +1 -1
  46. package/dist/server/asset-base-url-transform-plugin.d.ts +5 -6
  47. package/dist/server/asset-base-url-transform-plugin.js +9 -10
  48. package/dist/server/asset-base-url-transform-plugin.js.map +1 -1
  49. package/dist/server/asset-base-url-transform-plugin.test.js +41 -13
  50. package/dist/server/asset-base-url-transform-plugin.test.js.map +1 -1
  51. package/dist/server/content-helpers.d.ts +27 -0
  52. package/dist/server/content-helpers.js +46 -0
  53. package/dist/server/content-helpers.js.map +1 -0
  54. package/dist/server/content-helpers.test.d.ts +1 -0
  55. package/dist/server/content-helpers.test.js +70 -0
  56. package/dist/server/content-helpers.test.js.map +1 -0
  57. package/dist/server/express.d.ts +13 -3
  58. package/dist/server/express.js +53 -21
  59. package/dist/server/express.js.map +1 -1
  60. package/dist/server/express.test.d.ts +1 -0
  61. package/dist/server/express.test.js +348 -0
  62. package/dist/server/express.test.js.map +1 -0
  63. package/dist/server/index.d.ts +5 -3
  64. package/dist/server/index.js +3 -2
  65. package/dist/server/index.js.map +1 -1
  66. package/dist/server/inferUtilityTypes.d.ts +6 -6
  67. package/dist/server/metric.d.ts +14 -0
  68. package/dist/server/metric.js +62 -0
  69. package/dist/server/metric.js.map +1 -0
  70. package/dist/server/middleware.d.ts +124 -0
  71. package/dist/server/middleware.js +93 -0
  72. package/dist/server/middleware.js.map +1 -0
  73. package/dist/server/middleware.test-d.d.ts +1 -0
  74. package/dist/server/middleware.test-d.js +75 -0
  75. package/dist/server/middleware.test-d.js.map +1 -0
  76. package/dist/server/middleware.test.d.ts +1 -0
  77. package/dist/server/middleware.test.js +493 -0
  78. package/dist/server/middleware.test.js.map +1 -0
  79. package/dist/server/server.d.ts +133 -69
  80. package/dist/server/server.js +329 -94
  81. package/dist/server/server.js.map +1 -1
  82. package/dist/server/templateHelper.d.ts +5 -6
  83. package/dist/server/templateHelper.js.map +1 -1
  84. package/dist/server/templates/development.hbs +2 -57
  85. package/dist/server/templates/production.hbs +1 -1
  86. package/dist/server/tunnel-proxy-router.d.ts +7 -0
  87. package/dist/server/tunnel-proxy-router.js +110 -0
  88. package/dist/server/tunnel-proxy-router.js.map +1 -0
  89. package/dist/server/tunnel-proxy-router.test.d.ts +1 -0
  90. package/dist/server/tunnel-proxy-router.test.js +229 -0
  91. package/dist/server/tunnel-proxy-router.test.js.map +1 -0
  92. package/dist/server/viewsDevServer.d.ts +14 -0
  93. package/dist/server/viewsDevServer.js +45 -0
  94. package/dist/server/viewsDevServer.js.map +1 -0
  95. package/dist/test/utils.d.ts +16 -244
  96. package/dist/test/utils.js +42 -37
  97. package/dist/test/utils.js.map +1 -1
  98. package/dist/test/view.test.d.ts +1 -0
  99. package/dist/test/view.test.js +523 -0
  100. package/dist/test/view.test.js.map +1 -0
  101. package/dist/version.d.ts +1 -0
  102. package/dist/version.js +5 -0
  103. package/dist/version.js.map +1 -0
  104. package/dist/web/bridges/apps-sdk/adaptor.d.ts +8 -6
  105. package/dist/web/bridges/apps-sdk/adaptor.js +46 -17
  106. package/dist/web/bridges/apps-sdk/adaptor.js.map +1 -1
  107. package/dist/web/bridges/apps-sdk/bridge.d.ts +1 -1
  108. package/dist/web/bridges/apps-sdk/index.d.ts +1 -1
  109. package/dist/web/bridges/apps-sdk/index.js.map +1 -1
  110. package/dist/web/bridges/apps-sdk/types.d.ts +24 -13
  111. package/dist/web/bridges/apps-sdk/types.js.map +1 -1
  112. package/dist/web/bridges/mcp-app/adaptor.d.ts +20 -8
  113. package/dist/web/bridges/mcp-app/adaptor.js +141 -61
  114. package/dist/web/bridges/mcp-app/adaptor.js.map +1 -1
  115. package/dist/web/bridges/mcp-app/bridge.d.ts +13 -30
  116. package/dist/web/bridges/mcp-app/bridge.js +43 -196
  117. package/dist/web/bridges/mcp-app/bridge.js.map +1 -1
  118. package/dist/web/bridges/mcp-app/use-mcp-app-context.d.ts +5 -3
  119. package/dist/web/bridges/mcp-app/use-mcp-app-context.js +2 -2
  120. package/dist/web/bridges/mcp-app/use-mcp-app-context.js.map +1 -1
  121. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js +1 -41
  122. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js.map +1 -1
  123. package/dist/web/bridges/types.d.ts +22 -11
  124. package/dist/web/components/modal-provider.js +3 -5
  125. package/dist/web/components/modal-provider.js.map +1 -1
  126. package/dist/web/create-store.js +17 -3
  127. package/dist/web/create-store.js.map +1 -1
  128. package/dist/web/create-store.test.js +23 -20
  129. package/dist/web/create-store.test.js.map +1 -1
  130. package/dist/web/data-llm.d.ts +1 -1
  131. package/dist/web/data-llm.js +3 -3
  132. package/dist/web/data-llm.js.map +1 -1
  133. package/dist/web/data-llm.test.js +33 -30
  134. package/dist/web/data-llm.test.js.map +1 -1
  135. package/dist/web/generate-helpers.d.ts +20 -18
  136. package/dist/web/generate-helpers.js +20 -18
  137. package/dist/web/generate-helpers.js.map +1 -1
  138. package/dist/web/generate-helpers.test-d.js +26 -26
  139. package/dist/web/generate-helpers.test-d.js.map +1 -1
  140. package/dist/web/helpers/state.d.ts +2 -2
  141. package/dist/web/helpers/state.js +11 -11
  142. package/dist/web/helpers/state.js.map +1 -1
  143. package/dist/web/helpers/state.test.js +9 -9
  144. package/dist/web/helpers/state.test.js.map +1 -1
  145. package/dist/web/hooks/index.d.ts +2 -2
  146. package/dist/web/hooks/index.js +1 -1
  147. package/dist/web/hooks/index.js.map +1 -1
  148. package/dist/web/hooks/test/utils.js +4 -0
  149. package/dist/web/hooks/test/utils.js.map +1 -1
  150. package/dist/web/hooks/use-display-mode.d.ts +3 -3
  151. package/dist/web/hooks/use-display-mode.js.map +1 -1
  152. package/dist/web/hooks/use-display-mode.test-d.d.ts +1 -0
  153. package/dist/web/hooks/use-display-mode.test-d.js +8 -0
  154. package/dist/web/hooks/use-display-mode.test-d.js.map +1 -0
  155. package/dist/web/hooks/use-files.d.ts +2 -1
  156. package/dist/web/hooks/use-files.js +1 -0
  157. package/dist/web/hooks/use-files.js.map +1 -1
  158. package/dist/web/hooks/use-files.test.js +27 -3
  159. package/dist/web/hooks/use-files.test.js.map +1 -1
  160. package/dist/web/hooks/use-layout.test.js +3 -3
  161. package/dist/web/hooks/use-layout.test.js.map +1 -1
  162. package/dist/web/hooks/use-open-external.d.ts +3 -1
  163. package/dist/web/hooks/use-open-external.js +1 -1
  164. package/dist/web/hooks/use-open-external.js.map +1 -1
  165. package/dist/web/hooks/use-open-external.test.js +26 -11
  166. package/dist/web/hooks/use-open-external.test.js.map +1 -1
  167. package/dist/web/hooks/use-request-modal.d.ts +1 -1
  168. package/dist/web/hooks/use-request-modal.js +4 -4
  169. package/dist/web/hooks/use-request-modal.js.map +1 -1
  170. package/dist/web/hooks/use-request-modal.test.js +5 -1
  171. package/dist/web/hooks/use-request-modal.test.js.map +1 -1
  172. package/dist/web/hooks/use-set-open-in-app-url.test.js +5 -11
  173. package/dist/web/hooks/use-set-open-in-app-url.test.js.map +1 -1
  174. package/dist/web/hooks/use-tool-info.test.js +1 -1
  175. package/dist/web/hooks/use-tool-info.test.js.map +1 -1
  176. package/dist/web/hooks/use-user.js +18 -2
  177. package/dist/web/hooks/use-user.js.map +1 -1
  178. package/dist/web/hooks/use-user.test.js +29 -1
  179. package/dist/web/hooks/use-user.test.js.map +1 -1
  180. package/dist/web/hooks/use-view-state.d.ts +4 -0
  181. package/dist/web/hooks/use-view-state.js +32 -0
  182. package/dist/web/hooks/use-view-state.js.map +1 -0
  183. package/dist/web/hooks/use-view-state.test.d.ts +1 -0
  184. package/dist/web/hooks/use-view-state.test.js +177 -0
  185. package/dist/web/hooks/use-view-state.test.js.map +1 -0
  186. package/dist/web/index.d.ts +1 -2
  187. package/dist/web/index.js +1 -2
  188. package/dist/web/index.js.map +1 -1
  189. package/dist/web/mount-view.d.ts +1 -0
  190. package/dist/web/{mount-widget.js → mount-view.js} +2 -2
  191. package/dist/web/mount-view.js.map +1 -0
  192. package/dist/web/plugin/plugin.d.ts +4 -1
  193. package/dist/web/plugin/plugin.js +135 -18
  194. package/dist/web/plugin/plugin.js.map +1 -1
  195. package/dist/web/plugin/scan-views.d.ts +16 -0
  196. package/dist/web/plugin/scan-views.js +88 -0
  197. package/dist/web/plugin/scan-views.js.map +1 -0
  198. package/dist/web/plugin/scan-views.test.d.ts +1 -0
  199. package/dist/web/plugin/scan-views.test.js +99 -0
  200. package/dist/web/plugin/scan-views.test.js.map +1 -0
  201. package/dist/web/plugin/transform-data-llm.js +1 -1
  202. package/dist/web/plugin/transform-data-llm.js.map +1 -1
  203. package/dist/web/plugin/validate-view.d.ts +1 -0
  204. package/dist/web/plugin/validate-view.js +9 -0
  205. package/dist/web/plugin/validate-view.js.map +1 -0
  206. package/dist/web/plugin/validate-view.test.d.ts +1 -0
  207. package/dist/web/plugin/validate-view.test.js +24 -0
  208. package/dist/web/plugin/validate-view.test.js.map +1 -0
  209. package/package.json +34 -24
  210. package/tsconfig.base.json +33 -0
  211. package/dist/server/widgetsDevServer.d.ts +0 -12
  212. package/dist/server/widgetsDevServer.js +0 -57
  213. package/dist/server/widgetsDevServer.js.map +0 -1
  214. package/dist/test/widget.test.js +0 -261
  215. package/dist/test/widget.test.js.map +0 -1
  216. package/dist/web/hooks/use-widget-state.d.ts +0 -4
  217. package/dist/web/hooks/use-widget-state.js +0 -32
  218. package/dist/web/hooks/use-widget-state.js.map +0 -1
  219. package/dist/web/hooks/use-widget-state.test.js +0 -61
  220. package/dist/web/hooks/use-widget-state.test.js.map +0 -1
  221. package/dist/web/mount-widget.d.ts +0 -1
  222. package/dist/web/mount-widget.js.map +0 -1
  223. /package/dist/{test/widget.test.d.ts → cli/tunnel-control-server.test.d.ts} +0 -0
  224. /package/dist/{web/hooks/use-widget-state.test.d.ts → cli/tunnel-handler.test.d.ts} +0 -0
@@ -0,0 +1,46 @@
1
+ /**
2
+ * Returns a base64-encoded string.
3
+ * - `Uint8Array` input is encoded via `Buffer.toString("base64")`.
4
+ * - `string` input is assumed to be **already base64-encoded** and is returned
5
+ * as-is. Passing raw/unencoded string bytes will produce invalid MCP content.
6
+ */
7
+ function toBase64(data) {
8
+ if (typeof data === "string") {
9
+ return data;
10
+ }
11
+ return Buffer.from(data).toString("base64");
12
+ }
13
+ export function text(value, annotations) {
14
+ return { type: "text", text: value, ...(annotations && { annotations }) };
15
+ }
16
+ export function image(data, mimeType, annotations) {
17
+ return {
18
+ type: "image",
19
+ data: toBase64(data),
20
+ mimeType,
21
+ ...(annotations && { annotations }),
22
+ };
23
+ }
24
+ export function audio(data, mimeType, annotations) {
25
+ return {
26
+ type: "audio",
27
+ data: toBase64(data),
28
+ mimeType,
29
+ ...(annotations && { annotations }),
30
+ };
31
+ }
32
+ export function embeddedResource(resource, annotations) {
33
+ return {
34
+ type: "resource",
35
+ resource,
36
+ ...(annotations && { annotations }),
37
+ };
38
+ }
39
+ export function resourceLink(link, annotations) {
40
+ return {
41
+ type: "resource_link",
42
+ ...link,
43
+ ...(annotations && { annotations }),
44
+ };
45
+ }
46
+ //# sourceMappingURL=content-helpers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"content-helpers.js","sourceRoot":"","sources":["../../src/server/content-helpers.ts"],"names":[],"mappings":"AAcA;;;;;GAKG;AACH,SAAS,QAAQ,CAAC,IAAyB;IACzC,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;QAC7B,OAAO,IAAI,CAAC;IACd,CAAC;IACD,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;AAC9C,CAAC;AAED,MAAM,UAAU,IAAI,CAClB,KAAa,EACb,WAAgC;IAEhC,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,GAAG,CAAC,WAAW,IAAI,EAAE,WAAW,EAAE,CAAC,EAAE,CAAC;AAC5E,CAAC;AAED,MAAM,UAAU,KAAK,CACnB,IAAyB,EACzB,QAAgB,EAChB,WAAgC;IAEhC,OAAO;QACL,IAAI,EAAE,OAAO;QACb,IAAI,EAAE,QAAQ,CAAC,IAAI,CAAC;QACpB,QAAQ;QACR,GAAG,CAAC,WAAW,IAAI,EAAE,WAAW,EAAE,CAAC;KACpC,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,KAAK,CACnB,IAAyB,EACzB,QAAgB,EAChB,WAAgC;IAEhC,OAAO;QACL,IAAI,EAAE,OAAO;QACb,IAAI,EAAE,QAAQ,CAAC,IAAI,CAAC;QACpB,QAAQ;QACR,GAAG,CAAC,WAAW,IAAI,EAAE,WAAW,EAAE,CAAC;KACpC,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,gBAAgB,CAC9B,QAEoD,EACpD,WAAgC;IAEhC,OAAO;QACL,IAAI,EAAE,UAAU;QAChB,QAAQ;QACR,GAAG,CAAC,WAAW,IAAI,EAAE,WAAW,EAAE,CAAC;KACpC,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,YAAY,CAC1B,IAOC,EACD,WAAgC;IAEhC,OAAO;QACL,IAAI,EAAE,eAAe;QACrB,GAAG,IAAI;QACP,GAAG,CAAC,WAAW,IAAI,EAAE,WAAW,EAAE,CAAC;KACpC,CAAC;AACJ,CAAC"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,70 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import { audio, embeddedResource, image, resourceLink, text, } from "./content-helpers.js";
3
+ describe("text", () => {
4
+ it("returns a TextContent without annotations when none given", () => {
5
+ expect(text("hello")).toEqual({ type: "text", text: "hello" });
6
+ });
7
+ it("includes annotations when provided", () => {
8
+ expect(text("hi", { priority: 1 })).toEqual({
9
+ type: "text",
10
+ text: "hi",
11
+ annotations: { priority: 1 },
12
+ });
13
+ });
14
+ });
15
+ describe("image", () => {
16
+ it("base64-encodes Uint8Array data", () => {
17
+ const bytes = new Uint8Array([104, 105]);
18
+ expect(image(bytes, "image/png")).toEqual({
19
+ type: "image",
20
+ data: "aGk=",
21
+ mimeType: "image/png",
22
+ });
23
+ });
24
+ it("passes string data through unchanged (caller is responsible for base64)", () => {
25
+ expect(image("YWxyZWFkeS1iNjQ=", "image/png")).toEqual({
26
+ type: "image",
27
+ data: "YWxyZWFkeS1iNjQ=",
28
+ mimeType: "image/png",
29
+ });
30
+ });
31
+ });
32
+ describe("audio", () => {
33
+ it("base64-encodes Uint8Array data", () => {
34
+ const bytes = new Uint8Array([104, 105]);
35
+ expect(audio(bytes, "audio/mpeg")).toMatchObject({
36
+ type: "audio",
37
+ data: "aGk=",
38
+ mimeType: "audio/mpeg",
39
+ });
40
+ });
41
+ });
42
+ describe("embeddedResource", () => {
43
+ it("wraps a text resource with a type tag", () => {
44
+ expect(embeddedResource({
45
+ uri: "file:///a.txt",
46
+ mimeType: "text/plain",
47
+ text: "x",
48
+ })).toEqual({
49
+ type: "resource",
50
+ resource: { uri: "file:///a.txt", mimeType: "text/plain", text: "x" },
51
+ });
52
+ });
53
+ it("wraps a blob resource with a type tag", () => {
54
+ expect(embeddedResource({ uri: "file:///a.bin", blob: "YmFy" })).toEqual({
55
+ type: "resource",
56
+ resource: { uri: "file:///a.bin", blob: "YmFy" },
57
+ });
58
+ });
59
+ });
60
+ describe("resourceLink", () => {
61
+ it("spreads link fields alongside the type tag", () => {
62
+ expect(resourceLink({ uri: "file:///a", name: "a", title: "A" })).toEqual({
63
+ type: "resource_link",
64
+ uri: "file:///a",
65
+ name: "a",
66
+ title: "A",
67
+ });
68
+ });
69
+ });
70
+ //# sourceMappingURL=content-helpers.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"content-helpers.test.js","sourceRoot":"","sources":["../../src/server/content-helpers.test.ts"],"names":[],"mappings":"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"}
@@ -1,5 +1,15 @@
1
+ import type http from "node:http";
1
2
  import express from "express";
2
- import type { McpServer } from "./server";
3
- export declare function createServer({ server, }: {
4
- server: McpServer;
3
+ import type { McpServer } from "./server.js";
4
+ export declare function createApp({ mcpServer, httpServer, customMiddleware, errorMiddleware, }: {
5
+ mcpServer: McpServer;
6
+ httpServer: http.Server;
7
+ customMiddleware?: {
8
+ path?: string;
9
+ handlers: express.RequestHandler[];
10
+ }[];
11
+ errorMiddleware?: {
12
+ path?: string;
13
+ handlers: express.ErrorRequestHandler[];
14
+ }[];
5
15
  }): Promise<express.Express>;
@@ -1,29 +1,68 @@
1
1
  import path from "node:path";
2
- import { fileURLToPath } from "node:url";
3
2
  import { StreamableHTTPServerTransport } from "@modelcontextprotocol/sdk/server/streamableHttp.js";
4
3
  import cors from "cors";
5
4
  import express from "express";
6
- export async function createServer({ server, }) {
5
+ function parseControlPort(raw) {
6
+ if (raw === undefined) {
7
+ return null;
8
+ }
9
+ const n = Number.parseInt(raw, 10);
10
+ if (!Number.isFinite(n) || n <= 0 || n >= 65536) {
11
+ return null;
12
+ }
13
+ return n;
14
+ }
15
+ function applyMiddlewares(app, middlewares) {
16
+ for (const middleware of middlewares) {
17
+ if (middleware.path) {
18
+ app.use(middleware.path, ...middleware.handlers);
19
+ }
20
+ else {
21
+ app.use(...middleware.handlers);
22
+ }
23
+ }
24
+ }
25
+ function defaultErrorHandler(err, _req, res, _next) {
26
+ console.error("Error handling MCP request:", err);
27
+ if (!res.headersSent) {
28
+ res.status(500).json({
29
+ jsonrpc: "2.0",
30
+ error: { code: -32603, message: "Internal server error" },
31
+ id: null,
32
+ });
33
+ }
34
+ }
35
+ export async function createApp({ mcpServer, httpServer, customMiddleware = [], errorMiddleware = [], }) {
7
36
  const app = express();
8
37
  app.use(express.json());
9
38
  const env = process.env.NODE_ENV || "development";
39
+ applyMiddlewares(app, customMiddleware);
10
40
  if (env !== "production") {
11
41
  const { devtoolsStaticServer } = await import("@skybridge/devtools");
12
42
  app.use(await devtoolsStaticServer());
13
- const { widgetsDevServer } = await import("./widgetsDevServer.js");
14
- app.use(await widgetsDevServer());
43
+ const { viewsDevServer } = await import("./viewsDevServer.js");
44
+ app.use(await viewsDevServer(httpServer));
45
+ const controlPort = parseControlPort(process.env.__TUNNEL_CONTROL_PORT);
46
+ if (controlPort !== null) {
47
+ const { createTunnelProxyRouter } = await import("./tunnel-proxy-router.js");
48
+ app.use(createTunnelProxyRouter(controlPort));
49
+ }
50
+ else if (process.env.__TUNNEL_CONTROL_PORT !== undefined) {
51
+ console.warn(`Ignoring invalid __TUNNEL_CONTROL_PORT=${process.env.__TUNNEL_CONTROL_PORT}`);
52
+ }
15
53
  }
16
54
  if (env === "production") {
17
- const __filename = fileURLToPath(import.meta.url);
18
- const __dirname = path.dirname(__filename);
55
+ const assetsPath = path.join(process.cwd(), "dist", "assets");
19
56
  app.use("/assets", cors());
20
- app.use("/assets", express.static(path.join(__dirname, "assets")));
57
+ app.use("/assets", express.static(assetsPath));
21
58
  }
22
- app.use("/mcp", mcpMiddleware(server));
59
+ app.use("/mcp", mcpMiddleware(mcpServer));
60
+ applyMiddlewares(app, errorMiddleware);
61
+ app.use("/mcp", defaultErrorHandler);
23
62
  return app;
24
63
  }
25
64
  const mcpMiddleware = (server) => {
26
- return async (req, res, _next) => {
65
+ return async (req, res, next) => {
27
66
  if (req.method !== "POST") {
28
67
  res.writeHead(405).end(JSON.stringify({
29
68
  jsonrpc: "2.0",
@@ -42,21 +81,14 @@ const mcpMiddleware = (server) => {
42
81
  res.on("close", () => {
43
82
  transport.close();
44
83
  });
45
- await server.connect(transport);
84
+ await server.connectStatelessTransport(transport);
85
+ // Express strips the mount path from req.url (e.g. "/mcp" becomes "/").
86
+ // Restore it so the SDK builds the correct requestInfo.url.
87
+ req.url = req.originalUrl;
46
88
  await transport.handleRequest(req, res, req.body);
47
89
  }
48
90
  catch (error) {
49
- console.error("Error handling MCP request:", error);
50
- if (!res.headersSent) {
51
- res.status(500).json({
52
- jsonrpc: "2.0",
53
- error: {
54
- code: -32603,
55
- message: "Internal server error",
56
- },
57
- id: null,
58
- });
59
- }
91
+ next(error);
60
92
  }
61
93
  };
62
94
  };
@@ -1 +1 @@
1
- {"version":3,"file":"express.js","sourceRoot":"","sources":["../../src/server/express.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"express.js","sourceRoot":"","sources":["../../src/server/express.ts"],"names":[],"mappings":"AACA,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,EAAE,6BAA6B,EAAE,MAAM,oDAAoD,CAAC;AACnG,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,OAAO,MAAM,SAAS,CAAC;AAG9B,SAAS,gBAAgB,CAAC,GAAuB;IAC/C,IAAI,GAAG,KAAK,SAAS,EAAE,CAAC;QACtB,OAAO,IAAI,CAAC;IACd,CAAC;IACD,MAAM,CAAC,GAAG,MAAM,CAAC,QAAQ,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;IACnC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,EAAE,CAAC;QAChD,OAAO,IAAI,CAAC;IACd,CAAC;IACD,OAAO,CAAC,CAAC;AACX,CAAC;AAED,SAAS,gBAAgB,CACvB,GAAoB,EACpB,WAGE;IAEF,KAAK,MAAM,UAAU,IAAI,WAAW,EAAE,CAAC;QACrC,IAAI,UAAU,CAAC,IAAI,EAAE,CAAC;YACpB,GAAG,CAAC,GAAG,CAAC,UAAU,CAAC,IAAI,EAAE,GAAG,UAAU,CAAC,QAAQ,CAAC,CAAC;QACnD,CAAC;aAAM,CAAC;YACN,GAAG,CAAC,GAAG,CAAC,GAAG,UAAU,CAAC,QAAQ,CAAC,CAAC;QAClC,CAAC;IACH,CAAC;AACH,CAAC;AAED,SAAS,mBAAmB,CAC1B,GAAY,EACZ,IAAqB,EACrB,GAAqB,EACrB,KAA2B;IAE3B,OAAO,CAAC,KAAK,CAAC,6BAA6B,EAAE,GAAG,CAAC,CAAC;IAClD,IAAI,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC;QACrB,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC;YACnB,OAAO,EAAE,KAAK;YACd,KAAK,EAAE,EAAE,IAAI,EAAE,CAAC,KAAK,EAAE,OAAO,EAAE,uBAAuB,EAAE;YACzD,EAAE,EAAE,IAAI;SACT,CAAC,CAAC;IACL,CAAC;AACH,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,SAAS,CAAC,EAC9B,SAAS,EACT,UAAU,EACV,gBAAgB,GAAG,EAAE,EACrB,eAAe,GAAG,EAAE,GASrB;IACC,MAAM,GAAG,GAAG,OAAO,EAAE,CAAC;IACtB,GAAG,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;IACxB,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,QAAQ,IAAI,aAAa,CAAC;IAElD,gBAAgB,CAAC,GAAG,EAAE,gBAAgB,CAAC,CAAC;IAExC,IAAI,GAAG,KAAK,YAAY,EAAE,CAAC;QACzB,MAAM,EAAE,oBAAoB,EAAE,GAAG,MAAM,MAAM,CAAC,qBAAqB,CAAC,CAAC;QACrE,GAAG,CAAC,GAAG,CAAC,MAAM,oBAAoB,EAAE,CAAC,CAAC;QACtC,MAAM,EAAE,cAAc,EAAE,GAAG,MAAM,MAAM,CAAC,qBAAqB,CAAC,CAAC;QAC/D,GAAG,CAAC,GAAG,CAAC,MAAM,cAAc,CAAC,UAAU,CAAC,CAAC,CAAC;QAE1C,MAAM,WAAW,GAAG,gBAAgB,CAAC,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,CAAC;QACxE,IAAI,WAAW,KAAK,IAAI,EAAE,CAAC;YACzB,MAAM,EAAE,uBAAuB,EAAE,GAAG,MAAM,MAAM,CAC9C,0BAA0B,CAC3B,CAAC;YACF,GAAG,CAAC,GAAG,CAAC,uBAAuB,CAAC,WAAW,CAAC,CAAC,CAAC;QAChD,CAAC;aAAM,IAAI,OAAO,CAAC,GAAG,CAAC,qBAAqB,KAAK,SAAS,EAAE,CAAC;YAC3D,OAAO,CAAC,IAAI,CACV,0CAA0C,OAAO,CAAC,GAAG,CAAC,qBAAqB,EAAE,CAC9E,CAAC;QACJ,CAAC;IACH,CAAC;IAED,IAAI,GAAG,KAAK,YAAY,EAAE,CAAC;QACzB,MAAM,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAC;QAE9D,GAAG,CAAC,GAAG,CAAC,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC3B,GAAG,CAAC,GAAG,CAAC,SAAS,EAAE,OAAO,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC;IACjD,CAAC;IAED,GAAG,CAAC,GAAG,CAAC,MAAM,EAAE,aAAa,CAAC,SAAS,CAAC,CAAC,CAAC;IAE1C,gBAAgB,CAAC,GAAG,EAAE,eAAe,CAAC,CAAC;IAEvC,GAAG,CAAC,GAAG,CAAC,MAAM,EAAE,mBAAmB,CAAC,CAAC;IAErC,OAAO,GAAG,CAAC;AACb,CAAC;AAED,MAAM,aAAa,GAAG,CAAC,MAAiB,EAA0B,EAAE;IAClE,OAAO,KAAK,EACV,GAAoB,EACpB,GAAqB,EACrB,IAA0B,EAC1B,EAAE;QACF,IAAI,GAAG,CAAC,MAAM,KAAK,MAAM,EAAE,CAAC;YAC1B,GAAG,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,GAAG,CACpB,IAAI,CAAC,SAAS,CAAC;gBACb,OAAO,EAAE,KAAK;gBACd,KAAK,EAAE;oBACL,IAAI,EAAE,CAAC,KAAK;oBACZ,OAAO,EAAE,qBAAqB;iBAC/B;gBACD,EAAE,EAAE,IAAI;aACT,CAAC,CACH,CAAC;YACF,OAAO;QACT,CAAC;QAED,IAAI,CAAC;YACH,MAAM,SAAS,GAAG,IAAI,6BAA6B,CAAC;gBAClD,kBAAkB,EAAE,SAAS;aAC9B,CAAC,CAAC;YAEH,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE;gBACnB,SAAS,CAAC,KAAK,EAAE,CAAC;YACpB,CAAC,CAAC,CAAC;YAEH,MAAM,MAAM,CAAC,yBAAyB,CAAC,SAAS,CAAC,CAAC;YAClD,wEAAwE;YACxE,4DAA4D;YAC5D,GAAG,CAAC,GAAG,GAAG,GAAG,CAAC,WAAW,CAAC;YAC1B,MAAM,SAAS,CAAC,aAAa,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC,IAAI,CAAC,CAAC;QACpD,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAI,CAAC,KAAK,CAAC,CAAC;QACd,CAAC;IACH,CAAC,CAAC;AACJ,CAAC,CAAC"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,348 @@
1
+ import http from "node:http";
2
+ import { afterEach, describe, expect, it, vi } from "vitest";
3
+ import { McpServer } from "./server.js";
4
+ vi.mock("@skybridge/devtools", () => ({
5
+ devtoolsStaticServer: () => ((_req, _res, next) => next()),
6
+ }));
7
+ vi.mock("./viewsDevServer.js", () => ({
8
+ viewsDevServer: (_httpServer) => ((_req, _res, next) => next()),
9
+ }));
10
+ const fakeServer = {};
11
+ async function listen(app) {
12
+ const server = http.createServer(app);
13
+ await new Promise((resolve) => server.listen(0, resolve));
14
+ const port = server.address().port;
15
+ return { port, server };
16
+ }
17
+ let openServer;
18
+ afterEach(() => openServer?.close());
19
+ async function postMcp(port) {
20
+ return fetch(`http://localhost:${port}/mcp`, {
21
+ method: "POST",
22
+ headers: { "Content-Type": "application/json" },
23
+ body: JSON.stringify({ jsonrpc: "2.0", method: "initialize", id: 1 }),
24
+ });
25
+ }
26
+ async function postApi(port) {
27
+ return fetch(`http://localhost:${port}/api/test`, { method: "POST" });
28
+ }
29
+ describe("createApp", () => {
30
+ it("runs global custom middleware before the /mcp handler", async () => {
31
+ const { createApp } = await import("./express.js");
32
+ const calls = [];
33
+ const mw = (_req, _res, next) => {
34
+ calls.push("custom");
35
+ next();
36
+ };
37
+ const httpServer = http.createServer();
38
+ const app = await createApp({
39
+ mcpServer: fakeServer,
40
+ httpServer,
41
+ customMiddleware: [{ handlers: [mw] }],
42
+ });
43
+ const { port, server } = await listen(app);
44
+ openServer = server;
45
+ await postMcp(port);
46
+ expect(calls).toEqual(["custom"]);
47
+ });
48
+ it("runs path-scoped middleware on /mcp", async () => {
49
+ const { createApp } = await import("./express.js");
50
+ const calls = [];
51
+ const mw = (_req, _res, next) => {
52
+ calls.push("auth");
53
+ next();
54
+ };
55
+ const httpServer = http.createServer();
56
+ const app = await createApp({
57
+ mcpServer: fakeServer,
58
+ httpServer,
59
+ customMiddleware: [{ path: "/mcp", handlers: [mw] }],
60
+ });
61
+ const { port, server } = await listen(app);
62
+ openServer = server;
63
+ await postMcp(port);
64
+ expect(calls).toEqual(["auth"]);
65
+ });
66
+ it("allows middleware to short-circuit with 401", async () => {
67
+ const { createApp } = await import("./express.js");
68
+ const calls = [];
69
+ const reject = (_req, res) => {
70
+ calls.push("reject");
71
+ res.status(401).json({ error: "Unauthorized" });
72
+ };
73
+ const httpServer = http.createServer();
74
+ const app = await createApp({
75
+ mcpServer: fakeServer,
76
+ httpServer,
77
+ customMiddleware: [{ path: "/mcp", handlers: [reject] }],
78
+ });
79
+ const { port, server } = await listen(app);
80
+ openServer = server;
81
+ const res = await postMcp(port);
82
+ expect(calls).toEqual(["reject"]);
83
+ expect(res.status).toBe(401);
84
+ expect(await res.json()).toEqual({ error: "Unauthorized" });
85
+ });
86
+ it("runs multiple global middleware in registration order", async () => {
87
+ const { createApp } = await import("./express.js");
88
+ const calls = [];
89
+ const mwA = (_req, _res, next) => {
90
+ calls.push("A");
91
+ next();
92
+ };
93
+ const mwB = (_req, _res, next) => {
94
+ calls.push("B");
95
+ next();
96
+ };
97
+ const httpServer = http.createServer();
98
+ const app = await createApp({
99
+ mcpServer: fakeServer,
100
+ httpServer,
101
+ customMiddleware: [{ handlers: [mwA] }, { handlers: [mwB] }],
102
+ });
103
+ const { port, server } = await listen(app);
104
+ openServer = server;
105
+ await postMcp(port);
106
+ expect(calls).toEqual(["A", "B"]);
107
+ });
108
+ it("path-scoped middleware does not run on non-matching paths", async () => {
109
+ const { createApp } = await import("./express.js");
110
+ const calls = [];
111
+ const apiMw = (_req, _res, next) => {
112
+ calls.push("api");
113
+ next();
114
+ };
115
+ const httpServer = http.createServer();
116
+ const app = await createApp({
117
+ mcpServer: fakeServer,
118
+ httpServer,
119
+ customMiddleware: [{ path: "/api", handlers: [apiMw] }],
120
+ });
121
+ const { port, server } = await listen(app);
122
+ openServer = server;
123
+ // Hit /mcp — the /api middleware should NOT fire
124
+ await postMcp(port);
125
+ expect(calls).toEqual([]);
126
+ });
127
+ it("supports Express Router via custom middleware", async () => {
128
+ const { createApp } = await import("./express.js");
129
+ const { Router } = await import("express");
130
+ const router = Router();
131
+ router.get("/health", (_req, res) => {
132
+ res.json({ status: "ok" });
133
+ });
134
+ const httpServer = http.createServer();
135
+ const app = await createApp({
136
+ mcpServer: fakeServer,
137
+ httpServer,
138
+ customMiddleware: [{ handlers: [router] }],
139
+ });
140
+ const { port, server } = await listen(app);
141
+ openServer = server;
142
+ const res = await fetch(`http://localhost:${port}/health`);
143
+ expect(res.status).toBe(200);
144
+ expect(await res.json()).toEqual({ status: "ok" });
145
+ });
146
+ it("supports path-prefixed Router", async () => {
147
+ const { createApp } = await import("./express.js");
148
+ const { Router } = await import("express");
149
+ const router = Router();
150
+ router.get("/data", (_req, res) => {
151
+ res.json({ value: 42 });
152
+ });
153
+ const httpServer = http.createServer();
154
+ const app = await createApp({
155
+ mcpServer: fakeServer,
156
+ httpServer,
157
+ customMiddleware: [
158
+ { path: "/api", handlers: [router] },
159
+ ],
160
+ });
161
+ const { port, server } = await listen(app);
162
+ openServer = server;
163
+ const res = await fetch(`http://localhost:${port}/api/data`);
164
+ expect(res.status).toBe(200);
165
+ expect(await res.json()).toEqual({ value: 42 });
166
+ });
167
+ it("server survives middleware errors without crashing", async () => {
168
+ const { createApp } = await import("./express.js");
169
+ const throwing = () => {
170
+ throw new Error("boom");
171
+ };
172
+ const httpServer = http.createServer();
173
+ const app = await createApp({
174
+ mcpServer: fakeServer,
175
+ httpServer,
176
+ customMiddleware: [{ path: "/explode", handlers: [throwing] }],
177
+ });
178
+ const { port, server } = await listen(app);
179
+ openServer = server;
180
+ const res = await fetch(`http://localhost:${port}/explode`);
181
+ expect(res.status).toBe(500);
182
+ // Server process did not crash — it still accepts connections
183
+ const followUp = await fetch(`http://localhost:${port}/explode`);
184
+ expect(followUp.status).toBe(500);
185
+ });
186
+ it("returns 500 JSON-RPC error when the MCP handler throws and no error middleware is registered", async () => {
187
+ const { createApp } = await import("./express.js");
188
+ const consoleSpy = vi.spyOn(console, "error").mockImplementation(() => { });
189
+ const httpServer = http.createServer();
190
+ const app = await createApp({ mcpServer: fakeServer, httpServer });
191
+ const { port, server } = await listen(app);
192
+ openServer = server;
193
+ const res = await postMcp(port);
194
+ expect(res.status).toBe(500);
195
+ expect(await res.json()).toEqual({
196
+ jsonrpc: "2.0",
197
+ error: { code: -32603, message: "Internal server error" },
198
+ id: null,
199
+ });
200
+ expect(consoleSpy).toHaveBeenCalledWith("Error handling MCP request:", expect.any(Error));
201
+ consoleSpy.mockRestore();
202
+ });
203
+ it("invokes a custom error handler when the MCP handler throws", async () => {
204
+ const { createApp } = await import("./express.js");
205
+ const calls = [];
206
+ const errorHandler = (_err, _req, res, _next) => {
207
+ calls.push("error-handler");
208
+ res.status(503).json({ custom: true });
209
+ };
210
+ const httpServer = http.createServer();
211
+ const app = await createApp({
212
+ mcpServer: fakeServer,
213
+ httpServer,
214
+ errorMiddleware: [{ handlers: [errorHandler] }],
215
+ });
216
+ const { port, server } = await listen(app);
217
+ openServer = server;
218
+ const res = await postMcp(port);
219
+ expect(calls).toEqual(["error-handler"]);
220
+ expect(res.status).toBe(503);
221
+ expect(await res.json()).toEqual({ custom: true });
222
+ });
223
+ it("invokes a path-scoped error handler only for matching routes", async () => {
224
+ const { createApp } = await import("./express.js");
225
+ const calls = [];
226
+ const mcpErrorHandler = (_err, _req, res, _next) => {
227
+ calls.push("mcp-error-handler");
228
+ res.status(503).json({ from: "mcp-error-handler" });
229
+ };
230
+ const throwingApiRoute = (_req, _res, next) => {
231
+ next(new Error("api error"));
232
+ };
233
+ const httpServer = http.createServer();
234
+ const app = await createApp({
235
+ mcpServer: fakeServer,
236
+ httpServer,
237
+ customMiddleware: [{ path: "/api/test", handlers: [throwingApiRoute] }],
238
+ errorMiddleware: [{ path: "/mcp", handlers: [mcpErrorHandler] }],
239
+ });
240
+ const { port, server } = await listen(app);
241
+ openServer = server;
242
+ const mcpRes = await postMcp(port);
243
+ expect(calls).toEqual(["mcp-error-handler"]);
244
+ expect(mcpRes.status).toBe(503);
245
+ expect(await mcpRes.json()).toEqual({ from: "mcp-error-handler" });
246
+ const consoleSpy = vi.spyOn(console, "error").mockImplementation(() => { });
247
+ const apiRes = await postApi(port);
248
+ expect(calls).toEqual(["mcp-error-handler"]);
249
+ expect(apiRes.status).toBe(500);
250
+ consoleSpy.mockRestore();
251
+ });
252
+ it("handles concurrent /mcp requests without 'Already connected to a transport'", async () => {
253
+ const { createApp } = await import("./express.js");
254
+ const mcpServer = new McpServer({
255
+ name: "concurrent-test",
256
+ version: "0.0.0",
257
+ });
258
+ // Slow tool: keeps the underlying transport bound long enough to overlap
259
+ // with concurrent requests, exposing the shared-McpServer race.
260
+ mcpServer.registerTool({ name: "slow", description: "slow" }, async () => {
261
+ await new Promise((r) => setTimeout(r, 50));
262
+ return { content: [{ type: "text", text: "done" }] };
263
+ });
264
+ const consoleSpy = vi.spyOn(console, "error").mockImplementation(() => { });
265
+ const httpServer = http.createServer();
266
+ const app = await createApp({ mcpServer, httpServer });
267
+ const { port, server } = await listen(app);
268
+ openServer = server;
269
+ const callBody = (id) => JSON.stringify({
270
+ jsonrpc: "2.0",
271
+ method: "tools/call",
272
+ id,
273
+ params: { name: "slow", arguments: {} },
274
+ });
275
+ const N = 10;
276
+ const responses = await Promise.all(Array.from({ length: N }, (_, i) => fetch(`http://localhost:${port}/mcp`, {
277
+ method: "POST",
278
+ headers: {
279
+ "Content-Type": "application/json",
280
+ Accept: "application/json, text/event-stream",
281
+ },
282
+ body: callBody(i + 1),
283
+ })));
284
+ expect(responses.map((r) => r.status)).toEqual(Array(N).fill(200));
285
+ expect(consoleSpy).not.toHaveBeenCalledWith("Error handling MCP request:", expect.any(Error));
286
+ consoleSpy.mockRestore();
287
+ });
288
+ });
289
+ describe("createApp tunnel routes", () => {
290
+ it("proxies POST /__skybridge/tunnel to the cli control server in dev mode", async () => {
291
+ // Stand up a fake control listener that returns a known JSON body.
292
+ const control = http.createServer((_req, res) => {
293
+ res.writeHead(200, { "Content-Type": "application/json" });
294
+ res.end('{"status":"idle"}');
295
+ });
296
+ await new Promise((resolve) => control.listen(0, "127.0.0.1", resolve));
297
+ const controlAddr = control.address();
298
+ if (typeof controlAddr === "string" || controlAddr === null) {
299
+ control.close();
300
+ throw new Error("control server has no address");
301
+ }
302
+ const controlPort = controlAddr.port;
303
+ const prev = process.env.__TUNNEL_CONTROL_PORT;
304
+ process.env.__TUNNEL_CONTROL_PORT = String(controlPort);
305
+ try {
306
+ const { createApp } = await import("./express.js");
307
+ const httpServer = http.createServer();
308
+ const app = await createApp({ mcpServer: fakeServer, httpServer });
309
+ const { port, server } = await listen(app);
310
+ openServer = server;
311
+ const res = await fetch(`http://localhost:${port}/__skybridge/tunnel`, {
312
+ method: "POST",
313
+ });
314
+ expect(res.status).toBe(200);
315
+ expect(await res.json()).toEqual({ status: "idle" });
316
+ }
317
+ finally {
318
+ if (prev === undefined) {
319
+ delete process.env.__TUNNEL_CONTROL_PORT;
320
+ }
321
+ else {
322
+ process.env.__TUNNEL_CONTROL_PORT = prev;
323
+ }
324
+ await new Promise((resolve) => control.close(() => resolve()));
325
+ }
326
+ });
327
+ it("does not expose /__skybridge/tunnel in production mode", async () => {
328
+ const prevEnv = process.env.NODE_ENV;
329
+ process.env.NODE_ENV = "production";
330
+ try {
331
+ vi.resetModules();
332
+ const { createApp } = await import("./express.js");
333
+ const httpServer = http.createServer();
334
+ const app = await createApp({ mcpServer: fakeServer, httpServer });
335
+ const { port, server } = await listen(app);
336
+ openServer = server;
337
+ const res = await fetch(`http://localhost:${port}/__skybridge/tunnel`, {
338
+ method: "POST",
339
+ });
340
+ expect(res.status).toBe(404);
341
+ }
342
+ finally {
343
+ process.env.NODE_ENV = prevEnv;
344
+ vi.resetModules();
345
+ }
346
+ });
347
+ });
348
+ //# sourceMappingURL=express.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"express.test.js","sourceRoot":"","sources":["../../src/server/express.test.ts"],"names":[],"mappings":"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"}