vafast 0.4.24 → 0.5.2
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 +165 -2
- package/dist/{base-server-B7MYJNsl.mjs → base-server-CE0mfqPY.mjs} +11 -23
- package/dist/base-server-CE0mfqPY.mjs.map +1 -0
- package/dist/{base-server-DLxtulAO.d.mts → base-server-Contwrlf.d.mts} +5 -6
- package/dist/{base64url-C2zopQdH.mjs → base64url-Cc77f1ms.mjs} +1 -1
- package/dist/{base64url-C2zopQdH.mjs.map → base64url-Cc77f1ms.mjs.map} +1 -1
- package/dist/{base64url-Dwi2Afhc.d.mts → base64url-DNUGwekK.d.mts} +1 -1
- package/dist/{component-route-nrrO0iSI.d.mts → component-route-Di7R40-2.d.mts} +2 -2
- package/dist/{component-router-RwPL20vN.mjs → component-router-B6M5XDnp.mjs} +1 -1
- package/dist/{component-router-RwPL20vN.mjs.map → component-router-B6M5XDnp.mjs.map} +1 -1
- package/dist/{component-server-DomPJ_7S.mjs → component-server-Cya46YN3.mjs} +7 -7
- package/dist/component-server-Cya46YN3.mjs.map +1 -0
- package/dist/{component-server-JqpDC7wy.d.mts → component-server-fR4UV6Jq.d.mts} +4 -4
- package/dist/contract-BKqc9fFH.mjs +114 -0
- package/dist/contract-BKqc9fFH.mjs.map +1 -0
- package/dist/contract-BL3JflJ7.d.mts +71 -0
- package/dist/defineRoute-zjOfHyZA.d.mts +208 -0
- package/dist/defineRoute.d.mts +2 -60
- package/dist/defineRoute.mjs +160 -25
- package/dist/defineRoute.mjs.map +1 -1
- package/dist/{dependency-manager-DCmh7xFc.mjs → dependency-manager-CGMZJTer.mjs} +1 -1
- package/dist/{dependency-manager-DCmh7xFc.mjs.map → dependency-manager-CGMZJTer.mjs.map} +1 -1
- package/dist/{dependency-manager-C3_7ic4h.d.mts → dependency-manager-DIN9X0Gj.d.mts} +1 -1
- package/dist/{formats-Dk-DSBY4.d.mts → formats-DDDSFWP0.d.mts} +1 -1
- package/dist/{go-await-C4ZdEUwY.mjs → go-await-B1U27PgB.mjs} +1 -1
- package/dist/{go-await-C4ZdEUwY.mjs.map → go-await-B1U27PgB.mjs.map} +1 -1
- package/dist/{go-await-DL1A_-4X.d.mts → go-await-DPtVBug4.d.mts} +1 -1
- package/dist/{html-renderer-DTtJ_Yic.mjs → html-renderer-D1zzDVQM.mjs} +1 -1
- package/dist/{html-renderer-DTtJ_Yic.mjs.map → html-renderer-D1zzDVQM.mjs.map} +1 -1
- package/dist/{html-renderer-CfKK2BrP.d.mts → html-renderer-DhQxRuyi.d.mts} +1 -1
- package/dist/{index-Bj3SWrMU.d.mts → index-B8iPRxj1.d.mts} +6 -6
- package/dist/index.d.mts +22 -27
- package/dist/index.mjs +19 -22
- package/dist/index.mjs.map +1 -1
- package/dist/middleware/component-router.d.mts +1 -1
- package/dist/middleware/component-router.mjs +1 -1
- package/dist/{middleware-KXEoefLX.d.mts → middleware-BR-R4p0M.d.mts} +2 -2
- package/dist/middleware.d.mts +1 -3
- package/dist/middleware.mjs +1 -1
- package/dist/monitoring/index.d.mts +3 -6
- package/dist/monitoring/native-monitor.d.mts +3 -6
- package/dist/node-server/index.d.mts +1 -1
- package/dist/node-server/index.mjs +3 -3
- package/dist/node-server/request.mjs +1 -1
- package/dist/node-server/response.mjs +1 -1
- package/dist/node-server/serve.d.mts +1 -1
- package/dist/node-server/serve.mjs +2 -2
- package/dist/{parsers-BerGr2_q.d.mts → parsers-8hIAx0OV.d.mts} +1 -1
- package/dist/{parsers-DpH_mD0H.mjs → parsers-BrG_mRLq.mjs} +1 -1
- package/dist/{parsers-DpH_mD0H.mjs.map → parsers-BrG_mRLq.mjs.map} +1 -1
- package/dist/{path-matcher-CGczAIl_.mjs → path-matcher-Dp-DRUV9.mjs} +1 -1
- package/dist/{path-matcher-CGczAIl_.mjs.map → path-matcher-Dp-DRUV9.mjs.map} +1 -1
- package/dist/{radix-tree-qqSjnVXF.mjs → radix-tree-CccjvyTP.mjs} +1 -1
- package/dist/{radix-tree-qqSjnVXF.mjs.map → radix-tree-CccjvyTP.mjs.map} +1 -1
- package/dist/{request-B-Nct5f7.mjs → request-CppAFyqX.mjs} +1 -1
- package/dist/{request-B-Nct5f7.mjs.map → request-CppAFyqX.mjs.map} +1 -1
- package/dist/{response-BMfdEcTm.d.mts → response-BNLzz4Tq.d.mts} +2 -2
- package/dist/{middleware-CewKbtb4.mjs → response-CQ1IgWei.mjs} +55 -55
- package/dist/response-CQ1IgWei.mjs.map +1 -0
- package/dist/{response-BnkYA4pj.mjs → response-DNdvtn-K.mjs} +1 -1
- package/dist/{response-BnkYA4pj.mjs.map → response-DNdvtn-K.mjs.map} +1 -1
- package/dist/{route-registry-CmABJA2V.d.mts → route-registry-BQKQ0qZp.d.mts} +21 -7
- package/dist/{route-registry-emTmRrWQ.mjs → route-registry-BzExlM2t.mjs} +23 -9
- package/dist/route-registry-BzExlM2t.mjs.map +1 -0
- package/dist/router/index.d.mts +2 -4
- package/dist/router/index.mjs +3 -3
- package/dist/router/radix-tree.d.mts +1 -3
- package/dist/router/radix-tree.mjs +1 -1
- package/dist/router.d.mts +4 -7
- package/dist/router.mjs +27 -2
- package/dist/router.mjs.map +1 -0
- package/dist/{serve-48LEIkPa.mjs → serve-Df3HzwZM.mjs} +3 -3
- package/dist/{serve-48LEIkPa.mjs.map → serve-Df3HzwZM.mjs.map} +1 -1
- package/dist/{serve-AG80VaIr.d.mts → serve-DlzxgOhz.d.mts} +1 -1
- package/dist/serve.d.mts +1 -1
- package/dist/serve.mjs +2 -2
- package/dist/server/base-server.d.mts +1 -3
- package/dist/server/base-server.mjs +1 -1
- package/dist/server/component-server.d.mts +1 -4
- package/dist/server/component-server.mjs +2 -2
- package/dist/server/index.d.mts +5 -6
- package/dist/server/index.mjs +6 -6
- package/dist/server/server-factory.d.mts +3 -6
- package/dist/server/server-factory.mjs +5 -5
- package/dist/server/server.d.mts +2 -4
- package/dist/server/server.mjs +2 -2
- package/dist/{server-OUUug5DR.mjs → server-Bm0BGm01.mjs} +19 -30
- package/dist/server-Bm0BGm01.mjs.map +1 -0
- package/dist/{server-BUom6ISO.mjs → server-CZLmrJSk.mjs} +3 -3
- package/dist/server-CZLmrJSk.mjs.map +1 -0
- package/dist/server-CpKaM0Kq.d.mts +48 -0
- package/dist/{sse-CBl-szg1.mjs → sse-CWNz0ky7.mjs} +3 -4
- package/dist/sse-CWNz0ky7.mjs.map +1 -0
- package/dist/{sse-BgLhEo43.d.mts → sse-CsjEeJNP.d.mts} +2 -4
- package/dist/types/component-route.d.mts +1 -1
- package/dist/types/index.d.mts +3 -6
- package/dist/types/index.mjs +1 -3
- package/dist/types/types.d.mts +2 -2
- package/dist/types-B8Z3cMtZ.d.mts +12 -0
- package/dist/utils/base64url.d.mts +1 -1
- package/dist/utils/base64url.mjs +1 -1
- package/dist/utils/contract.d.mts +2 -0
- package/dist/utils/contract.mjs +3 -0
- package/dist/utils/dependency-manager.d.mts +1 -1
- package/dist/utils/dependency-manager.mjs +1 -1
- package/dist/utils/formats.d.mts +1 -1
- package/dist/utils/go-await.d.mts +1 -1
- package/dist/utils/go-await.mjs +1 -1
- package/dist/utils/handle.d.mts +13 -2
- package/dist/utils/handle.mjs +19 -3
- package/dist/utils/handle.mjs.map +1 -0
- package/dist/utils/html-renderer.d.mts +1 -1
- package/dist/utils/html-renderer.mjs +1 -1
- package/dist/utils/index.d.mts +13 -17
- package/dist/utils/index.mjs +11 -13
- package/dist/utils/parsers.d.mts +1 -1
- package/dist/utils/parsers.mjs +1 -1
- package/dist/utils/path-matcher.mjs +1 -1
- package/dist/utils/response.d.mts +1 -4
- package/dist/utils/response.mjs +1 -1
- package/dist/utils/route-registry.d.mts +3 -4
- package/dist/utils/route-registry.mjs +2 -2
- package/dist/utils/sse.d.mts +2 -2
- package/dist/utils/sse.mjs +3 -3
- package/dist/utils/validators/validators.d.mts +1 -1
- package/dist/utils/validators/validators.mjs +1 -1
- package/dist/{validators-Ch71zkT8.d.mts → validators-BFC6S_fr.d.mts} +1 -1
- package/dist/{validators-DBkyw6BG.mjs → validators-CkfvNBbK.mjs} +1 -1
- package/dist/{validators-DBkyw6BG.mjs.map → validators-CkfvNBbK.mjs.map} +1 -1
- package/package.json +1 -1
- package/dist/base-server-B7MYJNsl.mjs.map +0 -1
- package/dist/component-server-DomPJ_7S.mjs.map +0 -1
- package/dist/create-handler-DKw-sQOV.d.mts +0 -87
- package/dist/create-handler-RconAcAB.mjs +0 -165
- package/dist/create-handler-RconAcAB.mjs.map +0 -1
- package/dist/handle-BhR3oyky.d.mts +0 -15
- package/dist/handle-BxJwSvV0.mjs +0 -30
- package/dist/handle-BxJwSvV0.mjs.map +0 -1
- package/dist/middleware-CewKbtb4.mjs.map +0 -1
- package/dist/request-validator-Bz9X48FX.mjs +0 -77
- package/dist/request-validator-Bz9X48FX.mjs.map +0 -1
- package/dist/request-validator-Coo8dI-p.d.mts +0 -67
- package/dist/route-6A7umH7b.d.mts +0 -44
- package/dist/route-Ds53PR4M.mjs +0 -11
- package/dist/route-Ds53PR4M.mjs.map +0 -1
- package/dist/route-registry-emTmRrWQ.mjs.map +0 -1
- package/dist/router-CgYfGB4J.mjs +0 -98
- package/dist/router-CgYfGB4J.mjs.map +0 -1
- package/dist/schema-1fwiv7cm.mjs +0 -1
- package/dist/schema-B6DFN5c2.d.mts +0 -81
- package/dist/server-BUom6ISO.mjs.map +0 -1
- package/dist/server-C2RPKSvC.d.mts +0 -51
- package/dist/server-OUUug5DR.mjs.map +0 -1
- package/dist/sse-CBl-szg1.mjs.map +0 -1
- package/dist/types/route.d.mts +0 -2
- package/dist/types/route.mjs +0 -3
- package/dist/types/schema.d.mts +0 -2
- package/dist/types/schema.mjs +0 -1
- package/dist/types-D1PUFkda.d.mts +0 -45
- package/dist/utils/create-handler.d.mts +0 -3
- package/dist/utils/create-handler.mjs +0 -6
- package/dist/utils/request-validator.d.mts +0 -3
- package/dist/utils/request-validator.mjs +0 -5
- /package/dist/{component-route-DNgAj6VC.mjs → component-route-BiUHBq7a.mjs} +0 -0
- /package/dist/{index-CREkvfw9.d.mts → index-Dflz2i1X.d.mts} +0 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"response-CQ1IgWei.mjs","names":["err","stream"],"sources":["../src/middleware.ts","../src/utils/response.ts"],"sourcesContent":["// src/middleware.ts\n\nimport { json, mapResponse } from \"./utils/response\";\n\nimport type { Handler, Middleware } from \"./types\";\n/** 中间件类型:使用 next() 传递给下一个处理 */\n\n/** Vafast 自定义错误类型 */\nexport class VafastError extends Error {\n status: number;\n type: string;\n expose: boolean;\n\n constructor(\n message: string,\n options: {\n status?: number;\n type?: string;\n expose?: boolean;\n cause?: unknown;\n } = {},\n ) {\n super(message);\n this.name = \"VafastError\";\n this.status = options.status ?? 500;\n this.type = options.type ?? \"internal_error\";\n this.expose = options.expose ?? false;\n if (options.cause) (this as any).cause = options.cause;\n }\n}\n\n/**\n * 组合类型: 自动注入错误处理器进行中间件组合\n */\nexport function composeMiddleware(\n middleware: Middleware[],\n finalHandler: Handler,\n): (req: Request) => Promise<Response> {\n const all = [errorHandler, ...middleware];\n\n return function composedHandler(req: Request): Promise<Response> {\n let i = -1;\n\n const dispatch = (index: number): Promise<Response> => {\n if (index <= i)\n return Promise.reject(new Error(\"next() called multiple times\"));\n i = index;\n\n // 中间件阶段\n if (index < all.length) {\n const mw = all[index];\n return Promise.resolve(mw(req, () => dispatch(index + 1)));\n }\n\n // 最终 handler - 使用 mapResponse 转换返回值\n return Promise.resolve(finalHandler(req)).then(mapResponse);\n };\n\n return dispatch(0);\n };\n}\n\n/** 默认包含的全局错误处理器 */\nconst errorHandler: Middleware = async (req, next) => {\n try {\n return await next();\n } catch (err) {\n console.error(\"未处理的错误:\", err);\n\n if (err instanceof VafastError) {\n return json(\n {\n error: err.type,\n message: err.expose ? err.message : \"发生了一个错误\",\n },\n err.status,\n );\n }\n\n return json({ error: \"internal_error\", message: \"出现了一些问题\" }, 500);\n }\n};\n","// src/response.ts\n\nimport { VafastError } from \"../middleware\";\n\n/** 生成 JSON 响应 */\nexport function json(\n data: unknown,\n status = 200,\n headers: HeadersInit = {},\n): Response {\n const body = JSON.stringify(data);\n\n // 优化:只在有自定义 headers 时才创建 Headers 对象\n if (Object.keys(headers).length === 0) {\n return new Response(body, {\n status,\n headers: { \"Content-Type\": \"application/json\" },\n });\n }\n\n // 有自定义 headers 时才创建 Headers 对象\n const h = new Headers({\n \"Content-Type\": \"application/json\",\n ...headers,\n });\n\n return new Response(body, {\n status,\n headers: h,\n });\n}\n\n// JSON 响应的预创建 headers(避免每次创建)\nconst JSON_HEADERS = { \"Content-Type\": \"application/json\" };\nconst TEXT_HEADERS = { \"Content-Type\": \"text/plain\" };\n\n/**\n * 类型特化的响应映射\n * 根据返回值类型直接生成 Response,避免不必要的检查\n */\nexport function mapResponse(response: unknown): Response {\n // 快速路径:已经是 Response\n if (response instanceof Response) return response;\n\n // 使用 constructor.name 进行类型判断(比 instanceof 更快)\n switch (response?.constructor?.name) {\n case \"String\":\n return new Response(response as string, { headers: TEXT_HEADERS });\n\n case \"Object\":\n case \"Array\":\n return new Response(JSON.stringify(response), { headers: JSON_HEADERS });\n\n case \"Number\":\n case \"Boolean\":\n return new Response(String(response), { headers: TEXT_HEADERS });\n\n case undefined:\n // null 或 undefined\n return new Response(null, { status: 204 });\n\n case \"ReadableStream\":\n return new Response(response as ReadableStream);\n\n case \"Blob\":\n return new Response(response as Blob);\n\n case \"ArrayBuffer\":\n return new Response(response as ArrayBuffer);\n\n case \"Uint8Array\":\n return new Response(response as unknown as BodyInit);\n\n default:\n // Promise 处理\n if (response instanceof Promise) {\n return response.then(mapResponse) as unknown as Response;\n }\n // 其他情况使用 JSON 序列化\n return new Response(JSON.stringify(response), { headers: JSON_HEADERS });\n }\n}\n\n/** 生成重定向响应 */\nexport function redirect(location: string, status: 301 | 302 = 302): Response {\n return new Response(null, {\n status,\n headers: {\n Location: location,\n },\n });\n}\n\n/** 生成纯文本响应 */\nexport function text(\n content: string,\n status = 200,\n headers: HeadersInit = {},\n): Response {\n const h = new Headers({\n \"Content-Type\": \"text/plain; charset=utf-8\",\n ...headers,\n });\n\n return new Response(content, {\n status,\n headers: h,\n });\n}\n\n/** 生成HTML响应 */\nexport function html(\n content: string,\n status = 200,\n headers: HeadersInit = {},\n): Response {\n const h = new Headers({\n \"Content-Type\": \"text/html; charset=utf-8\",\n ...headers,\n });\n\n return new Response(content, {\n status,\n headers: h,\n });\n}\n\n/** 生成空响应(204 No Content) */\nexport function empty(status = 204, headers: HeadersInit = {}): Response {\n return new Response(null, {\n status,\n headers,\n });\n}\n\n/** 生成流式响应 */\nexport function stream(\n stream: ReadableStream,\n status = 200,\n headers: HeadersInit = {},\n): Response {\n const h = new Headers({\n \"Content-Type\": \"application/octet-stream\",\n ...headers,\n });\n\n return new Response(stream, {\n status,\n headers: h,\n });\n}\n\n// ==================== 错误响应工具 ====================\n\n/**\n * 创建错误响应\n *\n * @example\n * ```typescript\n * // 自定义错误\n * throw err('用户不存在', 404, 'NOT_FOUND')\n *\n * // 预定义错误\n * throw err.notFound('用户不存在')\n * throw err.badRequest('参数错误')\n * throw err.unauthorized('请先登录')\n * ```\n */\nexport function err(message: string, status = 500, type = \"ERROR\") {\n return new VafastError(message, { status, type, expose: true });\n}\n\n/** 400 Bad Request */\nerr.badRequest = (message = \"请求参数错误\") =>\n err(message, 400, \"BAD_REQUEST\");\n\n/** 401 Unauthorized */\nerr.unauthorized = (message = \"未授权\") =>\n err(message, 401, \"UNAUTHORIZED\");\n\n/** 403 Forbidden */\nerr.forbidden = (message = \"禁止访问\") =>\n err(message, 403, \"FORBIDDEN\");\n\n/** 404 Not Found */\nerr.notFound = (message = \"资源不存在\") =>\n err(message, 404, \"NOT_FOUND\");\n\n/** 409 Conflict */\nerr.conflict = (message = \"资源冲突\") =>\n err(message, 409, \"CONFLICT\");\n\n/** 422 Unprocessable Entity */\nerr.unprocessable = (message = \"无法处理的实体\") =>\n err(message, 422, \"UNPROCESSABLE_ENTITY\");\n\n/** 429 Too Many Requests */\nerr.tooMany = (message = \"请求过于频繁\") =>\n err(message, 429, \"TOO_MANY_REQUESTS\");\n\n/** 500 Internal Server Error */\nerr.internal = (message = \"服务器内部错误\") =>\n err(message, 500, \"INTERNAL_ERROR\");\n"],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"response-
|
|
1
|
+
{"version":3,"file":"response-DNdvtn-K.mjs","names":[],"sources":["../src/node-server/response.ts"],"sourcesContent":["/**\n * 优化的 Response 写入\n * 流式写入,避免内存拷贝\n */\n\nimport type { ServerResponse } from \"node:http\";\n\n/**\n * 构建 Node.js 响应头\n * 处理 set-cookie 多值情况\n */\nfunction buildOutgoingHeaders(\n headers: Headers,\n): Record<string, string | string[]> {\n const result: Record<string, string | string[]> = {};\n const cookies: string[] = [];\n\n headers.forEach((value, key) => {\n if (key === \"set-cookie\") {\n cookies.push(value);\n } else {\n result[key] = value;\n }\n });\n\n if (cookies.length > 0) {\n result[\"set-cookie\"] = cookies;\n }\n\n return result;\n}\n\n/**\n * 流式写入 Response body 到 ServerResponse\n * 支持背压处理,避免内存溢出\n */\nasync function writeBodyStream(\n body: ReadableStream<Uint8Array>,\n outgoing: ServerResponse,\n): Promise<void> {\n const reader = body.getReader();\n\n try {\n while (true) {\n const { done, value } = await reader.read();\n\n if (done) {\n break;\n }\n\n // 背压处理:如果写入返回 false,等待 drain 事件\n const canContinue = outgoing.write(value);\n if (!canContinue) {\n await new Promise<void>((resolve) => {\n outgoing.once(\"drain\", resolve);\n });\n }\n }\n } finally {\n reader.releaseLock();\n }\n}\n\n/**\n * 将 Web Response 写入 Node.js ServerResponse\n * 流式写入,零拷贝\n */\nexport async function writeResponse(\n response: Response,\n outgoing: ServerResponse,\n): Promise<void> {\n // 设置状态码\n outgoing.statusCode = response.status;\n\n // 设置响应头\n const headers = buildOutgoingHeaders(response.headers);\n for (const [key, value] of Object.entries(headers)) {\n outgoing.setHeader(key, value);\n }\n\n const body = response.body;\n\n // 无 body 的情况\n if (!body) {\n outgoing.end();\n return;\n }\n\n // 流式写入 body\n try {\n await writeBodyStream(body, outgoing);\n outgoing.end();\n } catch (error) {\n // 处理客户端提前断开等情况\n if (!outgoing.destroyed) {\n outgoing.destroy(\n error instanceof Error ? error : new Error(String(error)),\n );\n }\n }\n}\n\n/**\n * 简化版写入(用于已知小体积响应)\n * 直接 arrayBuffer 转换,适用于确定的小响应\n */\nexport async function writeResponseSimple(\n response: Response,\n outgoing: ServerResponse,\n): Promise<void> {\n outgoing.statusCode = response.status;\n\n const headers = buildOutgoingHeaders(response.headers);\n for (const [key, value] of Object.entries(headers)) {\n outgoing.setHeader(key, value);\n }\n\n const body = response.body;\n if (!body) {\n outgoing.end();\n return;\n }\n\n // 对于小响应,直接读取全部内容\n const buffer = await response.arrayBuffer();\n outgoing.end(Buffer.from(buffer));\n}\n"],"mappings":";;;;;AAWA,SAAS,qBACP,SACmC;CACnC,MAAM,SAA4C,EAAE;CACpD,MAAM,UAAoB,EAAE;AAE5B,SAAQ,SAAS,OAAO,QAAQ;AAC9B,MAAI,QAAQ,aACV,SAAQ,KAAK,MAAM;MAEnB,QAAO,OAAO;GAEhB;AAEF,KAAI,QAAQ,SAAS,EACnB,QAAO,gBAAgB;AAGzB,QAAO;;;;;;AAOT,eAAe,gBACb,MACA,UACe;CACf,MAAM,SAAS,KAAK,WAAW;AAE/B,KAAI;AACF,SAAO,MAAM;GACX,MAAM,EAAE,MAAM,UAAU,MAAM,OAAO,MAAM;AAE3C,OAAI,KACF;AAKF,OAAI,CADgB,SAAS,MAAM,MAAM,CAEvC,OAAM,IAAI,SAAe,YAAY;AACnC,aAAS,KAAK,SAAS,QAAQ;KAC/B;;WAGE;AACR,SAAO,aAAa;;;;;;;AAQxB,eAAsB,cACpB,UACA,UACe;AAEf,UAAS,aAAa,SAAS;CAG/B,MAAM,UAAU,qBAAqB,SAAS,QAAQ;AACtD,MAAK,MAAM,CAAC,KAAK,UAAU,OAAO,QAAQ,QAAQ,CAChD,UAAS,UAAU,KAAK,MAAM;CAGhC,MAAM,OAAO,SAAS;AAGtB,KAAI,CAAC,MAAM;AACT,WAAS,KAAK;AACd;;AAIF,KAAI;AACF,QAAM,gBAAgB,MAAM,SAAS;AACrC,WAAS,KAAK;UACP,OAAO;AAEd,MAAI,CAAC,SAAS,UACZ,UAAS,QACP,iBAAiB,QAAQ,QAAQ,IAAI,MAAM,OAAO,MAAM,CAAC,CAC1D;;;;;;;AASP,eAAsB,oBACpB,UACA,UACe;AACf,UAAS,aAAa,SAAS;CAE/B,MAAM,UAAU,qBAAqB,SAAS,QAAQ;AACtD,MAAK,MAAM,CAAC,KAAK,UAAU,OAAO,QAAQ,QAAQ,CAChD,UAAS,UAAU,KAAK,MAAM;AAIhC,KAAI,CADS,SAAS,MACX;AACT,WAAS,KAAK;AACd;;CAIF,MAAM,SAAS,MAAM,SAAS,aAAa;AAC3C,UAAS,IAAI,OAAO,KAAK,OAAO,CAAC"}
|
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { i as RouteSchema, r as ProcessedRoute } from "./defineRoute-zjOfHyZA.mjs";
|
|
2
|
+
import { n as Method } from "./types-B8Z3cMtZ.mjs";
|
|
2
3
|
|
|
3
4
|
//#region src/utils/route-registry.d.ts
|
|
4
5
|
|
|
@@ -8,9 +9,10 @@ import { a as Method, n as FlattenedRoute } from "./types-D1PUFkda.mjs";
|
|
|
8
9
|
interface RouteMeta {
|
|
9
10
|
method: Method;
|
|
10
11
|
path: string;
|
|
11
|
-
fullPath: string;
|
|
12
12
|
name?: string;
|
|
13
13
|
description?: string;
|
|
14
|
+
/** Schema 定义(用于契约生成) */
|
|
15
|
+
schema?: RouteSchema;
|
|
14
16
|
/** 扩展字段 */
|
|
15
17
|
[key: string]: unknown;
|
|
16
18
|
}
|
|
@@ -22,11 +24,11 @@ interface RouteMeta {
|
|
|
22
24
|
declare class RouteRegistry<T extends Record<string, unknown> = Record<string, unknown>> {
|
|
23
25
|
/** 所有路由元信息 */
|
|
24
26
|
private routes;
|
|
25
|
-
/** 路由映射表:METHOD:
|
|
27
|
+
/** 路由映射表:METHOD:path -> RouteMeta */
|
|
26
28
|
private routeMap;
|
|
27
29
|
/** 分类映射表:category -> RouteMeta[] */
|
|
28
30
|
private categoryMap;
|
|
29
|
-
constructor(routes:
|
|
31
|
+
constructor(routes: ProcessedRoute[]);
|
|
30
32
|
/**
|
|
31
33
|
* 构建注册表
|
|
32
34
|
*/
|
|
@@ -109,7 +111,7 @@ declare class RouteRegistry<T extends Record<string, unknown> = Record<string, u
|
|
|
109
111
|
* }
|
|
110
112
|
* ```
|
|
111
113
|
*/
|
|
112
|
-
declare function createRouteRegistry<T extends Record<string, unknown> = Record<string, unknown>>(routes:
|
|
114
|
+
declare function createRouteRegistry<T extends Record<string, unknown> = Record<string, unknown>>(routes: ProcessedRoute[]): RouteRegistry<T>;
|
|
113
115
|
/**
|
|
114
116
|
* 设置全局 registry(框架内部使用)
|
|
115
117
|
* @internal
|
|
@@ -171,6 +173,18 @@ declare function getAllRoutes(): RouteMeta[];
|
|
|
171
173
|
* ```
|
|
172
174
|
*/
|
|
173
175
|
declare function filterRoutes<K extends string>(field: K): (RouteMeta & Record<K, unknown>)[];
|
|
176
|
+
/**
|
|
177
|
+
* 按 HTTP 方法获取路由
|
|
178
|
+
*
|
|
179
|
+
* @example
|
|
180
|
+
* ```typescript
|
|
181
|
+
* import { getRoutesByMethod } from 'vafast'
|
|
182
|
+
*
|
|
183
|
+
* const getRoutes = getRoutesByMethod('GET')
|
|
184
|
+
* const postRoutes = getRoutesByMethod('POST')
|
|
185
|
+
* ```
|
|
186
|
+
*/
|
|
187
|
+
declare function getRoutesByMethod(method: string): RouteMeta[];
|
|
174
188
|
//#endregion
|
|
175
|
-
export { getAllRoutes as a,
|
|
176
|
-
//# sourceMappingURL=route-registry-
|
|
189
|
+
export { getAllRoutes as a, getRoutesByMethod as c, filterRoutes as i, setGlobalRegistry as l, RouteRegistry as n, getRoute as o, createRouteRegistry as r, getRouteRegistry as s, RouteMeta as t };
|
|
190
|
+
//# sourceMappingURL=route-registry-BQKQ0qZp.d.mts.map
|
|
@@ -7,7 +7,7 @@
|
|
|
7
7
|
var RouteRegistry = class {
|
|
8
8
|
/** 所有路由元信息 */
|
|
9
9
|
routes = [];
|
|
10
|
-
/** 路由映射表:METHOD:
|
|
10
|
+
/** 路由映射表:METHOD:path -> RouteMeta */
|
|
11
11
|
routeMap = /* @__PURE__ */ new Map();
|
|
12
12
|
/** 分类映射表:category -> RouteMeta[] */
|
|
13
13
|
categoryMap = /* @__PURE__ */ new Map();
|
|
@@ -22,23 +22,23 @@ var RouteRegistry = class {
|
|
|
22
22
|
const meta = {
|
|
23
23
|
method: route.method,
|
|
24
24
|
path: route.path,
|
|
25
|
-
fullPath: route.fullPath,
|
|
26
25
|
name: route.name,
|
|
27
|
-
description: route.description
|
|
26
|
+
description: route.description,
|
|
27
|
+
schema: route.schema
|
|
28
28
|
};
|
|
29
29
|
for (const key of Object.keys(route)) if (![
|
|
30
30
|
"method",
|
|
31
31
|
"path",
|
|
32
|
-
"fullPath",
|
|
33
32
|
"name",
|
|
34
33
|
"description",
|
|
35
34
|
"handler",
|
|
36
35
|
"middleware",
|
|
37
|
-
"
|
|
36
|
+
"schema",
|
|
37
|
+
"docs"
|
|
38
38
|
].includes(key)) meta[key] = route[key];
|
|
39
39
|
this.routes.push(meta);
|
|
40
|
-
this.routeMap.set(`${route.method}:${route.
|
|
41
|
-
const category = this.extractCategory(route.
|
|
40
|
+
this.routeMap.set(`${route.method}:${route.path}`, meta);
|
|
41
|
+
const category = this.extractCategory(route.path);
|
|
42
42
|
if (!this.categoryMap.has(category)) this.categoryMap.set(category, []);
|
|
43
43
|
this.categoryMap.get(category).push(meta);
|
|
44
44
|
}
|
|
@@ -220,7 +220,21 @@ function getAllRoutes() {
|
|
|
220
220
|
function filterRoutes(field) {
|
|
221
221
|
return getRouteRegistry().filter(field);
|
|
222
222
|
}
|
|
223
|
+
/**
|
|
224
|
+
* 按 HTTP 方法获取路由
|
|
225
|
+
*
|
|
226
|
+
* @example
|
|
227
|
+
* ```typescript
|
|
228
|
+
* import { getRoutesByMethod } from 'vafast'
|
|
229
|
+
*
|
|
230
|
+
* const getRoutes = getRoutesByMethod('GET')
|
|
231
|
+
* const postRoutes = getRoutesByMethod('POST')
|
|
232
|
+
* ```
|
|
233
|
+
*/
|
|
234
|
+
function getRoutesByMethod(method) {
|
|
235
|
+
return getRouteRegistry().filterBy((r) => r.method === method);
|
|
236
|
+
}
|
|
223
237
|
|
|
224
238
|
//#endregion
|
|
225
|
-
export { getRoute as a, getAllRoutes as i, createRouteRegistry as n, getRouteRegistry as o, filterRoutes as r,
|
|
226
|
-
//# sourceMappingURL=route-registry-
|
|
239
|
+
export { getRoute as a, setGlobalRegistry as c, getAllRoutes as i, createRouteRegistry as n, getRouteRegistry as o, filterRoutes as r, getRoutesByMethod as s, RouteRegistry as t };
|
|
240
|
+
//# sourceMappingURL=route-registry-BzExlM2t.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"route-registry-BzExlM2t.mjs","names":[],"sources":["../src/utils/route-registry.ts"],"sourcesContent":["/**\n * 路由注册表\n *\n * 提供路由元信息的收集和查询能力\n * 可用于:API 文档生成、Webhook 事件注册、权限检查、审计日志等\n *\n * @example\n * ```typescript\n * // 创建注册表\n * const registry = createRouteRegistry(server.getRoutes())\n *\n * // 查询路由\n * const route = registry.get('POST', '/auth/signIn')\n *\n * // 筛选有 webhook 配置的路由\n * const webhookRoutes = registry.filter('webhook')\n *\n * // 按分类获取\n * const authRoutes = registry.getByCategory('auth')\n * ```\n */\n\nimport type { Method } from '../types'\nimport type { ProcessedRoute, RouteSchema } from '../defineRoute'\n\n/**\n * 路由元信息(不含 handler 和 middleware)\n */\nexport interface RouteMeta {\n method: Method\n path: string\n name?: string\n description?: string\n /** Schema 定义(用于契约生成) */\n schema?: RouteSchema\n /** 扩展字段 */\n [key: string]: unknown\n}\n\n/**\n * 路由注册表\n *\n * 泛型 T 用于定义扩展字段的类型\n */\nexport class RouteRegistry<T extends Record<string, unknown> = Record<string, unknown>> {\n /** 所有路由元信息 */\n private routes: RouteMeta[] = []\n\n /** 路由映射表:METHOD:path -> RouteMeta */\n private routeMap = new Map<string, RouteMeta>()\n\n /** 分类映射表:category -> RouteMeta[] */\n private categoryMap = new Map<string, RouteMeta[]>()\n\n constructor(routes: ProcessedRoute[]) {\n this.buildRegistry(routes)\n }\n\n /**\n * 构建注册表\n */\n private buildRegistry(routes: ProcessedRoute[]): void {\n for (const route of routes) {\n // 提取元信息(排除 handler 和 middleware)\n const meta: RouteMeta = {\n method: route.method as Method,\n path: route.path,\n name: route.name,\n description: route.description,\n schema: route.schema, // 保留 schema 用于契约生成\n }\n\n // 复制扩展字段\n for (const key of Object.keys(route)) {\n if (!['method', 'path', 'name', 'description', 'handler', 'middleware', 'schema', 'docs'].includes(key)) {\n meta[key] = route[key]\n }\n }\n\n this.routes.push(meta)\n this.routeMap.set(`${route.method}:${route.path}`, meta)\n\n // 按分类索引\n const category = this.extractCategory(route.path)\n if (!this.categoryMap.has(category)) {\n this.categoryMap.set(category, [])\n }\n this.categoryMap.get(category)!.push(meta)\n }\n }\n\n /**\n * 提取分类(第一段路径)\n */\n private extractCategory(path: string): string {\n const segments = path.split('/').filter(Boolean)\n return segments[0] || 'root'\n }\n\n // ============================================\n // 查询接口\n // ============================================\n\n /**\n * 获取所有路由元信息\n */\n getAll(): RouteMeta[] {\n return [...this.routes]\n }\n\n /**\n * 按 method + path 查询路由\n */\n get(method: string, path: string): (RouteMeta & T) | undefined {\n return this.routeMap.get(`${method}:${path}`) as (RouteMeta & T) | undefined\n }\n\n /**\n * 检查路由是否存在\n */\n has(method: string, path: string): boolean {\n return this.routeMap.has(`${method}:${path}`)\n }\n\n /**\n * 按分类获取路由\n */\n getByCategory(category: string): RouteMeta[] {\n return this.categoryMap.get(category) || []\n }\n\n /**\n * 获取所有分类\n */\n getCategories(): string[] {\n return Array.from(this.categoryMap.keys()).sort()\n }\n\n /**\n * 筛选有特定字段的路由\n *\n * @example\n * ```typescript\n * // 获取所有配置了 webhook 的路由\n * const webhookRoutes = registry.filter('webhook')\n * ```\n */\n filter<K extends string>(field: K): (RouteMeta & Record<K, unknown>)[] {\n return this.routes.filter((r) => field in r && r[field] !== undefined) as (RouteMeta & Record<K, unknown>)[]\n }\n\n /**\n * 按条件筛选路由\n *\n * @example\n * ```typescript\n * // 获取所有 POST 请求\n * const postRoutes = registry.filterBy(r => r.method === 'POST')\n * ```\n */\n filterBy(predicate: (route: RouteMeta) => boolean): RouteMeta[] {\n return this.routes.filter(predicate)\n }\n\n /**\n * 获取路由数量\n */\n get size(): number {\n return this.routes.length\n }\n\n /**\n * 遍历所有路由\n */\n forEach(callback: (route: RouteMeta, index: number) => void): void {\n this.routes.forEach(callback)\n }\n\n /**\n * 映射所有路由\n */\n map<R>(callback: (route: RouteMeta, index: number) => R): R[] {\n return this.routes.map(callback)\n }\n}\n\n/**\n * 创建路由注册表\n *\n * @example\n * ```typescript\n * // 定义扩展字段类型\n * interface MyRouteMeta {\n * webhook?: { eventKey: string }\n * permission?: string\n * }\n *\n * // 创建带类型的注册表\n * const registry = createRouteRegistry<MyRouteMeta>(server.getRoutes())\n *\n * // 类型安全的查询\n * const route = registry.get('POST', '/auth/signIn')\n * if (route?.webhook) {\n * console.log(route.webhook.eventKey)\n * }\n * ```\n */\nexport function createRouteRegistry<T extends Record<string, unknown> = Record<string, unknown>>(\n routes: ProcessedRoute[]\n): RouteRegistry<T> {\n return new RouteRegistry<T>(routes)\n}\n\n// ============================================\n// 全局 Registry(单例模式)\n// ============================================\n\n/** 全局 registry 实例 */\nlet globalRegistry: RouteRegistry | null = null\n\n/**\n * 设置全局 registry(框架内部使用)\n * @internal\n */\nexport function setGlobalRegistry(registry: RouteRegistry): void {\n globalRegistry = registry\n}\n\n/**\n * 获取全局路由注册表\n *\n * @example\n * ```typescript\n * // 在任意文件中\n * import { getRouteRegistry } from 'vafast'\n *\n * const registry = getRouteRegistry()\n * const webhookRoutes = registry.filter('webhook')\n * ```\n *\n * @throws 如果 Server 尚未创建\n */\nexport function getRouteRegistry<T extends Record<string, unknown> = Record<string, unknown>>(): RouteRegistry<T> {\n if (!globalRegistry) {\n throw new Error('RouteRegistry not initialized. Make sure Server is created first.')\n }\n return globalRegistry as RouteRegistry<T>\n}\n\n/**\n * 按 method + path 获取单个路由\n *\n * 便捷函数,无需先获取 registry\n *\n * @example\n * ```typescript\n * // 在中间件或接口中\n * import { getRoute } from 'vafast'\n *\n * const route = getRoute('POST', '/auth/signIn')\n * if (route?.webhook) {\n * console.log('This route has webhook:', route.webhook)\n * }\n * ```\n */\nexport function getRoute<T extends Record<string, unknown> = Record<string, unknown>>(\n method: string,\n path: string\n): (RouteMeta & T) | undefined {\n return getRouteRegistry<T>().get(method, path)\n}\n\n/**\n * 获取所有路由\n *\n * @example\n * ```typescript\n * import { getAllRoutes } from 'vafast'\n *\n * const routes = getAllRoutes()\n * console.log(`Total ${routes.length} routes`)\n * ```\n */\nexport function getAllRoutes(): RouteMeta[] {\n return getRouteRegistry().getAll()\n}\n\n/**\n * 筛选有特定字段的路由\n *\n * @example\n * ```typescript\n * import { filterRoutes } from 'vafast'\n *\n * // 获取所有配置了 webhook 的路由\n * const webhookRoutes = filterRoutes('webhook')\n * ```\n */\nexport function filterRoutes<K extends string>(field: K): (RouteMeta & Record<K, unknown>)[] {\n return getRouteRegistry().filter(field)\n}\n\n/**\n * 按 HTTP 方法获取路由\n *\n * @example\n * ```typescript\n * import { getRoutesByMethod } from 'vafast'\n *\n * const getRoutes = getRoutesByMethod('GET')\n * const postRoutes = getRoutesByMethod('POST')\n * ```\n */\nexport function getRoutesByMethod(method: string): RouteMeta[] {\n return getRouteRegistry().filterBy(r => r.method === method)\n}\n\n"],"mappings":";;;;;;AA4CA,IAAa,gBAAb,MAAwF;;CAEtF,AAAQ,SAAsB,EAAE;;CAGhC,AAAQ,2BAAW,IAAI,KAAwB;;CAG/C,AAAQ,8BAAc,IAAI,KAA0B;CAEpD,YAAY,QAA0B;AACpC,OAAK,cAAc,OAAO;;;;;CAM5B,AAAQ,cAAc,QAAgC;AACpD,OAAK,MAAM,SAAS,QAAQ;GAE1B,MAAM,OAAkB;IACtB,QAAQ,MAAM;IACd,MAAM,MAAM;IACZ,MAAM,MAAM;IACZ,aAAa,MAAM;IACnB,QAAQ,MAAM;IACf;AAGD,QAAK,MAAM,OAAO,OAAO,KAAK,MAAM,CAClC,KAAI,CAAC;IAAC;IAAU;IAAQ;IAAQ;IAAe;IAAW;IAAc;IAAU;IAAO,CAAC,SAAS,IAAI,CACrG,MAAK,OAAO,MAAM;AAItB,QAAK,OAAO,KAAK,KAAK;AACtB,QAAK,SAAS,IAAI,GAAG,MAAM,OAAO,GAAG,MAAM,QAAQ,KAAK;GAGxD,MAAM,WAAW,KAAK,gBAAgB,MAAM,KAAK;AACjD,OAAI,CAAC,KAAK,YAAY,IAAI,SAAS,CACjC,MAAK,YAAY,IAAI,UAAU,EAAE,CAAC;AAEpC,QAAK,YAAY,IAAI,SAAS,CAAE,KAAK,KAAK;;;;;;CAO9C,AAAQ,gBAAgB,MAAsB;AAE5C,SADiB,KAAK,MAAM,IAAI,CAAC,OAAO,QAAQ,CAChC,MAAM;;;;;CAUxB,SAAsB;AACpB,SAAO,CAAC,GAAG,KAAK,OAAO;;;;;CAMzB,IAAI,QAAgB,MAA2C;AAC7D,SAAO,KAAK,SAAS,IAAI,GAAG,OAAO,GAAG,OAAO;;;;;CAM/C,IAAI,QAAgB,MAAuB;AACzC,SAAO,KAAK,SAAS,IAAI,GAAG,OAAO,GAAG,OAAO;;;;;CAM/C,cAAc,UAA+B;AAC3C,SAAO,KAAK,YAAY,IAAI,SAAS,IAAI,EAAE;;;;;CAM7C,gBAA0B;AACxB,SAAO,MAAM,KAAK,KAAK,YAAY,MAAM,CAAC,CAAC,MAAM;;;;;;;;;;;CAYnD,OAAyB,OAA8C;AACrE,SAAO,KAAK,OAAO,QAAQ,MAAM,SAAS,KAAK,EAAE,WAAW,OAAU;;;;;;;;;;;CAYxE,SAAS,WAAuD;AAC9D,SAAO,KAAK,OAAO,OAAO,UAAU;;;;;CAMtC,IAAI,OAAe;AACjB,SAAO,KAAK,OAAO;;;;;CAMrB,QAAQ,UAA2D;AACjE,OAAK,OAAO,QAAQ,SAAS;;;;;CAM/B,IAAO,UAAuD;AAC5D,SAAO,KAAK,OAAO,IAAI,SAAS;;;;;;;;;;;;;;;;;;;;;;;;AAyBpC,SAAgB,oBACd,QACkB;AAClB,QAAO,IAAI,cAAiB,OAAO;;;AAQrC,IAAI,iBAAuC;;;;;AAM3C,SAAgB,kBAAkB,UAA+B;AAC/D,kBAAiB;;;;;;;;;;;;;;;;AAiBnB,SAAgB,mBAAkG;AAChH,KAAI,CAAC,eACH,OAAM,IAAI,MAAM,oEAAoE;AAEtF,QAAO;;;;;;;;;;;;;;;;;;AAmBT,SAAgB,SACd,QACA,MAC6B;AAC7B,QAAO,kBAAqB,CAAC,IAAI,QAAQ,KAAK;;;;;;;;;;;;;AAchD,SAAgB,eAA4B;AAC1C,QAAO,kBAAkB,CAAC,QAAQ;;;;;;;;;;;;;AAcpC,SAAgB,aAA+B,OAA8C;AAC3F,QAAO,kBAAkB,CAAC,OAAO,MAAM;;;;;;;;;;;;;AAczC,SAAgB,kBAAkB,QAA6B;AAC7D,QAAO,kBAAkB,CAAC,UAAS,MAAK,EAAE,WAAW,OAAO"}
|
package/dist/router/index.d.mts
CHANGED
|
@@ -1,5 +1,3 @@
|
|
|
1
|
-
import "../
|
|
2
|
-
import "../index-CREkvfw9.mjs";
|
|
3
|
-
import { flattenNestedRoutes, normalizePath } from "../router.mjs";
|
|
1
|
+
import { normalizePath } from "../router.mjs";
|
|
4
2
|
import { RadixRouter } from "./radix-tree.mjs";
|
|
5
|
-
export { RadixRouter,
|
|
3
|
+
export { RadixRouter, normalizePath };
|
package/dist/router/index.mjs
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import {
|
|
1
|
+
import { t as RadixRouter } from "../radix-tree-CccjvyTP.mjs";
|
|
2
|
+
import { normalizePath } from "../router.mjs";
|
|
3
3
|
|
|
4
|
-
export { RadixRouter,
|
|
4
|
+
export { RadixRouter, normalizePath };
|
|
@@ -1,6 +1,4 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import "../schema-B6DFN5c2.mjs";
|
|
3
|
-
import "../index-CREkvfw9.mjs";
|
|
1
|
+
import { n as Method, r as Middleware, t as Handler } from "../types-B8Z3cMtZ.mjs";
|
|
4
2
|
|
|
5
3
|
//#region src/router/radix-tree.d.ts
|
|
6
4
|
|
package/dist/router.d.mts
CHANGED
|
@@ -1,10 +1,7 @@
|
|
|
1
|
-
import { l as Route, n as FlattenedRoute, s as NestedRoute } from "./types-D1PUFkda.mjs";
|
|
2
|
-
import "./schema-B6DFN5c2.mjs";
|
|
3
|
-
import "./index-CREkvfw9.mjs";
|
|
4
|
-
|
|
5
1
|
//#region src/router.d.ts
|
|
6
|
-
|
|
7
|
-
|
|
2
|
+
/**
|
|
3
|
+
* 路由工具函数
|
|
4
|
+
*/
|
|
8
5
|
/**
|
|
9
6
|
* 标准化路径
|
|
10
7
|
*
|
|
@@ -20,5 +17,5 @@ declare function flattenNestedRoutes(routes: (Route | NestedRoute)[]): Flattened
|
|
|
20
17
|
*/
|
|
21
18
|
declare function normalizePath(path: string): string;
|
|
22
19
|
//#endregion
|
|
23
|
-
export {
|
|
20
|
+
export { normalizePath };
|
|
24
21
|
//# sourceMappingURL=router.d.mts.map
|
package/dist/router.mjs
CHANGED
|
@@ -1,3 +1,28 @@
|
|
|
1
|
-
|
|
1
|
+
//#region src/router.ts
|
|
2
|
+
/**
|
|
3
|
+
* 路由工具函数
|
|
4
|
+
*/
|
|
5
|
+
/**
|
|
6
|
+
* 标准化路径
|
|
7
|
+
*
|
|
8
|
+
* - 解码 URL 编码字符
|
|
9
|
+
* - 去除重复斜杠
|
|
10
|
+
* - 处理结尾斜杠
|
|
11
|
+
*
|
|
12
|
+
* @example
|
|
13
|
+
* ```typescript
|
|
14
|
+
* normalizePath("//api//users/") // "/api/users"
|
|
15
|
+
* normalizePath("/api/%20test") // "/api/ test"
|
|
16
|
+
* ```
|
|
17
|
+
*/
|
|
18
|
+
function normalizePath(path) {
|
|
19
|
+
let normalized = decodeURIComponent(path);
|
|
20
|
+
normalized = normalized.replace(/\/+/g, "/");
|
|
21
|
+
if (normalized === "") return "/";
|
|
22
|
+
if (normalized !== "/" && normalized.endsWith("/")) normalized = normalized.slice(0, -1);
|
|
23
|
+
return normalized;
|
|
24
|
+
}
|
|
2
25
|
|
|
3
|
-
|
|
26
|
+
//#endregion
|
|
27
|
+
export { normalizePath };
|
|
28
|
+
//# sourceMappingURL=router.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"router.mjs","names":[],"sources":["../src/router.ts"],"sourcesContent":["/**\n * 路由工具函数\n */\n\n/**\n * 标准化路径\n *\n * - 解码 URL 编码字符\n * - 去除重复斜杠\n * - 处理结尾斜杠\n *\n * @example\n * ```typescript\n * normalizePath(\"//api//users/\") // \"/api/users\"\n * normalizePath(\"/api/%20test\") // \"/api/ test\"\n * ```\n */\nexport function normalizePath(path: string): string {\n // 解码 URL 编码\n let normalized = decodeURIComponent(path);\n\n // 去除重复斜杠\n normalized = normalized.replace(/\\/+/g, \"/\");\n\n // 空路径转为根路径\n if (normalized === \"\") return \"/\";\n\n // 去除结尾斜杠(根路径除外)\n if (normalized !== \"/\" && normalized.endsWith(\"/\")) {\n normalized = normalized.slice(0, -1);\n }\n\n return normalized;\n}\n"],"mappings":";;;;;;;;;;;;;;;;;AAiBA,SAAgB,cAAc,MAAsB;CAElD,IAAI,aAAa,mBAAmB,KAAK;AAGzC,cAAa,WAAW,QAAQ,QAAQ,IAAI;AAG5C,KAAI,eAAe,GAAI,QAAO;AAG9B,KAAI,eAAe,OAAO,WAAW,SAAS,IAAI,CAChD,cAAa,WAAW,MAAM,GAAG,GAAG;AAGtC,QAAO"}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import { t as createProxyRequest } from "./request-
|
|
2
|
-
import { t as writeResponse } from "./response-
|
|
1
|
+
import { t as createProxyRequest } from "./request-CppAFyqX.mjs";
|
|
2
|
+
import { t as writeResponse } from "./response-DNdvtn-K.mjs";
|
|
3
3
|
import { createServer } from "node:http";
|
|
4
4
|
|
|
5
5
|
//#region src/node-server/serve.ts
|
|
@@ -104,4 +104,4 @@ function createAdaptorServer(fetch, onError) {
|
|
|
104
104
|
|
|
105
105
|
//#endregion
|
|
106
106
|
export { serve as n, createAdaptorServer as t };
|
|
107
|
-
//# sourceMappingURL=serve-
|
|
107
|
+
//# sourceMappingURL=serve-Df3HzwZM.mjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"serve-
|
|
1
|
+
{"version":3,"file":"serve-Df3HzwZM.mjs","names":[],"sources":["../src/node-server/serve.ts"],"sourcesContent":["/**\n * Node.js 服务器适配器\n * 提供类似 Bun.serve 的 API\n */\n\nimport {\n createServer,\n type Server as HttpServer,\n type IncomingMessage,\n type ServerResponse,\n} from \"node:http\";\nimport { createProxyRequest } from \"./request\";\nimport { writeResponse } from \"./response\";\n\n/** fetch 函数类型 */\nexport type FetchHandler = (request: Request) => Response | Promise<Response>;\n\n/** 优雅关闭配置 */\nexport interface GracefulShutdownOptions {\n /** 关闭超时时间(毫秒),默认 30000 */\n timeout?: number;\n /** 关闭前回调 */\n onShutdown?: () => void | Promise<void>;\n /** 关闭完成回调 */\n onShutdownComplete?: () => void;\n /** 监听的信号,默认 ['SIGINT', 'SIGTERM'] */\n signals?: NodeJS.Signals[];\n}\n\n/** serve 配置选项 */\nexport interface ServeOptions {\n /** fetch 处理函数 */\n fetch: FetchHandler;\n /** 端口号,默认 3000 */\n port?: number;\n /** 主机名,默认 0.0.0.0 */\n hostname?: string;\n /** 错误处理函数 */\n onError?: (error: Error) => Response | Promise<Response>;\n /** 优雅关闭配置,设置为 true 使用默认配置 */\n gracefulShutdown?: boolean | GracefulShutdownOptions;\n}\n\n/** serve 返回的服务器信息 */\nexport interface ServeResult {\n /** Node.js HTTP Server 实例 */\n server: HttpServer;\n /** 服务器端口 */\n port: number;\n /** 服务器主机名 */\n hostname: string;\n /** 关闭服务器 */\n stop: () => Promise<void>;\n /** 优雅关闭(等待现有请求完成) */\n shutdown: () => Promise<void>;\n}\n\n/**\n * 创建请求处理函数\n */\nfunction createRequestHandler(\n fetch: FetchHandler,\n defaultHost: string,\n onError?: (error: Error) => Response | Promise<Response>,\n) {\n return async (incoming: IncomingMessage, outgoing: ServerResponse) => {\n try {\n // 创建代理 Request(延迟创建真实 Request)\n const request = createProxyRequest(incoming, defaultHost);\n\n // 调用 fetch handler\n const response = await fetch(request);\n\n // 流式写入 Response\n await writeResponse(response, outgoing);\n } catch (error) {\n // 错误处理\n const err = error instanceof Error ? error : new Error(String(error));\n\n if (onError) {\n try {\n const errorResponse = await onError(err);\n await writeResponse(errorResponse, outgoing);\n return;\n } catch {\n // onError 也失败了,返回 500\n }\n }\n\n // 默认错误响应\n if (!outgoing.headersSent) {\n outgoing.statusCode = 500;\n outgoing.setHeader(\"Content-Type\", \"text/plain\");\n outgoing.end(\"Internal Server Error\");\n }\n }\n };\n}\n\n/**\n * 启动 HTTP 服务器\n *\n * @example\n * ```ts\n * import { serve } from \"@vafast/node-server\";\n * import { Server } from \"vafast\";\n *\n * const app = new Server([\n * { method: \"GET\", path: \"/\", handler: () => \"Hello World\" },\n * ]);\n *\n * serve({ fetch: app.fetch, port: 3000 }, () => {\n * console.log(\"Server running on http://localhost:3000\");\n * });\n * ```\n */\nexport function serve(\n options: ServeOptions,\n callback?: () => void,\n): ServeResult {\n const { fetch, port = 3000, hostname = \"0.0.0.0\", onError, gracefulShutdown } = options;\n\n const defaultHost = `${hostname === \"0.0.0.0\" ? \"localhost\" : hostname}:${port}`;\n const handler = createRequestHandler(fetch, defaultHost, onError);\n\n const server = createServer(handler);\n\n // 追踪活跃连接\n const connections = new Set<import(\"node:net\").Socket>();\n\n server.on(\"connection\", (socket) => {\n connections.add(socket);\n socket.on(\"close\", () => connections.delete(socket));\n });\n\n // 优雅关闭函数\n let isShuttingDown = false;\n\n const shutdown = async (): Promise<void> => {\n if (isShuttingDown) return;\n isShuttingDown = true;\n\n const shutdownOptions: GracefulShutdownOptions =\n typeof gracefulShutdown === \"object\" ? gracefulShutdown : {};\n\n const timeout = shutdownOptions.timeout ?? 30000;\n\n // 执行关闭前回调\n if (shutdownOptions.onShutdown) {\n await shutdownOptions.onShutdown();\n }\n\n return new Promise<void>((resolve) => {\n // 设置超时强制关闭\n const forceCloseTimer = setTimeout(() => {\n // 强制关闭所有连接\n for (const socket of connections) {\n socket.destroy();\n }\n connections.clear();\n resolve();\n }, timeout);\n\n // 停止接受新连接\n server.close(() => {\n clearTimeout(forceCloseTimer);\n shutdownOptions.onShutdownComplete?.();\n resolve();\n });\n\n // 关闭空闲连接\n for (const socket of connections) {\n // 如果连接空闲,立即关闭\n if (!socket.writableLength) {\n socket.end();\n }\n }\n });\n };\n\n // 注册信号处理\n if (gracefulShutdown) {\n const shutdownOptions: GracefulShutdownOptions =\n typeof gracefulShutdown === \"object\" ? gracefulShutdown : {};\n\n const signals = shutdownOptions.signals ?? [\"SIGINT\", \"SIGTERM\"];\n\n for (const signal of signals) {\n process.on(signal, () => {\n shutdown().then(() => process.exit(0));\n });\n }\n }\n\n // 启动服务器\n server.listen(port, hostname, callback);\n\n return {\n server,\n port,\n hostname,\n stop: () =>\n new Promise<void>((resolve, reject) => {\n server.close((err) => {\n if (err) reject(err);\n else resolve();\n });\n }),\n shutdown,\n };\n}\n\n/**\n * 创建适配器服务器(不自动启动)\n * 用于需要更多控制的场景\n */\nexport function createAdaptorServer(\n fetch: FetchHandler,\n onError?: (error: Error) => Response | Promise<Response>,\n): HttpServer {\n const handler = createRequestHandler(fetch, \"localhost\", onError);\n return createServer(handler);\n}\n"],"mappings":";;;;;;;;;;;;AA4DA,SAAS,qBACP,OACA,aACA,SACA;AACA,QAAO,OAAO,UAA2B,aAA6B;AACpE,MAAI;AAQF,SAAM,cAHW,MAAM,MAHP,mBAAmB,UAAU,YAAY,CAGpB,EAGP,SAAS;WAChC,OAAO;GAEd,MAAM,MAAM,iBAAiB,QAAQ,QAAQ,IAAI,MAAM,OAAO,MAAM,CAAC;AAErE,OAAI,QACF,KAAI;AAEF,UAAM,cADgB,MAAM,QAAQ,IAAI,EACL,SAAS;AAC5C;WACM;AAMV,OAAI,CAAC,SAAS,aAAa;AACzB,aAAS,aAAa;AACtB,aAAS,UAAU,gBAAgB,aAAa;AAChD,aAAS,IAAI,wBAAwB;;;;;;;;;;;;;;;;;;;;;;AAuB7C,SAAgB,MACd,SACA,UACa;CACb,MAAM,EAAE,OAAO,OAAO,KAAM,WAAW,WAAW,SAAS,qBAAqB;CAKhF,MAAM,SAAS,aAFC,qBAAqB,OADjB,GAAG,aAAa,YAAY,cAAc,SAAS,GAAG,QACjB,QAAQ,CAE7B;CAGpC,MAAM,8BAAc,IAAI,KAAgC;AAExD,QAAO,GAAG,eAAe,WAAW;AAClC,cAAY,IAAI,OAAO;AACvB,SAAO,GAAG,eAAe,YAAY,OAAO,OAAO,CAAC;GACpD;CAGF,IAAI,iBAAiB;CAErB,MAAM,WAAW,YAA2B;AAC1C,MAAI,eAAgB;AACpB,mBAAiB;EAEjB,MAAM,kBACJ,OAAO,qBAAqB,WAAW,mBAAmB,EAAE;EAE9D,MAAM,UAAU,gBAAgB,WAAW;AAG3C,MAAI,gBAAgB,WAClB,OAAM,gBAAgB,YAAY;AAGpC,SAAO,IAAI,SAAe,YAAY;GAEpC,MAAM,kBAAkB,iBAAiB;AAEvC,SAAK,MAAM,UAAU,YACnB,QAAO,SAAS;AAElB,gBAAY,OAAO;AACnB,aAAS;MACR,QAAQ;AAGX,UAAO,YAAY;AACjB,iBAAa,gBAAgB;AAC7B,oBAAgB,sBAAsB;AACtC,aAAS;KACT;AAGF,QAAK,MAAM,UAAU,YAEnB,KAAI,CAAC,OAAO,eACV,QAAO,KAAK;IAGhB;;AAIJ,KAAI,kBAAkB;EAIpB,MAAM,WAFJ,OAAO,qBAAqB,WAAW,mBAAmB,EAAE,EAE9B,WAAW,CAAC,UAAU,UAAU;AAEhE,OAAK,MAAM,UAAU,QACnB,SAAQ,GAAG,cAAc;AACvB,aAAU,CAAC,WAAW,QAAQ,KAAK,EAAE,CAAC;IACtC;;AAKN,QAAO,OAAO,MAAM,UAAU,SAAS;AAEvC,QAAO;EACL;EACA;EACA;EACA,YACE,IAAI,SAAe,SAAS,WAAW;AACrC,UAAO,OAAO,QAAQ;AACpB,QAAI,IAAK,QAAO,IAAI;QACf,UAAS;KACd;IACF;EACJ;EACD;;;;;;AAOH,SAAgB,oBACd,OACA,SACY;AAEZ,QAAO,aADS,qBAAqB,OAAO,aAAa,QAAQ,CACrC"}
|
|
@@ -66,4 +66,4 @@ declare function serve(options: ServeOptions, callback?: () => void): ServeResul
|
|
|
66
66
|
declare function createAdaptorServer(fetch: FetchHandler, onError?: (error: Error) => Response | Promise<Response>): Server;
|
|
67
67
|
//#endregion
|
|
68
68
|
export { createAdaptorServer as a, ServeResult as i, GracefulShutdownOptions as n, serve as o, ServeOptions as r, FetchHandler as t };
|
|
69
|
-
//# sourceMappingURL=serve-
|
|
69
|
+
//# sourceMappingURL=serve-DlzxgOhz.d.mts.map
|
package/dist/serve.d.mts
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import { a as createAdaptorServer, i as ServeResult, o as serve, r as ServeOptions, t as FetchHandler } from "./serve-
|
|
1
|
+
import { a as createAdaptorServer, i as ServeResult, o as serve, r as ServeOptions, t as FetchHandler } from "./serve-DlzxgOhz.mjs";
|
|
2
2
|
export { type FetchHandler, type ServeOptions, type ServeResult, createAdaptorServer, serve };
|
package/dist/serve.mjs
CHANGED
|
@@ -1,5 +1,2 @@
|
|
|
1
|
-
import "../
|
|
2
|
-
import "../index-CREkvfw9.mjs";
|
|
3
|
-
import "../base-server-DLxtulAO.mjs";
|
|
4
|
-
import { t as ComponentServer } from "../component-server-JqpDC7wy.mjs";
|
|
1
|
+
import { t as ComponentServer } from "../component-server-fR4UV6Jq.mjs";
|
|
5
2
|
export { ComponentServer };
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import "../dependency-manager-
|
|
2
|
-
import { t as ComponentServer } from "../component-server-
|
|
1
|
+
import "../dependency-manager-CGMZJTer.mjs";
|
|
2
|
+
import { t as ComponentServer } from "../component-server-Cya46YN3.mjs";
|
|
3
3
|
|
|
4
4
|
export { ComponentServer };
|
package/dist/server/index.d.mts
CHANGED
|
@@ -1,7 +1,6 @@
|
|
|
1
|
-
import "../
|
|
2
|
-
import "../
|
|
3
|
-
import { t as
|
|
4
|
-
import { t as
|
|
5
|
-
import { t as
|
|
6
|
-
import { t as ServerFactory } from "../index-Bj3SWrMU.mjs";
|
|
1
|
+
import "../defineRoute-zjOfHyZA.mjs";
|
|
2
|
+
import { t as BaseServer } from "../base-server-Contwrlf.mjs";
|
|
3
|
+
import { t as Server } from "../server-CpKaM0Kq.mjs";
|
|
4
|
+
import { t as ComponentServer } from "../component-server-fR4UV6Jq.mjs";
|
|
5
|
+
import { t as ServerFactory } from "../index-B8iPRxj1.mjs";
|
|
7
6
|
export { BaseServer, ComponentServer, Server, ServerFactory };
|
package/dist/server/index.mjs
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
|
-
import "../
|
|
2
|
-
import { t as BaseServer } from "../base-server-
|
|
3
|
-
import { t as Server } from "../server-
|
|
4
|
-
import "../dependency-manager-
|
|
5
|
-
import { t as ComponentServer } from "../component-server-
|
|
6
|
-
import { t as ServerFactory } from "../server-
|
|
1
|
+
import "../response-CQ1IgWei.mjs";
|
|
2
|
+
import { t as BaseServer } from "../base-server-CE0mfqPY.mjs";
|
|
3
|
+
import { t as Server } from "../server-Bm0BGm01.mjs";
|
|
4
|
+
import "../dependency-manager-CGMZJTer.mjs";
|
|
5
|
+
import { t as ComponentServer } from "../component-server-Cya46YN3.mjs";
|
|
6
|
+
import { t as ServerFactory } from "../server-CZLmrJSk.mjs";
|
|
7
7
|
|
|
8
8
|
export { BaseServer, ComponentServer, Server, ServerFactory };
|
|
@@ -1,7 +1,4 @@
|
|
|
1
|
-
import "../
|
|
2
|
-
import "../
|
|
3
|
-
import "../
|
|
4
|
-
import "../server-C2RPKSvC.mjs";
|
|
5
|
-
import "../component-server-JqpDC7wy.mjs";
|
|
6
|
-
import { t as ServerFactory } from "../index-Bj3SWrMU.mjs";
|
|
1
|
+
import "../defineRoute-zjOfHyZA.mjs";
|
|
2
|
+
import "../server-CpKaM0Kq.mjs";
|
|
3
|
+
import { t as ServerFactory } from "../index-B8iPRxj1.mjs";
|
|
7
4
|
export { ServerFactory };
|
|
@@ -1,7 +1,7 @@
|
|
|
1
|
-
import "../
|
|
2
|
-
import "../server-
|
|
3
|
-
import "../dependency-manager-
|
|
4
|
-
import "../component-server-
|
|
5
|
-
import { t as ServerFactory } from "../server-
|
|
1
|
+
import "../response-CQ1IgWei.mjs";
|
|
2
|
+
import "../server-Bm0BGm01.mjs";
|
|
3
|
+
import "../dependency-manager-CGMZJTer.mjs";
|
|
4
|
+
import "../component-server-Cya46YN3.mjs";
|
|
5
|
+
import { t as ServerFactory } from "../server-CZLmrJSk.mjs";
|
|
6
6
|
|
|
7
7
|
export { ServerFactory };
|
package/dist/server/server.d.mts
CHANGED
|
@@ -1,5 +1,3 @@
|
|
|
1
|
-
import "../
|
|
2
|
-
import "../
|
|
3
|
-
import "../base-server-DLxtulAO.mjs";
|
|
4
|
-
import { t as Server } from "../server-C2RPKSvC.mjs";
|
|
1
|
+
import "../defineRoute-zjOfHyZA.mjs";
|
|
2
|
+
import { t as Server } from "../server-CpKaM0Kq.mjs";
|
|
5
3
|
export { Server };
|
package/dist/server/server.mjs
CHANGED
|
@@ -1,8 +1,7 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import {
|
|
3
|
-
import { t as
|
|
4
|
-
import { t as
|
|
5
|
-
import { s as setGlobalRegistry, t as RouteRegistry } from "./route-registry-emTmRrWQ.mjs";
|
|
1
|
+
import { i as json, u as composeMiddleware } from "./response-CQ1IgWei.mjs";
|
|
2
|
+
import { t as BaseServer } from "./base-server-CE0mfqPY.mjs";
|
|
3
|
+
import { t as RadixRouter } from "./radix-tree-CccjvyTP.mjs";
|
|
4
|
+
import { c as setGlobalRegistry, t as RouteRegistry } from "./route-registry-BzExlM2t.mjs";
|
|
6
5
|
|
|
7
6
|
//#region src/server/server.ts
|
|
8
7
|
/**
|
|
@@ -10,10 +9,11 @@ import { s as setGlobalRegistry, t as RouteRegistry } from "./route-registry-emT
|
|
|
10
9
|
*
|
|
11
10
|
* @example
|
|
12
11
|
* ```typescript
|
|
13
|
-
* const
|
|
14
|
-
* { method: "GET", path: "/", handler: () =>
|
|
15
|
-
* ])
|
|
16
|
-
*
|
|
12
|
+
* const routes = defineRoutes([
|
|
13
|
+
* defineRoute({ method: "GET", path: "/", handler: () => "Hello" }),
|
|
14
|
+
* ])
|
|
15
|
+
* const server = new Server(routes)
|
|
16
|
+
* export default { fetch: server.fetch }
|
|
17
17
|
* ```
|
|
18
18
|
*/
|
|
19
19
|
var Server = class extends BaseServer {
|
|
@@ -26,11 +26,10 @@ var Server = class extends BaseServer {
|
|
|
26
26
|
if (routes.length > 0) this.registerRoutes([...routes]);
|
|
27
27
|
}
|
|
28
28
|
registerRoutes(routes) {
|
|
29
|
-
|
|
30
|
-
this.
|
|
31
|
-
|
|
32
|
-
this.
|
|
33
|
-
this.logFlattenedRoutes(flattened);
|
|
29
|
+
this.routes.push(...routes);
|
|
30
|
+
for (const route of routes) this.router.register(route.method, route.path, route.handler, route.middleware || []);
|
|
31
|
+
this.detectRouteConflicts(routes);
|
|
32
|
+
this.logRoutes(routes);
|
|
34
33
|
setGlobalRegistry(new RouteRegistry(this.routes));
|
|
35
34
|
}
|
|
36
35
|
/** 快速提取 pathname */
|
|
@@ -82,33 +81,23 @@ var Server = class extends BaseServer {
|
|
|
82
81
|
if (this.globalMiddleware.length > 0) return composeMiddleware(this.globalMiddleware, () => this.createErrorResponse(method, pathname))(req);
|
|
83
82
|
return this.createErrorResponse(method, pathname);
|
|
84
83
|
};
|
|
84
|
+
/** 动态添加单个路由 */
|
|
85
85
|
addRoute(route) {
|
|
86
|
-
|
|
87
|
-
...route,
|
|
88
|
-
fullPath: route.path,
|
|
89
|
-
middlewareChain: route.middleware || []
|
|
90
|
-
};
|
|
91
|
-
this.routes.push(flattenedRoute);
|
|
86
|
+
this.routes.push(route);
|
|
92
87
|
this.router.register(route.method, route.path, route.handler, route.middleware || []);
|
|
93
88
|
}
|
|
89
|
+
/** 动态添加多个路由 */
|
|
94
90
|
addRoutes(routes) {
|
|
95
91
|
this.registerRoutes([...routes]);
|
|
96
92
|
}
|
|
93
|
+
/** 获取路由列表 */
|
|
97
94
|
getRoutes() {
|
|
98
95
|
return this.router.getRoutes();
|
|
99
96
|
}
|
|
100
97
|
/**
|
|
101
|
-
*
|
|
98
|
+
* 获取完整的路由元信息
|
|
102
99
|
*
|
|
103
100
|
* 用于 API 文档生成、Webhook 事件注册、权限检查等场景
|
|
104
|
-
*
|
|
105
|
-
* @example
|
|
106
|
-
* ```typescript
|
|
107
|
-
* const routes = server.getRoutesWithMeta()
|
|
108
|
-
* for (const route of routes) {
|
|
109
|
-
* console.log(route.fullPath, route.name, route.description)
|
|
110
|
-
* }
|
|
111
|
-
* ```
|
|
112
101
|
*/
|
|
113
102
|
getRoutesWithMeta() {
|
|
114
103
|
return this.routes;
|
|
@@ -117,4 +106,4 @@ var Server = class extends BaseServer {
|
|
|
117
106
|
|
|
118
107
|
//#endregion
|
|
119
108
|
export { Server as t };
|
|
120
|
-
//# sourceMappingURL=server-
|
|
109
|
+
//# sourceMappingURL=server-Bm0BGm01.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"server-Bm0BGm01.mjs","names":[],"sources":["../src/server/server.ts"],"sourcesContent":["/**\n * Vafast 核心服务器\n *\n * 基于 Radix Tree 的高性能路由匹配\n * 时间复杂度: O(k),k 为路径段数\n */\n\nimport type { Method, Middleware } from \"../types\";\nimport type { ProcessedRoute } from \"../defineRoute\";\nimport { composeMiddleware } from \"../middleware\";\nimport { json } from \"../utils/response\";\nimport { BaseServer } from \"./base-server\";\nimport { RadixRouter } from \"../router/radix-tree\";\nimport { RouteRegistry, setGlobalRegistry } from \"../utils/route-registry\";\n\n/**\n * Vafast 服务器\n *\n * @example\n * ```typescript\n * const routes = defineRoutes([\n * defineRoute({ method: \"GET\", path: \"/\", handler: () => \"Hello\" }),\n * ])\n * const server = new Server(routes)\n * export default { fetch: server.fetch }\n * ```\n */\nexport class Server extends BaseServer {\n private router: RadixRouter;\n private routes: ProcessedRoute[];\n\n constructor(routes: readonly ProcessedRoute[] = []) {\n super();\n this.router = new RadixRouter();\n this.routes = [];\n\n if (routes.length > 0) {\n this.registerRoutes([...routes]);\n }\n }\n\n private registerRoutes(routes: ProcessedRoute[]): void {\n this.routes.push(...routes);\n\n for (const route of routes) {\n this.router.register(\n route.method as Method,\n route.path,\n route.handler,\n (route.middleware || []) as Middleware[],\n );\n }\n\n this.detectRouteConflicts(routes);\n this.logRoutes(routes);\n\n // 自动设置全局 RouteRegistry\n setGlobalRegistry(new RouteRegistry(this.routes));\n }\n\n /** 快速提取 pathname */\n private extractPathname(url: string): string {\n let start = url.indexOf(\"://\");\n start = start === -1 ? 0 : start + 3;\n\n const pathStart = url.indexOf(\"/\", start);\n if (pathStart === -1) return \"/\";\n\n let end = url.indexOf(\"?\", pathStart);\n if (end === -1) end = url.indexOf(\"#\", pathStart);\n if (end === -1) end = url.length;\n\n return url.substring(pathStart, end) || \"/\";\n }\n\n /** 生成 404/405 响应 */\n private createErrorResponse(method: string, pathname: string): Response {\n const allowedMethods = this.router.getAllowedMethods(pathname);\n if (allowedMethods.length > 0) {\n return json(\n {\n success: false,\n error: \"Method Not Allowed\",\n message: `Method ${method} not allowed for this endpoint`,\n allowedMethods,\n },\n 405,\n { Allow: allowedMethods.join(\", \") },\n );\n }\n return json({ success: false, error: \"Not Found\" }, 404);\n }\n\n /** 处理请求 */\n fetch = async (req: Request): Promise<Response> => {\n const pathname = this.extractPathname(req.url);\n const method = req.method as Method;\n\n const match = this.router.match(method, pathname);\n\n if (match) {\n (req as unknown as Record<string, unknown>).params = match.params;\n\n // 运行时组合中间件\n const allMiddleware = [...this.globalMiddleware, ...match.middleware];\n const handler = composeMiddleware(allMiddleware, match.handler);\n\n return handler(req);\n }\n\n // OPTIONS 预检请求特殊处理:查找同路径其他方法的路由,使用其中间件\n // 这允许路由级 CORS 中间件正常工作\n if (method === \"OPTIONS\") {\n const allowedMethods = this.router.getAllowedMethods(pathname);\n if (allowedMethods.length > 0) {\n // 尝试获取该路径任意方法的路由中间件\n const anyMatch = this.router.match(\n allowedMethods[0] as Method,\n pathname,\n );\n const routeMiddleware = anyMatch?.middleware || [];\n const allMiddleware = [...this.globalMiddleware, ...routeMiddleware];\n\n // OPTIONS 请求默认返回 204(中间件如 CORS 可能会提前响应)\n const optionsHandler = () =>\n new Response(null, {\n status: 204,\n headers: { Allow: allowedMethods.join(\", \") },\n });\n\n const handler = composeMiddleware(allMiddleware, optionsHandler);\n return handler(req);\n }\n }\n\n // 未匹配路由时,仍执行全局中间件(如 CORS 处理 OPTIONS 预检)\n if (this.globalMiddleware.length > 0) {\n const handler = composeMiddleware(this.globalMiddleware, () =>\n this.createErrorResponse(method, pathname),\n );\n return handler(req);\n }\n\n return this.createErrorResponse(method, pathname);\n };\n\n /** 动态添加单个路由 */\n addRoute(route: ProcessedRoute): void {\n this.routes.push(route);\n this.router.register(\n route.method as Method,\n route.path,\n route.handler,\n (route.middleware || []) as Middleware[],\n );\n }\n\n /** 动态添加多个路由 */\n addRoutes(routes: readonly ProcessedRoute[]): void {\n this.registerRoutes([...routes]);\n }\n\n /** 获取路由列表 */\n getRoutes(): Array<{ method: Method; path: string }> {\n return this.router.getRoutes();\n }\n\n /**\n * 获取完整的路由元信息\n *\n * 用于 API 文档生成、Webhook 事件注册、权限检查等场景\n */\n getRoutesWithMeta(): ProcessedRoute[] {\n return this.routes;\n }\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;AA2BA,IAAa,SAAb,cAA4B,WAAW;CACrC,AAAQ;CACR,AAAQ;CAER,YAAY,SAAoC,EAAE,EAAE;AAClD,SAAO;AACP,OAAK,SAAS,IAAI,aAAa;AAC/B,OAAK,SAAS,EAAE;AAEhB,MAAI,OAAO,SAAS,EAClB,MAAK,eAAe,CAAC,GAAG,OAAO,CAAC;;CAIpC,AAAQ,eAAe,QAAgC;AACrD,OAAK,OAAO,KAAK,GAAG,OAAO;AAE3B,OAAK,MAAM,SAAS,OAClB,MAAK,OAAO,SACV,MAAM,QACN,MAAM,MACN,MAAM,SACL,MAAM,cAAc,EAAE,CACxB;AAGH,OAAK,qBAAqB,OAAO;AACjC,OAAK,UAAU,OAAO;AAGtB,oBAAkB,IAAI,cAAc,KAAK,OAAO,CAAC;;;CAInD,AAAQ,gBAAgB,KAAqB;EAC3C,IAAI,QAAQ,IAAI,QAAQ,MAAM;AAC9B,UAAQ,UAAU,KAAK,IAAI,QAAQ;EAEnC,MAAM,YAAY,IAAI,QAAQ,KAAK,MAAM;AACzC,MAAI,cAAc,GAAI,QAAO;EAE7B,IAAI,MAAM,IAAI,QAAQ,KAAK,UAAU;AACrC,MAAI,QAAQ,GAAI,OAAM,IAAI,QAAQ,KAAK,UAAU;AACjD,MAAI,QAAQ,GAAI,OAAM,IAAI;AAE1B,SAAO,IAAI,UAAU,WAAW,IAAI,IAAI;;;CAI1C,AAAQ,oBAAoB,QAAgB,UAA4B;EACtE,MAAM,iBAAiB,KAAK,OAAO,kBAAkB,SAAS;AAC9D,MAAI,eAAe,SAAS,EAC1B,QAAO,KACL;GACE,SAAS;GACT,OAAO;GACP,SAAS,UAAU,OAAO;GAC1B;GACD,EACD,KACA,EAAE,OAAO,eAAe,KAAK,KAAK,EAAE,CACrC;AAEH,SAAO,KAAK;GAAE,SAAS;GAAO,OAAO;GAAa,EAAE,IAAI;;;CAI1D,QAAQ,OAAO,QAAoC;EACjD,MAAM,WAAW,KAAK,gBAAgB,IAAI,IAAI;EAC9C,MAAM,SAAS,IAAI;EAEnB,MAAM,QAAQ,KAAK,OAAO,MAAM,QAAQ,SAAS;AAEjD,MAAI,OAAO;AACT,GAAC,IAA2C,SAAS,MAAM;AAM3D,UAFgB,kBADM,CAAC,GAAG,KAAK,kBAAkB,GAAG,MAAM,WAAW,EACpB,MAAM,QAAQ,CAEhD,IAAI;;AAKrB,MAAI,WAAW,WAAW;GACxB,MAAM,iBAAiB,KAAK,OAAO,kBAAkB,SAAS;AAC9D,OAAI,eAAe,SAAS,GAAG;IAM7B,MAAM,kBAJW,KAAK,OAAO,MAC3B,eAAe,IACf,SACD,EACiC,cAAc,EAAE;IAClD,MAAM,gBAAgB,CAAC,GAAG,KAAK,kBAAkB,GAAG,gBAAgB;IAGpE,MAAM,uBACJ,IAAI,SAAS,MAAM;KACjB,QAAQ;KACR,SAAS,EAAE,OAAO,eAAe,KAAK,KAAK,EAAE;KAC9C,CAAC;AAGJ,WADgB,kBAAkB,eAAe,eAAe,CACjD,IAAI;;;AAKvB,MAAI,KAAK,iBAAiB,SAAS,EAIjC,QAHgB,kBAAkB,KAAK,wBACrC,KAAK,oBAAoB,QAAQ,SAAS,CAC3C,CACc,IAAI;AAGrB,SAAO,KAAK,oBAAoB,QAAQ,SAAS;;;CAInD,SAAS,OAA6B;AACpC,OAAK,OAAO,KAAK,MAAM;AACvB,OAAK,OAAO,SACV,MAAM,QACN,MAAM,MACN,MAAM,SACL,MAAM,cAAc,EAAE,CACxB;;;CAIH,UAAU,QAAyC;AACjD,OAAK,eAAe,CAAC,GAAG,OAAO,CAAC;;;CAIlC,YAAqD;AACnD,SAAO,KAAK,OAAO,WAAW;;;;;;;CAQhC,oBAAsC;AACpC,SAAO,KAAK"}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import { t as Server } from "./server-
|
|
2
|
-
import { t as ComponentServer } from "./component-server-
|
|
1
|
+
import { t as Server } from "./server-Bm0BGm01.mjs";
|
|
2
|
+
import { t as ComponentServer } from "./component-server-Cya46YN3.mjs";
|
|
3
3
|
|
|
4
4
|
//#region src/server/server-factory.ts
|
|
5
5
|
/**
|
|
@@ -67,4 +67,4 @@ var ServerFactory = class {
|
|
|
67
67
|
|
|
68
68
|
//#endregion
|
|
69
69
|
export { ServerFactory as t };
|
|
70
|
-
//# sourceMappingURL=server-
|
|
70
|
+
//# sourceMappingURL=server-CZLmrJSk.mjs.map
|