|
1 | 1 | import { createAdaptorServer, type ServerType } from "@hono/node-server" |
2 | 2 | import { createNodeWebSocket } from "@hono/node-ws" |
3 | 3 | import type { Hono } from "hono" |
4 | | -import type { Adapter } from "./adapter" |
| 4 | +import type { Adapter, FetchApp, Opts } from "./adapter" |
| 5 | + |
| 6 | +async function listen(app: FetchApp, opts: Opts, inject?: (server: ServerType) => void) { |
| 7 | + const start = (port: number) => |
| 8 | + new Promise<ServerType>((resolve, reject) => { |
| 9 | + const server = createAdaptorServer({ fetch: app.fetch }) |
| 10 | + inject?.(server) |
| 11 | + const fail = (err: Error) => { |
| 12 | + cleanup() |
| 13 | + reject(err) |
| 14 | + } |
| 15 | + const ready = () => { |
| 16 | + cleanup() |
| 17 | + resolve(server) |
| 18 | + } |
| 19 | + const cleanup = () => { |
| 20 | + server.off("error", fail) |
| 21 | + server.off("listening", ready) |
| 22 | + } |
| 23 | + server.once("error", fail) |
| 24 | + server.once("listening", ready) |
| 25 | + server.listen(port, opts.hostname) |
| 26 | + }) |
| 27 | + |
| 28 | + const server = opts.port === 0 ? await start(4096).catch(() => start(0)) : await start(opts.port) |
| 29 | + const addr = server.address() |
| 30 | + if (!addr || typeof addr === "string") { |
| 31 | + throw new Error(`Failed to resolve server address for port ${opts.port}`) |
| 32 | + } |
| 33 | + |
| 34 | + let closing: Promise<void> | undefined |
| 35 | + return { |
| 36 | + port: addr.port, |
| 37 | + stop(close?: boolean) { |
| 38 | + closing ??= new Promise<void>((resolve, reject) => { |
| 39 | + server.close((err) => { |
| 40 | + if (err) { |
| 41 | + reject(err) |
| 42 | + return |
| 43 | + } |
| 44 | + resolve() |
| 45 | + }) |
| 46 | + if (close) { |
| 47 | + if ("closeAllConnections" in server && typeof server.closeAllConnections === "function") { |
| 48 | + server.closeAllConnections() |
| 49 | + } |
| 50 | + if ("closeIdleConnections" in server && typeof server.closeIdleConnections === "function") { |
| 51 | + server.closeIdleConnections() |
| 52 | + } |
| 53 | + } |
| 54 | + }) |
| 55 | + return closing |
| 56 | + }, |
| 57 | + } |
| 58 | +} |
5 | 59 |
|
6 | 60 | export const adapter: Adapter = { |
7 | 61 | create(app: Hono) { |
8 | 62 | const ws = createNodeWebSocket({ app }) |
9 | 63 | return { |
10 | 64 | upgradeWebSocket: ws.upgradeWebSocket, |
11 | | - async listen(opts) { |
12 | | - const start = (port: number) => |
13 | | - new Promise<ServerType>((resolve, reject) => { |
14 | | - const server = createAdaptorServer({ fetch: app.fetch }) |
15 | | - ws.injectWebSocket(server) |
16 | | - const fail = (err: Error) => { |
17 | | - cleanup() |
18 | | - reject(err) |
19 | | - } |
20 | | - const ready = () => { |
21 | | - cleanup() |
22 | | - resolve(server) |
23 | | - } |
24 | | - const cleanup = () => { |
25 | | - server.off("error", fail) |
26 | | - server.off("listening", ready) |
27 | | - } |
28 | | - server.once("error", fail) |
29 | | - server.once("listening", ready) |
30 | | - server.listen(port, opts.hostname) |
31 | | - }) |
32 | | - |
33 | | - const server = opts.port === 0 ? await start(4096).catch(() => start(0)) : await start(opts.port) |
34 | | - const addr = server.address() |
35 | | - if (!addr || typeof addr === "string") { |
36 | | - throw new Error(`Failed to resolve server address for port ${opts.port}`) |
37 | | - } |
38 | | - |
39 | | - let closing: Promise<void> | undefined |
40 | | - return { |
41 | | - port: addr.port, |
42 | | - stop(close?: boolean) { |
43 | | - closing ??= new Promise((resolve, reject) => { |
44 | | - server.close((err) => { |
45 | | - if (err) { |
46 | | - reject(err) |
47 | | - return |
48 | | - } |
49 | | - resolve() |
50 | | - }) |
51 | | - if (close) { |
52 | | - if ("closeAllConnections" in server && typeof server.closeAllConnections === "function") { |
53 | | - server.closeAllConnections() |
54 | | - } |
55 | | - if ("closeIdleConnections" in server && typeof server.closeIdleConnections === "function") { |
56 | | - server.closeIdleConnections() |
57 | | - } |
58 | | - } |
59 | | - }) |
60 | | - return closing |
61 | | - }, |
62 | | - } |
63 | | - }, |
| 65 | + listen: (opts) => listen(app, opts, ws.injectWebSocket), |
| 66 | + } |
| 67 | + }, |
| 68 | + createFetch(app) { |
| 69 | + return { |
| 70 | + listen: (opts) => listen(app, opts), |
64 | 71 | } |
65 | 72 | }, |
66 | 73 | } |
0 commit comments