@canmi/seam-adapter-bun 0.5.31 → 0.5.38

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/README.md CHANGED
@@ -19,4 +19,4 @@ Exports `serveBun()` which starts a Bun HTTP server with seam routing, optional
19
19
 
20
20
  - Peer dependency: `@canmi/seam-server`
21
21
  - Tests use `bun:test`, not vitest
22
- - Options: `staticDir` for static files, `fallback` for unmatched routes
22
+ - Options: `staticDir` for static files, `publicDir` for serving `public/` directory files (from build output), `fallback` for unmatched routes
package/dist/index.d.ts CHANGED
@@ -1,20 +1,16 @@
1
- import { ChannelWsOptions, ChannelWsSession, DefinitionMap, HttpHandler, Router, RpcHashMap, SseOptions } from "@canmi/seam-server";
1
+ import { ChannelWsOptions, DefinitionMap, HttpHandler, Router, RpcHashMap, SseOptions } from "@canmi/seam-server";
2
2
 
3
3
  //#region src/index.d.ts
4
4
  interface ServeBunOptions {
5
5
  port?: number;
6
6
  staticDir?: string;
7
+ publicDir?: string;
7
8
  fallback?: HttpHandler;
8
9
  rpcHashMap?: RpcHashMap;
9
10
  wsOptions?: ChannelWsOptions;
10
11
  sseOptions?: SseOptions;
11
12
  }
12
- interface WsData {
13
- channelName: string;
14
- channelInput: unknown;
15
- session?: ChannelWsSession;
16
- }
17
- declare function serveBun<T extends DefinitionMap>(router: Router<T>, opts?: ServeBunOptions): Bun.Server<WsData>;
13
+ declare function serveBun<T extends DefinitionMap>(router: Router<T>, opts?: ServeBunOptions): any;
18
14
  //#endregion
19
15
  export { ServeBunOptions, serveBun };
20
16
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","names":[],"sources":["../src/index.ts"],"mappings":";;;UAaiB,eAAA;EAChB,IAAA;EACA,SAAA;EACA,QAAA,GAAW,WAAA;EACX,UAAA,GAAa,UAAA;EACb,SAAA,GAAY,gBAAA;EACZ,UAAA,GAAa,UAAA;AAAA;AAAA,UAMJ,MAAA;EACT,WAAA;EACA,YAAA;EACA,OAAA,GAAU,gBAAA;AAAA;AAAA,iBAGK,QAAA,WAAmB,aAAA,CAAA,CAAe,MAAA,EAAQ,MAAA,CAAO,CAAA,GAAI,IAAA,GAAO,eAAA,GAAe,GAAA,CAAA,MAAA,CAAA,MAAA"}
1
+ {"version":3,"file":"index.d.ts","names":[],"sources":["../src/index.ts"],"mappings":";;;UAkBiB,eAAA;EAChB,IAAA;EACA,SAAA;EACA,SAAA;EACA,QAAA,GAAW,WAAA;EACX,UAAA,GAAa,UAAA;EACb,SAAA,GAAY,gBAAA;EACZ,UAAA,GAAa,UAAA;AAAA;AAAA,iBAYE,QAAA,WAAmB,aAAA,CAAA,CAAe,MAAA,EAAQ,MAAA,CAAO,CAAA,GAAI,IAAA,GAAO,eAAA"}
package/dist/index.js CHANGED
@@ -1,11 +1,11 @@
1
- import { createHttpHandler, startChannelWs, toWebResponse } from "@canmi/seam-server";
2
-
1
+ import { buildMultipartFields, createHttpHandler, startChannelWs, toWebResponse } from "@canmi/seam-server";
3
2
  //#region src/index.ts
4
3
  const PROCEDURE_PREFIX = "/_seam/procedure/";
5
4
  const EVENTS_SUFFIX = ".events";
6
5
  function serveBun(router, opts) {
7
6
  const handler = createHttpHandler(router, {
8
7
  staticDir: opts?.staticDir,
8
+ publicDir: opts?.publicDir,
9
9
  fallback: opts?.fallback,
10
10
  rpcHashMap: opts?.rpcHashMap,
11
11
  sseOptions: opts?.sseOptions
@@ -32,18 +32,13 @@ function serveBun(router, opts) {
32
32
  return new Response("WebSocket upgrade failed", { status: 500 });
33
33
  }
34
34
  }
35
- const isMultipart = (req.headers.get("content-type") ?? "").startsWith("multipart/form-data");
36
- let formDataCache;
37
- const getFormData = async () => formDataCache ??= await req.formData();
35
+ const { body, file } = buildMultipartFields(req);
38
36
  return toWebResponse(await handler({
39
37
  method: req.method,
40
38
  url: req.url,
41
- body: isMultipart ? async () => JSON.parse((await getFormData()).get("metadata")) : () => req.json(),
39
+ body,
42
40
  header: (name) => req.headers.get(name),
43
- file: isMultipart ? async () => {
44
- const f = (await getFormData()).get("file");
45
- return f ? { stream: () => f.stream() } : null;
46
- } : void 0
41
+ file
47
42
  }));
48
43
  },
49
44
  websocket: {
@@ -68,7 +63,7 @@ function serveBun(router, opts) {
68
63
  }
69
64
  });
70
65
  }
71
-
72
66
  //#endregion
73
67
  export { serveBun };
68
+
74
69
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","names":[],"sources":["../src/index.ts"],"sourcesContent":["/* src/server/adapter/bun/src/index.ts */\n\nimport { createHttpHandler, toWebResponse, startChannelWs } from '@canmi/seam-server'\nimport type {\n\tDefinitionMap,\n\tRouter,\n\tHttpHandler,\n\tRpcHashMap,\n\tChannelWsSession,\n\tChannelWsOptions,\n\tSseOptions,\n} from '@canmi/seam-server'\n\nexport interface ServeBunOptions {\n\tport?: number\n\tstaticDir?: string\n\tfallback?: HttpHandler\n\trpcHashMap?: RpcHashMap\n\twsOptions?: ChannelWsOptions\n\tsseOptions?: SseOptions\n}\n\nconst PROCEDURE_PREFIX = '/_seam/procedure/'\nconst EVENTS_SUFFIX = '.events'\n\ninterface WsData {\n\tchannelName: string\n\tchannelInput: unknown\n\tsession?: ChannelWsSession\n}\n\nexport function serveBun<T extends DefinitionMap>(router: Router<T>, opts?: ServeBunOptions) {\n\tconst handler = createHttpHandler(router, {\n\t\tstaticDir: opts?.staticDir,\n\t\tfallback: opts?.fallback,\n\t\trpcHashMap: opts?.rpcHashMap,\n\t\tsseOptions: opts?.sseOptions,\n\t})\n\n\treturn Bun.serve<WsData>({\n\t\tport: opts?.port ?? 3000,\n\n\t\tasync fetch(req, server) {\n\t\t\t// WebSocket upgrade for channel paths\n\t\t\tif (req.method === 'GET' && req.headers.get('upgrade') === 'websocket') {\n\t\t\t\tconst url = new URL(req.url)\n\t\t\t\tconst { pathname } = url\n\t\t\t\tif (pathname.startsWith(PROCEDURE_PREFIX) && pathname.endsWith(EVENTS_SUFFIX)) {\n\t\t\t\t\tconst channelName = pathname.slice(PROCEDURE_PREFIX.length, -EVENTS_SUFFIX.length)\n\t\t\t\t\tconst rawInput = url.searchParams.get('input')\n\t\t\t\t\tlet channelInput: unknown\n\t\t\t\t\ttry {\n\t\t\t\t\t\tchannelInput = rawInput ? JSON.parse(rawInput) : {}\n\t\t\t\t\t} catch {\n\t\t\t\t\t\treturn new Response('Invalid input query parameter', { status: 400 })\n\t\t\t\t\t}\n\t\t\t\t\tconst upgraded = server.upgrade(req, {\n\t\t\t\t\t\tdata: { channelName, channelInput },\n\t\t\t\t\t})\n\t\t\t\t\tif (upgraded) return undefined\n\t\t\t\t\treturn new Response('WebSocket upgrade failed', { status: 500 })\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconst contentType = req.headers.get('content-type') ?? ''\n\t\t\tconst isMultipart = contentType.startsWith('multipart/form-data')\n\n\t\t\tlet formDataCache: FormData | undefined\n\t\t\tconst getFormData = async () => (formDataCache ??= await req.formData())\n\n\t\t\tconst result = await handler({\n\t\t\t\tmethod: req.method,\n\t\t\t\turl: req.url,\n\t\t\t\tbody: isMultipart\n\t\t\t\t\t? async () => JSON.parse((await getFormData()).get('metadata') as string) as unknown\n\t\t\t\t\t: () => req.json(),\n\t\t\t\theader: (name) => req.headers.get(name),\n\t\t\t\tfile: isMultipart\n\t\t\t\t\t? async () => {\n\t\t\t\t\t\t\tconst f = (await getFormData()).get('file') as File | null\n\t\t\t\t\t\t\treturn f ? { stream: () => f.stream() } : null\n\t\t\t\t\t\t}\n\t\t\t\t\t: undefined,\n\t\t\t})\n\t\t\treturn toWebResponse(result)\n\t\t},\n\n\t\twebsocket: {\n\t\t\topen(ws) {\n\t\t\t\tconst { channelName, channelInput } = ws.data\n\t\t\t\tws.data.session = startChannelWs(\n\t\t\t\t\trouter,\n\t\t\t\t\tchannelName,\n\t\t\t\t\tchannelInput,\n\t\t\t\t\t{\n\t\t\t\t\t\tsend: (data) => ws.send(data),\n\t\t\t\t\t\tping: () => ws.ping(),\n\t\t\t\t\t\tclose: () => ws.close(),\n\t\t\t\t\t},\n\t\t\t\t\topts?.wsOptions,\n\t\t\t\t)\n\t\t\t},\n\t\t\tmessage(ws, message) {\n\t\t\t\tconst text = typeof message === 'string' ? message : new TextDecoder().decode(message)\n\t\t\t\tws.data.session?.onMessage(text)\n\t\t\t},\n\t\t\tpong(ws) {\n\t\t\t\tws.data.session?.onPong()\n\t\t\t},\n\t\t\tclose(ws) {\n\t\t\t\tws.data.session?.close()\n\t\t\t},\n\t\t},\n\t})\n}\n"],"mappings":";;;AAsBA,MAAM,mBAAmB;AACzB,MAAM,gBAAgB;AAQtB,SAAgB,SAAkC,QAAmB,MAAwB;CAC5F,MAAM,UAAU,kBAAkB,QAAQ;EACzC,WAAW,MAAM;EACjB,UAAU,MAAM;EAChB,YAAY,MAAM;EAClB,YAAY,MAAM;EAClB,CAAC;AAEF,QAAO,IAAI,MAAc;EACxB,MAAM,MAAM,QAAQ;EAEpB,MAAM,MAAM,KAAK,QAAQ;AAExB,OAAI,IAAI,WAAW,SAAS,IAAI,QAAQ,IAAI,UAAU,KAAK,aAAa;IACvE,MAAM,MAAM,IAAI,IAAI,IAAI,IAAI;IAC5B,MAAM,EAAE,aAAa;AACrB,QAAI,SAAS,WAAW,iBAAiB,IAAI,SAAS,SAAS,cAAc,EAAE;KAC9E,MAAM,cAAc,SAAS,MAAM,IAAyB,GAAsB;KAClF,MAAM,WAAW,IAAI,aAAa,IAAI,QAAQ;KAC9C,IAAI;AACJ,SAAI;AACH,qBAAe,WAAW,KAAK,MAAM,SAAS,GAAG,EAAE;aAC5C;AACP,aAAO,IAAI,SAAS,iCAAiC,EAAE,QAAQ,KAAK,CAAC;;AAKtE,SAHiB,OAAO,QAAQ,KAAK,EACpC,MAAM;MAAE;MAAa;MAAc,EACnC,CAAC,CACY,QAAO;AACrB,YAAO,IAAI,SAAS,4BAA4B,EAAE,QAAQ,KAAK,CAAC;;;GAKlE,MAAM,eADc,IAAI,QAAQ,IAAI,eAAe,IAAI,IACvB,WAAW,sBAAsB;GAEjE,IAAI;GACJ,MAAM,cAAc,YAAa,kBAAkB,MAAM,IAAI,UAAU;AAgBvE,UAAO,cAdQ,MAAM,QAAQ;IAC5B,QAAQ,IAAI;IACZ,KAAK,IAAI;IACT,MAAM,cACH,YAAY,KAAK,OAAO,MAAM,aAAa,EAAE,IAAI,WAAW,CAAW,SACjE,IAAI,MAAM;IACnB,SAAS,SAAS,IAAI,QAAQ,IAAI,KAAK;IACvC,MAAM,cACH,YAAY;KACZ,MAAM,KAAK,MAAM,aAAa,EAAE,IAAI,OAAO;AAC3C,YAAO,IAAI,EAAE,cAAc,EAAE,QAAQ,EAAE,GAAG;QAE1C;IACH,CAAC,CAC0B;;EAG7B,WAAW;GACV,KAAK,IAAI;IACR,MAAM,EAAE,aAAa,iBAAiB,GAAG;AACzC,OAAG,KAAK,UAAU,eACjB,QACA,aACA,cACA;KACC,OAAO,SAAS,GAAG,KAAK,KAAK;KAC7B,YAAY,GAAG,MAAM;KACrB,aAAa,GAAG,OAAO;KACvB,EACD,MAAM,UACN;;GAEF,QAAQ,IAAI,SAAS;IACpB,MAAM,OAAO,OAAO,YAAY,WAAW,UAAU,IAAI,aAAa,CAAC,OAAO,QAAQ;AACtF,OAAG,KAAK,SAAS,UAAU,KAAK;;GAEjC,KAAK,IAAI;AACR,OAAG,KAAK,SAAS,QAAQ;;GAE1B,MAAM,IAAI;AACT,OAAG,KAAK,SAAS,OAAO;;GAEzB;EACD,CAAC"}
1
+ {"version":3,"file":"index.js","names":[],"sources":["../src/index.ts"],"sourcesContent":["/* src/server/adapter/bun/src/index.ts */\n\nimport {\n\tcreateHttpHandler,\n\ttoWebResponse,\n\tstartChannelWs,\n\tbuildMultipartFields,\n} from '@canmi/seam-server'\nimport type {\n\tDefinitionMap,\n\tRouter,\n\tHttpHandler,\n\tRpcHashMap,\n\tChannelWsSession,\n\tChannelWsOptions,\n\tSseOptions,\n} from '@canmi/seam-server'\n\nexport interface ServeBunOptions {\n\tport?: number\n\tstaticDir?: string\n\tpublicDir?: string\n\tfallback?: HttpHandler\n\trpcHashMap?: RpcHashMap\n\twsOptions?: ChannelWsOptions\n\tsseOptions?: SseOptions\n}\n\nconst PROCEDURE_PREFIX = '/_seam/procedure/'\nconst EVENTS_SUFFIX = '.events'\n\ninterface WsData {\n\tchannelName: string\n\tchannelInput: unknown\n\tsession?: ChannelWsSession\n}\n\nexport function serveBun<T extends DefinitionMap>(router: Router<T>, opts?: ServeBunOptions) {\n\tconst handler = createHttpHandler(router, {\n\t\tstaticDir: opts?.staticDir,\n\t\tpublicDir: opts?.publicDir,\n\t\tfallback: opts?.fallback,\n\t\trpcHashMap: opts?.rpcHashMap,\n\t\tsseOptions: opts?.sseOptions,\n\t})\n\n\treturn Bun.serve<WsData>({\n\t\tport: opts?.port ?? 3000,\n\n\t\tasync fetch(req, server) {\n\t\t\t// WebSocket upgrade for channel paths\n\t\t\tif (req.method === 'GET' && req.headers.get('upgrade') === 'websocket') {\n\t\t\t\tconst url = new URL(req.url)\n\t\t\t\tconst { pathname } = url\n\t\t\t\tif (pathname.startsWith(PROCEDURE_PREFIX) && pathname.endsWith(EVENTS_SUFFIX)) {\n\t\t\t\t\tconst channelName = pathname.slice(PROCEDURE_PREFIX.length, -EVENTS_SUFFIX.length)\n\t\t\t\t\tconst rawInput = url.searchParams.get('input')\n\t\t\t\t\tlet channelInput: unknown\n\t\t\t\t\ttry {\n\t\t\t\t\t\tchannelInput = rawInput ? JSON.parse(rawInput) : {}\n\t\t\t\t\t} catch {\n\t\t\t\t\t\treturn new Response('Invalid input query parameter', { status: 400 })\n\t\t\t\t\t}\n\t\t\t\t\tconst upgraded = server.upgrade(req, {\n\t\t\t\t\t\tdata: { channelName, channelInput },\n\t\t\t\t\t})\n\t\t\t\t\tif (upgraded) return undefined\n\t\t\t\t\treturn new Response('WebSocket upgrade failed', { status: 500 })\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconst { body, file } = buildMultipartFields(req)\n\n\t\t\tconst result = await handler({\n\t\t\t\tmethod: req.method,\n\t\t\t\turl: req.url,\n\t\t\t\tbody,\n\t\t\t\theader: (name) => req.headers.get(name),\n\t\t\t\tfile,\n\t\t\t})\n\t\t\treturn toWebResponse(result)\n\t\t},\n\n\t\twebsocket: {\n\t\t\topen(ws) {\n\t\t\t\tconst { channelName, channelInput } = ws.data\n\t\t\t\tws.data.session = startChannelWs(\n\t\t\t\t\trouter,\n\t\t\t\t\tchannelName,\n\t\t\t\t\tchannelInput,\n\t\t\t\t\t{\n\t\t\t\t\t\tsend: (data) => ws.send(data),\n\t\t\t\t\t\tping: () => ws.ping(),\n\t\t\t\t\t\tclose: () => ws.close(),\n\t\t\t\t\t},\n\t\t\t\t\topts?.wsOptions,\n\t\t\t\t)\n\t\t\t},\n\t\t\tmessage(ws, message) {\n\t\t\t\tconst text = typeof message === 'string' ? message : new TextDecoder().decode(message)\n\t\t\t\tws.data.session?.onMessage(text)\n\t\t\t},\n\t\t\tpong(ws) {\n\t\t\t\tws.data.session?.onPong()\n\t\t\t},\n\t\t\tclose(ws) {\n\t\t\t\tws.data.session?.close()\n\t\t\t},\n\t\t},\n\t})\n}\n"],"mappings":";;AA4BA,MAAM,mBAAmB;AACzB,MAAM,gBAAgB;AAQtB,SAAgB,SAAkC,QAAmB,MAAwB;CAC5F,MAAM,UAAU,kBAAkB,QAAQ;EACzC,WAAW,MAAM;EACjB,WAAW,MAAM;EACjB,UAAU,MAAM;EAChB,YAAY,MAAM;EAClB,YAAY,MAAM;EAClB,CAAC;AAEF,QAAO,IAAI,MAAc;EACxB,MAAM,MAAM,QAAQ;EAEpB,MAAM,MAAM,KAAK,QAAQ;AAExB,OAAI,IAAI,WAAW,SAAS,IAAI,QAAQ,IAAI,UAAU,KAAK,aAAa;IACvE,MAAM,MAAM,IAAI,IAAI,IAAI,IAAI;IAC5B,MAAM,EAAE,aAAa;AACrB,QAAI,SAAS,WAAW,iBAAiB,IAAI,SAAS,SAAS,cAAc,EAAE;KAC9E,MAAM,cAAc,SAAS,MAAM,IAAyB,GAAsB;KAClF,MAAM,WAAW,IAAI,aAAa,IAAI,QAAQ;KAC9C,IAAI;AACJ,SAAI;AACH,qBAAe,WAAW,KAAK,MAAM,SAAS,GAAG,EAAE;aAC5C;AACP,aAAO,IAAI,SAAS,iCAAiC,EAAE,QAAQ,KAAK,CAAC;;AAKtE,SAHiB,OAAO,QAAQ,KAAK,EACpC,MAAM;MAAE;MAAa;MAAc,EACnC,CAAC,CACY,QAAO,KAAA;AACrB,YAAO,IAAI,SAAS,4BAA4B,EAAE,QAAQ,KAAK,CAAC;;;GAIlE,MAAM,EAAE,MAAM,SAAS,qBAAqB,IAAI;AAShD,UAAO,cAPQ,MAAM,QAAQ;IAC5B,QAAQ,IAAI;IACZ,KAAK,IAAI;IACT;IACA,SAAS,SAAS,IAAI,QAAQ,IAAI,KAAK;IACvC;IACA,CAAC,CAC0B;;EAG7B,WAAW;GACV,KAAK,IAAI;IACR,MAAM,EAAE,aAAa,iBAAiB,GAAG;AACzC,OAAG,KAAK,UAAU,eACjB,QACA,aACA,cACA;KACC,OAAO,SAAS,GAAG,KAAK,KAAK;KAC7B,YAAY,GAAG,MAAM;KACrB,aAAa,GAAG,OAAO;KACvB,EACD,MAAM,UACN;;GAEF,QAAQ,IAAI,SAAS;IACpB,MAAM,OAAO,OAAO,YAAY,WAAW,UAAU,IAAI,aAAa,CAAC,OAAO,QAAQ;AACtF,OAAG,KAAK,SAAS,UAAU,KAAK;;GAEjC,KAAK,IAAI;AACR,OAAG,KAAK,SAAS,QAAQ;;GAE1B,MAAM,IAAI;AACT,OAAG,KAAK,SAAS,OAAO;;GAEzB;EACD,CAAC"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@canmi/seam-adapter-bun",
3
- "version": "0.5.31",
3
+ "version": "0.5.38",
4
4
  "license": "MIT",
5
5
  "files": [
6
6
  "dist"
@@ -17,13 +17,13 @@
17
17
  "test": "bun test"
18
18
  },
19
19
  "devDependencies": {
20
- "@canmi/seam-server": "0.5.31",
21
- "@types/bun": "^1.3.9",
22
- "bun-types": "^1.3.10",
23
- "tsdown": "^0.20.3",
24
- "typescript": "^5.9.3"
20
+ "@canmi/seam-server": "0.5.38",
21
+ "@types/bun": "1",
22
+ "bun-types": "1",
23
+ "tsdown": "0.21",
24
+ "typescript": "6"
25
25
  },
26
26
  "peerDependencies": {
27
- "@canmi/seam-server": "0.5.31"
27
+ "@canmi/seam-server": "0.5.38"
28
28
  }
29
29
  }