@richie-rpc/server 2.0.0 → 2.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cjs/index.cjs +24 -13
- package/dist/cjs/index.cjs.map +2 -2
- package/dist/cjs/package.json +1 -1
- package/dist/cjs/websocket.cjs +20 -10
- package/dist/cjs/websocket.cjs.map +3 -3
- package/dist/mjs/package.json +1 -1
- package/dist/mjs/websocket.mjs +2 -2
- package/dist/mjs/websocket.mjs.map +3 -3
- package/dist/types/websocket.d.ts +2 -1
- package/package.json +1 -1
package/dist/cjs/index.cjs
CHANGED
|
@@ -3,44 +3,55 @@
|
|
|
3
3
|
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
4
4
|
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
5
|
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
+
function __accessProp(key) {
|
|
7
|
+
return this[key];
|
|
8
|
+
}
|
|
6
9
|
var __reExport = (target, mod, secondTarget) => {
|
|
7
|
-
|
|
10
|
+
var keys = __getOwnPropNames(mod);
|
|
11
|
+
for (let key of keys)
|
|
8
12
|
if (!__hasOwnProp.call(target, key) && key !== "default")
|
|
9
13
|
__defProp(target, key, {
|
|
10
|
-
get: (
|
|
14
|
+
get: __accessProp.bind(mod, key),
|
|
11
15
|
enumerable: true
|
|
12
16
|
});
|
|
13
17
|
if (secondTarget) {
|
|
14
|
-
for (let key of
|
|
18
|
+
for (let key of keys)
|
|
15
19
|
if (!__hasOwnProp.call(secondTarget, key) && key !== "default")
|
|
16
20
|
__defProp(secondTarget, key, {
|
|
17
|
-
get: (
|
|
21
|
+
get: __accessProp.bind(mod, key),
|
|
18
22
|
enumerable: true
|
|
19
23
|
});
|
|
20
24
|
return secondTarget;
|
|
21
25
|
}
|
|
22
26
|
};
|
|
23
|
-
var __moduleCache = /* @__PURE__ */ new WeakMap;
|
|
24
27
|
var __toCommonJS = (from) => {
|
|
25
|
-
var entry = __moduleCache.get(from), desc;
|
|
28
|
+
var entry = (__moduleCache ??= new WeakMap).get(from), desc;
|
|
26
29
|
if (entry)
|
|
27
30
|
return entry;
|
|
28
31
|
entry = __defProp({}, "__esModule", { value: true });
|
|
29
|
-
if (from && typeof from === "object" || typeof from === "function")
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
32
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
33
|
+
for (var key of __getOwnPropNames(from))
|
|
34
|
+
if (!__hasOwnProp.call(entry, key))
|
|
35
|
+
__defProp(entry, key, {
|
|
36
|
+
get: __accessProp.bind(from, key),
|
|
37
|
+
enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
|
|
38
|
+
});
|
|
39
|
+
}
|
|
34
40
|
__moduleCache.set(from, entry);
|
|
35
41
|
return entry;
|
|
36
42
|
};
|
|
43
|
+
var __moduleCache;
|
|
44
|
+
var __returnValue = (v) => v;
|
|
45
|
+
function __exportSetter(name, newValue) {
|
|
46
|
+
this[name] = __returnValue.bind(null, newValue);
|
|
47
|
+
}
|
|
37
48
|
var __export = (target, all) => {
|
|
38
49
|
for (var name in all)
|
|
39
50
|
__defProp(target, name, {
|
|
40
51
|
get: all[name],
|
|
41
52
|
enumerable: true,
|
|
42
53
|
configurable: true,
|
|
43
|
-
set: (
|
|
54
|
+
set: __exportSetter.bind(all, name)
|
|
44
55
|
});
|
|
45
56
|
};
|
|
46
57
|
|
|
@@ -485,4 +496,4 @@ function createRouter(contract, handlers, options) {
|
|
|
485
496
|
}
|
|
486
497
|
})
|
|
487
498
|
|
|
488
|
-
//# debugId=
|
|
499
|
+
//# debugId=EE273BF08A26AE9264756E2164756E21
|
package/dist/cjs/index.cjs.map
CHANGED
|
@@ -4,7 +4,7 @@
|
|
|
4
4
|
"sourcesContent": [
|
|
5
5
|
"import type {\n Contract,\n DownloadEndpointDefinition,\n EndpointDefinition,\n ExtractBody,\n ExtractChunk,\n ExtractFinalResponse,\n ExtractHeaders,\n ExtractParams,\n ExtractQuery,\n ExtractSSEEventData,\n SSEEndpointDefinition,\n StandardEndpointDefinition,\n StreamingEndpointDefinition,\n} from '@richie-rpc/core';\nimport {\n formDataToObject,\n matchPath,\n parseQuery,\n Status,\n validatePathPattern,\n} from '@richie-rpc/core';\nimport { z } from 'zod';\n\n// Re-export Status for convenience\nexport { Status };\n\n// Handler input types (for standard endpoints)\nexport type HandlerInput<T extends StandardEndpointDefinition, C = unknown> = {\n params: ExtractParams<T>;\n query: ExtractQuery<T>;\n headers: ExtractHeaders<T>;\n body: ExtractBody<T>;\n request: Request;\n context: C;\n};\n\n// Handler response type (for standard endpoints)\nexport type HandlerResponse<T extends StandardEndpointDefinition> =\n | {\n [Status in keyof T['responses']]: {\n status: Status;\n body: T['responses'][Status] extends z.ZodTypeAny ? z.infer<T['responses'][Status]> : never;\n headers?: Record<string, string>;\n };\n }[keyof T['responses']]\n | (T['errorResponses'] extends Record<number, z.ZodTypeAny>\n ? {\n [S in keyof T['errorResponses']]: {\n status: S;\n body: T['errorResponses'][S] extends z.ZodTypeAny\n ? z.infer<T['errorResponses'][S]>\n : never;\n headers?: Record<string, string>;\n };\n }[keyof T['errorResponses']]\n : never);\n\n// Handler function type (for standard endpoints)\nexport type Handler<T extends StandardEndpointDefinition, C = unknown> = (\n input: HandlerInput<T, C>,\n) => Promise<HandlerResponse<T>> | HandlerResponse<T>;\n\n// ============================================\n// Streaming Endpoint Types\n// ============================================\n\n/**\n * Emitter for streaming responses - push-based API\n */\nexport interface StreamEmitter<T extends StreamingEndpointDefinition> {\n /** Send a chunk to the client */\n send(chunk: ExtractChunk<T>): void;\n /** Close the stream with optional final response */\n close(final?: ExtractFinalResponse<T>): void;\n /** Check if stream is still open */\n readonly isOpen: boolean;\n}\n\n/**\n * Handler input for streaming endpoints\n */\nexport type StreamingHandlerInput<T extends StreamingEndpointDefinition, C = unknown> = {\n params: ExtractParams<T>;\n query: ExtractQuery<T>;\n headers: ExtractHeaders<T>;\n body: ExtractBody<T>;\n request: Request;\n context: C;\n stream: StreamEmitter<T>;\n};\n\n/**\n * Handler function type for streaming endpoints\n */\nexport type StreamingHandler<T extends StreamingEndpointDefinition, C = unknown> = (\n input: StreamingHandlerInput<T, C>,\n) => void | Promise<void>;\n\n// ============================================\n// SSE Endpoint Types\n// ============================================\n\n/**\n * Emitter for SSE responses\n */\nexport interface SSEEmitter<T extends SSEEndpointDefinition> {\n /** Send an event to the client */\n send<K extends keyof T['events']>(\n event: K,\n data: ExtractSSEEventData<T, K>,\n options?: { id?: string },\n ): void;\n /** Close the connection */\n close(): void;\n /** Check if connection is still open */\n readonly isOpen: boolean;\n}\n\n/**\n * Handler input for SSE endpoints\n */\nexport type SSEHandlerInput<T extends SSEEndpointDefinition, C = unknown> = {\n params: ExtractParams<T>;\n query: ExtractQuery<T>;\n headers: ExtractHeaders<T>;\n request: Request;\n context: C;\n emitter: SSEEmitter<T>;\n /** AbortSignal for detecting client disconnect */\n signal: AbortSignal;\n};\n\n/**\n * Handler function type for SSE endpoints\n * Returns an optional cleanup function\n */\nexport type SSEHandler<T extends SSEEndpointDefinition, C = unknown> = (\n input: SSEHandlerInput<T, C>,\n) => void | (() => void) | Promise<void | (() => void)>;\n\n// ============================================\n// Download Endpoint Types\n// ============================================\n\n/**\n * Handler input for download endpoints\n */\nexport type DownloadHandlerInput<T extends DownloadEndpointDefinition, C = unknown> = {\n params: ExtractParams<T>;\n query: ExtractQuery<T>;\n headers: ExtractHeaders<T>;\n request: Request;\n context: C;\n};\n\n/**\n * Handler response for download endpoints\n * Success (200) returns File, errors return typed response\n */\nexport type DownloadHandlerResponse<T extends DownloadEndpointDefinition> =\n | { status: 200; body: File; headers?: Record<string, string> }\n | (T['errorResponses'] extends Record<number, z.ZodTypeAny>\n ? {\n [S in keyof T['errorResponses']]: {\n status: S;\n body: T['errorResponses'][S] extends z.ZodTypeAny\n ? z.infer<T['errorResponses'][S]>\n : never;\n headers?: Record<string, string>;\n };\n }[keyof T['errorResponses']]\n : never);\n\n/**\n * Handler function type for download endpoints\n */\nexport type DownloadHandler<T extends DownloadEndpointDefinition, C = unknown> = (\n input: DownloadHandlerInput<T, C>,\n) => Promise<DownloadHandlerResponse<T>> | DownloadHandlerResponse<T>;\n\n// ============================================\n// Contract Handlers (supports all endpoint types)\n// ============================================\n\n/**\n * Contract handlers mapping - conditionally applies handler type based on endpoint type\n */\nexport type ContractHandlers<T extends Contract, C = unknown> = {\n [K in keyof T]: T[K] extends StandardEndpointDefinition\n ? Handler<T[K], C>\n : T[K] extends StreamingEndpointDefinition\n ? StreamingHandler<T[K], C>\n : T[K] extends SSEEndpointDefinition\n ? SSEHandler<T[K], C>\n : T[K] extends DownloadEndpointDefinition\n ? DownloadHandler<T[K], C>\n : never;\n};\n\n// Error classes\nexport class ValidationError extends Error {\n constructor(\n public field: string,\n public zodError: z.ZodError<unknown>,\n ) {\n const pretty = z.prettifyError(zodError);\n super(`Validation failed for ${field}:\\n${pretty}`);\n this.name = 'ValidationError';\n }\n}\n\nexport class RouteNotFoundError extends Error {\n constructor(\n public path: string,\n public method: string,\n ) {\n super(`Route not found: ${method} ${path}`);\n this.name = 'RouteNotFoundError';\n }\n}\n\n/**\n * Parse and validate request data\n */\nasync function parseRequest<T extends StandardEndpointDefinition, C = unknown>(\n request: Request,\n endpoint: T,\n pathParams: Record<string, string>,\n context: C,\n): Promise<HandlerInput<T, C>> {\n const url = new URL(request.url);\n\n // Parse path params\n let params: any = pathParams;\n if (endpoint.params) {\n const result = endpoint.params.safeParse(pathParams);\n if (!result.success) {\n throw new ValidationError('params', result.error);\n }\n params = result.data;\n }\n\n // Parse query params\n let query: any = {};\n if (endpoint.query) {\n const queryData = parseQuery(url.searchParams);\n const result = endpoint.query.safeParse(queryData);\n if (!result.success) {\n throw new ValidationError('query', result.error);\n }\n query = result.data;\n }\n\n // Parse headers\n let headers: any = {};\n if (endpoint.headers) {\n const headersObj: Record<string, string> = {};\n request.headers.forEach((value, key) => {\n headersObj[key] = value;\n });\n const result = endpoint.headers.safeParse(headersObj);\n if (!result.success) {\n throw new ValidationError('headers', result.error);\n }\n headers = result.data;\n }\n\n // Parse body\n let body: any;\n if (endpoint.body) {\n const contentType = request.headers.get('content-type') || '';\n let bodyData: any;\n\n if (contentType.includes('application/json')) {\n bodyData = await request.json();\n } else if (contentType.includes('multipart/form-data')) {\n const formData = await request.formData();\n bodyData = formDataToObject(formData as FormData);\n } else {\n bodyData = await request.text();\n }\n\n const result = endpoint.body.safeParse(bodyData);\n if (!result.success) {\n throw new ValidationError('body', result.error);\n }\n body = result.data;\n }\n\n return { params, query, headers, body, request, context } as HandlerInput<T, C>;\n}\n\n/**\n * Parse and validate request data for streaming endpoints\n */\nasync function parseStreamingRequest<T extends StreamingEndpointDefinition, C = unknown>(\n request: Request,\n endpoint: T,\n pathParams: Record<string, string>,\n context: C,\n): Promise<Omit<StreamingHandlerInput<T, C>, 'stream'>> {\n const url = new URL(request.url);\n\n // Parse path params\n let params: any = pathParams;\n if (endpoint.params) {\n const result = endpoint.params.safeParse(pathParams);\n if (!result.success) {\n throw new ValidationError('params', result.error);\n }\n params = result.data;\n }\n\n // Parse query params\n let query: any = {};\n if (endpoint.query) {\n const queryData = parseQuery(url.searchParams);\n const result = endpoint.query.safeParse(queryData);\n if (!result.success) {\n throw new ValidationError('query', result.error);\n }\n query = result.data;\n }\n\n // Parse headers\n let headers: any = {};\n if (endpoint.headers) {\n const headersObj: Record<string, string> = {};\n request.headers.forEach((value, key) => {\n headersObj[key] = value;\n });\n const result = endpoint.headers.safeParse(headersObj);\n if (!result.success) {\n throw new ValidationError('headers', result.error);\n }\n headers = result.data;\n }\n\n // Parse body\n let body: any;\n if (endpoint.body) {\n const contentType = request.headers.get('content-type') || '';\n let bodyData: any;\n\n if (contentType.includes('application/json')) {\n bodyData = await request.json();\n } else if (contentType.includes('multipart/form-data')) {\n const formData = await request.formData();\n bodyData = formDataToObject(formData as FormData);\n } else {\n bodyData = await request.text();\n }\n\n const result = endpoint.body.safeParse(bodyData);\n if (!result.success) {\n throw new ValidationError('body', result.error);\n }\n body = result.data;\n }\n\n return { params, query, headers, body, request, context } as Omit<\n StreamingHandlerInput<T, C>,\n 'stream'\n >;\n}\n\n/**\n * Parse and validate request data for SSE endpoints\n */\nasync function parseSSERequest<T extends SSEEndpointDefinition, C = unknown>(\n request: Request,\n endpoint: T,\n pathParams: Record<string, string>,\n context: C,\n): Promise<Omit<SSEHandlerInput<T, C>, 'emitter' | 'signal'>> {\n const url = new URL(request.url);\n\n // Parse path params\n let params: any = pathParams;\n if (endpoint.params) {\n const result = endpoint.params.safeParse(pathParams);\n if (!result.success) {\n throw new ValidationError('params', result.error);\n }\n params = result.data;\n }\n\n // Parse query params\n let query: any = {};\n if (endpoint.query) {\n const queryData = parseQuery(url.searchParams);\n const result = endpoint.query.safeParse(queryData);\n if (!result.success) {\n throw new ValidationError('query', result.error);\n }\n query = result.data;\n }\n\n // Parse headers\n let headers: any = {};\n if (endpoint.headers) {\n const headersObj: Record<string, string> = {};\n request.headers.forEach((value, key) => {\n headersObj[key] = value;\n });\n const result = endpoint.headers.safeParse(headersObj);\n if (!result.success) {\n throw new ValidationError('headers', result.error);\n }\n headers = result.data;\n }\n\n // SSE endpoints don't have a body (GET only)\n return { params, query, headers, request, context } as Omit<\n SSEHandlerInput<T, C>,\n 'emitter' | 'signal'\n >;\n}\n\n/**\n * Parse and validate request data for download endpoints\n */\nasync function parseDownloadRequest<T extends DownloadEndpointDefinition, C = unknown>(\n request: Request,\n endpoint: T,\n pathParams: Record<string, string>,\n context: C,\n): Promise<DownloadHandlerInput<T, C>> {\n const url = new URL(request.url);\n\n // Parse path params\n let params: any = pathParams;\n if (endpoint.params) {\n const result = endpoint.params.safeParse(pathParams);\n if (!result.success) {\n throw new ValidationError('params', result.error);\n }\n params = result.data;\n }\n\n // Parse query params\n let query: any = {};\n if (endpoint.query) {\n const queryData = parseQuery(url.searchParams);\n const result = endpoint.query.safeParse(queryData);\n if (!result.success) {\n throw new ValidationError('query', result.error);\n }\n query = result.data;\n }\n\n // Parse headers\n let headers: any = {};\n if (endpoint.headers) {\n const headersObj: Record<string, string> = {};\n request.headers.forEach((value, key) => {\n headersObj[key] = value;\n });\n const result = endpoint.headers.safeParse(headersObj);\n if (!result.success) {\n throw new ValidationError('headers', result.error);\n }\n headers = result.data;\n }\n\n // Download endpoints don't have a body (GET only)\n return { params, query, headers, request, context } as DownloadHandlerInput<T, C>;\n}\n\n/**\n * Validate and create response\n */\nfunction createResponse<T extends StandardEndpointDefinition>(\n endpoint: T,\n handlerResponse: HandlerResponse<T>,\n): Response {\n const { status, body, headers: customHeaders } = handlerResponse;\n\n // Validate response body\n const responseSchema = endpoint.responses[status as keyof typeof endpoint.responses]\n ?? endpoint.errorResponses?.[status as number];\n if (responseSchema) {\n const result = responseSchema.safeParse(body);\n if (!result.success) {\n throw new ValidationError(`response[${String(status)}]`, result.error);\n }\n }\n\n // Create response headers\n const responseHeaders = new Headers(customHeaders);\n\n // Handle 204 No Content - must have no body\n if (status === 204) {\n return new Response(null, {\n status: 204,\n headers: responseHeaders,\n });\n }\n\n // For all other responses, return JSON\n if (!responseHeaders.has('content-type')) {\n responseHeaders.set('content-type', 'application/json');\n }\n\n return new Response(JSON.stringify(body), {\n status: status as number,\n headers: responseHeaders,\n });\n}\n\n/**\n * Create response for download endpoints\n */\nfunction createDownloadResponse<T extends DownloadEndpointDefinition>(\n _endpoint: T,\n handlerResponse: DownloadHandlerResponse<T>,\n): Response {\n const { status, body, headers: customHeaders } = handlerResponse;\n const responseHeaders = new Headers(customHeaders);\n\n // Success: return File as binary\n if (status === 200 && body instanceof File) {\n if (!responseHeaders.has('content-type')) {\n responseHeaders.set('content-type', body.type || 'application/octet-stream');\n }\n responseHeaders.set('content-length', String(body.size));\n if (!responseHeaders.has('content-disposition')) {\n const filename = encodeURIComponent(body.name);\n responseHeaders.set(\n 'content-disposition',\n `attachment; filename=\"${filename}\"; filename*=UTF-8''${filename}`,\n );\n }\n return new Response(body, { status: 200, headers: responseHeaders });\n }\n\n // Error: return JSON\n if (!responseHeaders.has('content-type')) {\n responseHeaders.set('content-type', 'application/json');\n }\n return new Response(JSON.stringify(body), {\n status: status as number,\n headers: responseHeaders,\n });\n}\n\n/**\n * Create a streaming NDJSON response\n */\nfunction createStreamingResponse<T extends StreamingEndpointDefinition>(\n handler: (stream: StreamEmitter<T>) => void | Promise<void>,\n): Response {\n const { readable, writable } = new TransformStream();\n const writer = writable.getWriter();\n const encoder = new TextEncoder();\n let closed = false;\n\n const stream: StreamEmitter<T> = {\n send(chunk) {\n if (!closed) {\n writer.write(encoder.encode(`${JSON.stringify(chunk)}\\n`)).catch(() => {\n // Stream was closed by client disconnect - ignore\n closed = true;\n });\n }\n },\n close(final) {\n if (!closed) {\n closed = true;\n const closeWriter = () => {\n writer.close().catch(() => {\n // Stream was already closed or errored - ignore\n });\n };\n if (final !== undefined) {\n writer\n .write(encoder.encode(`${JSON.stringify({ __final__: true, data: final })}\\n`))\n .catch(() => {\n // Stream was closed by client disconnect - ignore\n })\n .finally(closeWriter);\n } else {\n closeWriter();\n }\n }\n },\n get isOpen() {\n return !closed;\n },\n };\n\n // Execute handler (don't await - let it run in background)\n Promise.resolve(handler(stream)).catch((err) => {\n console.error('Streaming handler error:', err);\n stream.close();\n });\n\n return new Response(readable, {\n headers: {\n 'Content-Type': 'application/x-ndjson',\n 'Cache-Control': 'no-cache',\n Connection: 'keep-alive',\n },\n });\n}\n\n/**\n * Create an SSE response\n */\nfunction createSSEResponse<T extends SSEEndpointDefinition>(\n handler: (\n emitter: SSEEmitter<T>,\n signal: AbortSignal,\n ) => void | (() => void) | Promise<void | (() => void)>,\n): Response {\n const { readable, writable } = new TransformStream();\n const writer = writable.getWriter();\n const encoder = new TextEncoder();\n const controller = new AbortController();\n let closed = false;\n let cleanup: (() => void) | undefined;\n\n const emitter: SSEEmitter<T> = {\n send(event, data, options) {\n if (!closed) {\n let message = '';\n if (options?.id) {\n message += `id: ${options.id}\\n`;\n }\n message += `event: ${String(event)}\\n`;\n message += `data: ${JSON.stringify(data)}\\n\\n`;\n writer.write(encoder.encode(message)).catch(() => {\n // Stream was closed by client disconnect - ignore\n closed = true;\n });\n }\n },\n close() {\n if (!closed) {\n closed = true;\n if (cleanup) cleanup();\n writer.close().catch(() => {\n // Stream was already closed or errored - ignore\n });\n }\n },\n get isOpen() {\n return !closed;\n },\n };\n\n // Execute handler and get cleanup function\n Promise.resolve(handler(emitter, controller.signal))\n .then((cleanupFn) => {\n if (typeof cleanupFn === 'function') {\n cleanup = cleanupFn;\n }\n })\n .catch((err) => {\n console.error('SSE handler error:', err);\n emitter.close();\n });\n\n // Tee the stream - one for the response, one for disconnect detection\n const [responseStream, detectStream] = readable.tee();\n\n // Handle client disconnect by detecting when the stream is cancelled\n detectStream.pipeTo(new WritableStream()).catch(() => {\n controller.abort();\n emitter.close();\n });\n\n return new Response(responseStream, {\n headers: {\n 'Content-Type': 'text/event-stream',\n 'Cache-Control': 'no-cache',\n Connection: 'keep-alive',\n },\n });\n}\n\n/**\n * Router configuration options\n */\nexport interface RouterOptions<C = unknown> {\n basePath?: string;\n context?: (request: Request, routeName?: string, endpoint?: EndpointDefinition) => C | Promise<C>;\n}\n\n/**\n * Router class that manages contract endpoints\n */\nexport class Router<T extends Contract, C = unknown> {\n private basePath: string;\n private contextFactory?: (\n request: Request,\n routeName: string,\n endpoint: EndpointDefinition,\n ) => C | Promise<C>;\n\n constructor(\n private contract: T,\n private handlers: ContractHandlers<T, C>,\n options?: RouterOptions<C>,\n ) {\n // Validate all path patterns\n for (const [name, endpoint] of Object.entries(contract)) {\n try {\n validatePathPattern(endpoint.path);\n } catch (error) {\n throw new Error(`Invalid path in endpoint \"${name}\": ${(error as Error).message}`);\n }\n }\n\n // Normalize basePath: ensure it starts with / and doesn't end with /\n const bp = options?.basePath || '';\n if (bp) {\n this.basePath = bp.startsWith('/') ? bp : `/${bp}`;\n this.basePath = this.basePath.endsWith('/') ? this.basePath.slice(0, -1) : this.basePath;\n } else {\n this.basePath = '';\n }\n this.contextFactory = options?.context;\n }\n\n /**\n * Find matching endpoint for a request\n */\n private findEndpoint(\n method: string,\n path: string,\n ): {\n name: keyof T;\n endpoint: EndpointDefinition;\n params: Record<string, string>;\n } | null {\n for (const [name, endpoint] of Object.entries(this.contract)) {\n if (endpoint.method === method) {\n const params = matchPath(endpoint.path, path);\n if (params !== null) {\n return { name, endpoint, params };\n }\n }\n }\n return null;\n }\n\n /**\n * Handle a request\n */\n async handle(request: Request): Promise<Response> {\n const url = new URL(request.url);\n const method = request.method;\n let path = url.pathname;\n\n // Strip basePath if configured\n if (this.basePath && path.startsWith(this.basePath)) {\n path = path.slice(this.basePath.length) || '/';\n }\n\n const match = this.findEndpoint(method, path);\n if (!match) {\n throw new RouteNotFoundError(path, method);\n }\n\n const { name, endpoint, params } = match;\n const handler = this.handlers[name];\n\n // Create context if factory is provided\n const context = this.contextFactory\n ? await this.contextFactory(request, String(name), endpoint)\n : (undefined as C);\n\n // Dispatch based on endpoint type\n if (endpoint.type === 'streaming') {\n // Parse request for streaming endpoint\n const input = await parseStreamingRequest(request, endpoint, params, context);\n return createStreamingResponse((stream) => {\n return (handler as StreamingHandler<StreamingEndpointDefinition, C>)({\n ...input,\n stream,\n });\n });\n }\n\n if (endpoint.type === 'sse') {\n // Parse request for SSE endpoint\n const input = await parseSSERequest(request, endpoint, params, context);\n return createSSEResponse((emitter, signal) => {\n return (handler as SSEHandler<SSEEndpointDefinition, C>)({\n ...input,\n emitter,\n signal,\n });\n });\n }\n\n if (endpoint.type === 'download') {\n // Parse request for download endpoint\n const input = await parseDownloadRequest(request, endpoint, params, context);\n const downloadHandler = handler as unknown as DownloadHandler<DownloadEndpointDefinition, C>;\n const response = await downloadHandler(\n input as DownloadHandlerInput<DownloadEndpointDefinition, C>,\n );\n return createDownloadResponse(\n endpoint as DownloadEndpointDefinition,\n response as DownloadHandlerResponse<DownloadEndpointDefinition>,\n );\n }\n\n // Standard endpoint\n const input = await parseRequest(request, endpoint, params, context);\n const standardHandler = handler as unknown as Handler<StandardEndpointDefinition, C>;\n const handlerResponse = await standardHandler(\n input as HandlerInput<StandardEndpointDefinition, C>,\n );\n return createResponse(\n endpoint as StandardEndpointDefinition,\n handlerResponse as HandlerResponse<StandardEndpointDefinition>,\n );\n }\n\n /**\n * Get fetch handler compatible with Bun.serve\n */\n get fetch() {\n return (request: Request) => this.handle(request);\n }\n}\n\n/**\n * Create a router from a contract and handlers\n */\nexport function createRouter<T extends Contract, C = unknown>(\n contract: T,\n handlers: ContractHandlers<T, C>,\n options?: RouterOptions<C>,\n): Router<T, C> {\n return new Router(contract, handlers, options);\n}\n\nexport * from './websocket.cjs';\n"
|
|
6
6
|
],
|
|
7
|
-
"mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAqBO,IANP;AAOkB,IAAlB;AAozBA;AAjoBO,MAAM,wBAAwB,MAAM;AAAA,EAEhC;AAAA,EACA;AAAA,EAFT,WAAW,CACF,OACA,UACP;AAAA,IACA,MAAM,SAAS,aAAE,cAAc,QAAQ;AAAA,IACvC,MAAM,yBAAyB;AAAA,EAAW,QAAQ;AAAA,IAJ3C;AAAA,IACA;AAAA,IAIP,KAAK,OAAO;AAAA;AAEhB;AAAA;AAEO,MAAM,2BAA2B,MAAM;AAAA,EAEnC;AAAA,EACA;AAAA,EAFT,WAAW,CACF,MACA,QACP;AAAA,IACA,MAAM,oBAAoB,UAAU,MAAM;AAAA,IAHnC;AAAA,IACA;AAAA,IAGP,KAAK,OAAO;AAAA;AAEhB;AAKA,eAAe,YAA+D,CAC5E,SACA,UACA,YACA,SAC6B;AAAA,EAC7B,MAAM,MAAM,IAAI,IAAI,QAAQ,GAAG;AAAA,EAG/B,IAAI,SAAc;AAAA,EAClB,IAAI,SAAS,QAAQ;AAAA,IACnB,MAAM,SAAS,SAAS,OAAO,UAAU,UAAU;AAAA,IACnD,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,UAAU,OAAO,KAAK;AAAA,IAClD;AAAA,IACA,SAAS,OAAO;AAAA,EAClB;AAAA,EAGA,IAAI,QAAa,CAAC;AAAA,EAClB,IAAI,SAAS,OAAO;AAAA,IAClB,MAAM,YAAY,uBAAW,IAAI,YAAY;AAAA,IAC7C,MAAM,SAAS,SAAS,MAAM,UAAU,SAAS;AAAA,IACjD,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,SAAS,OAAO,KAAK;AAAA,IACjD;AAAA,IACA,QAAQ,OAAO;AAAA,EACjB;AAAA,EAGA,IAAI,UAAe,CAAC;AAAA,EACpB,IAAI,SAAS,SAAS;AAAA,IACpB,MAAM,aAAqC,CAAC;AAAA,IAC5C,QAAQ,QAAQ,QAAQ,CAAC,OAAO,QAAQ;AAAA,MACtC,WAAW,OAAO;AAAA,KACnB;AAAA,IACD,MAAM,SAAS,SAAS,QAAQ,UAAU,UAAU;AAAA,IACpD,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,WAAW,OAAO,KAAK;AAAA,IACnD;AAAA,IACA,UAAU,OAAO;AAAA,EACnB;AAAA,EAGA,IAAI;AAAA,EACJ,IAAI,SAAS,MAAM;AAAA,IACjB,MAAM,cAAc,QAAQ,QAAQ,IAAI,cAAc,KAAK;AAAA,IAC3D,IAAI;AAAA,IAEJ,IAAI,YAAY,SAAS,kBAAkB,GAAG;AAAA,MAC5C,WAAW,MAAM,QAAQ,KAAK;AAAA,IAChC,EAAO,SAAI,YAAY,SAAS,qBAAqB,GAAG;AAAA,MACtD,MAAM,WAAW,MAAM,QAAQ,SAAS;AAAA,MACxC,WAAW,6BAAiB,QAAoB;AAAA,IAClD,EAAO;AAAA,MACL,WAAW,MAAM,QAAQ,KAAK;AAAA;AAAA,IAGhC,MAAM,SAAS,SAAS,KAAK,UAAU,QAAQ;AAAA,IAC/C,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,QAAQ,OAAO,KAAK;AAAA,IAChD;AAAA,IACA,OAAO,OAAO;AAAA,EAChB;AAAA,EAEA,OAAO,EAAE,QAAQ,OAAO,SAAS,MAAM,SAAS,QAAQ;AAAA;AAM1D,eAAe,qBAAyE,CACtF,SACA,UACA,YACA,SACsD;AAAA,EACtD,MAAM,MAAM,IAAI,IAAI,QAAQ,GAAG;AAAA,EAG/B,IAAI,SAAc;AAAA,EAClB,IAAI,SAAS,QAAQ;AAAA,IACnB,MAAM,SAAS,SAAS,OAAO,UAAU,UAAU;AAAA,IACnD,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,UAAU,OAAO,KAAK;AAAA,IAClD;AAAA,IACA,SAAS,OAAO;AAAA,EAClB;AAAA,EAGA,IAAI,QAAa,CAAC;AAAA,EAClB,IAAI,SAAS,OAAO;AAAA,IAClB,MAAM,YAAY,uBAAW,IAAI,YAAY;AAAA,IAC7C,MAAM,SAAS,SAAS,MAAM,UAAU,SAAS;AAAA,IACjD,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,SAAS,OAAO,KAAK;AAAA,IACjD;AAAA,IACA,QAAQ,OAAO;AAAA,EACjB;AAAA,EAGA,IAAI,UAAe,CAAC;AAAA,EACpB,IAAI,SAAS,SAAS;AAAA,IACpB,MAAM,aAAqC,CAAC;AAAA,IAC5C,QAAQ,QAAQ,QAAQ,CAAC,OAAO,QAAQ;AAAA,MACtC,WAAW,OAAO;AAAA,KACnB;AAAA,IACD,MAAM,SAAS,SAAS,QAAQ,UAAU,UAAU;AAAA,IACpD,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,WAAW,OAAO,KAAK;AAAA,IACnD;AAAA,IACA,UAAU,OAAO;AAAA,EACnB;AAAA,EAGA,IAAI;AAAA,EACJ,IAAI,SAAS,MAAM;AAAA,IACjB,MAAM,cAAc,QAAQ,QAAQ,IAAI,cAAc,KAAK;AAAA,IAC3D,IAAI;AAAA,IAEJ,IAAI,YAAY,SAAS,kBAAkB,GAAG;AAAA,MAC5C,WAAW,MAAM,QAAQ,KAAK;AAAA,IAChC,EAAO,SAAI,YAAY,SAAS,qBAAqB,GAAG;AAAA,MACtD,MAAM,WAAW,MAAM,QAAQ,SAAS;AAAA,MACxC,WAAW,6BAAiB,QAAoB;AAAA,IAClD,EAAO;AAAA,MACL,WAAW,MAAM,QAAQ,KAAK;AAAA;AAAA,IAGhC,MAAM,SAAS,SAAS,KAAK,UAAU,QAAQ;AAAA,IAC/C,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,QAAQ,OAAO,KAAK;AAAA,IAChD;AAAA,IACA,OAAO,OAAO;AAAA,EAChB;AAAA,EAEA,OAAO,EAAE,QAAQ,OAAO,SAAS,MAAM,SAAS,QAAQ;AAAA;AAS1D,eAAe,eAA6D,CAC1E,SACA,UACA,YACA,SAC4D;AAAA,EAC5D,MAAM,MAAM,IAAI,IAAI,QAAQ,GAAG;AAAA,EAG/B,IAAI,SAAc;AAAA,EAClB,IAAI,SAAS,QAAQ;AAAA,IACnB,MAAM,SAAS,SAAS,OAAO,UAAU,UAAU;AAAA,IACnD,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,UAAU,OAAO,KAAK;AAAA,IAClD;AAAA,IACA,SAAS,OAAO;AAAA,EAClB;AAAA,EAGA,IAAI,QAAa,CAAC;AAAA,EAClB,IAAI,SAAS,OAAO;AAAA,IAClB,MAAM,YAAY,uBAAW,IAAI,YAAY;AAAA,IAC7C,MAAM,SAAS,SAAS,MAAM,UAAU,SAAS;AAAA,IACjD,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,SAAS,OAAO,KAAK;AAAA,IACjD;AAAA,IACA,QAAQ,OAAO;AAAA,EACjB;AAAA,EAGA,IAAI,UAAe,CAAC;AAAA,EACpB,IAAI,SAAS,SAAS;AAAA,IACpB,MAAM,aAAqC,CAAC;AAAA,IAC5C,QAAQ,QAAQ,QAAQ,CAAC,OAAO,QAAQ;AAAA,MACtC,WAAW,OAAO;AAAA,KACnB;AAAA,IACD,MAAM,SAAS,SAAS,QAAQ,UAAU,UAAU;AAAA,IACpD,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,WAAW,OAAO,KAAK;AAAA,IACnD;AAAA,IACA,UAAU,OAAO;AAAA,EACnB;AAAA,EAGA,OAAO,EAAE,QAAQ,OAAO,SAAS,SAAS,QAAQ;AAAA;AASpD,eAAe,oBAAuE,CACpF,SACA,UACA,YACA,SACqC;AAAA,EACrC,MAAM,MAAM,IAAI,IAAI,QAAQ,GAAG;AAAA,EAG/B,IAAI,SAAc;AAAA,EAClB,IAAI,SAAS,QAAQ;AAAA,IACnB,MAAM,SAAS,SAAS,OAAO,UAAU,UAAU;AAAA,IACnD,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,UAAU,OAAO,KAAK;AAAA,IAClD;AAAA,IACA,SAAS,OAAO;AAAA,EAClB;AAAA,EAGA,IAAI,QAAa,CAAC;AAAA,EAClB,IAAI,SAAS,OAAO;AAAA,IAClB,MAAM,YAAY,uBAAW,IAAI,YAAY;AAAA,IAC7C,MAAM,SAAS,SAAS,MAAM,UAAU,SAAS;AAAA,IACjD,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,SAAS,OAAO,KAAK;AAAA,IACjD;AAAA,IACA,QAAQ,OAAO;AAAA,EACjB;AAAA,EAGA,IAAI,UAAe,CAAC;AAAA,EACpB,IAAI,SAAS,SAAS;AAAA,IACpB,MAAM,aAAqC,CAAC;AAAA,IAC5C,QAAQ,QAAQ,QAAQ,CAAC,OAAO,QAAQ;AAAA,MACtC,WAAW,OAAO;AAAA,KACnB;AAAA,IACD,MAAM,SAAS,SAAS,QAAQ,UAAU,UAAU;AAAA,IACpD,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,WAAW,OAAO,KAAK;AAAA,IACnD;AAAA,IACA,UAAU,OAAO;AAAA,EACnB;AAAA,EAGA,OAAO,EAAE,QAAQ,OAAO,SAAS,SAAS,QAAQ;AAAA;AAMpD,SAAS,cAAoD,CAC3D,UACA,iBACU;AAAA,EACV,QAAQ,QAAQ,MAAM,SAAS,kBAAkB;AAAA,EAGjD,MAAM,iBAAiB,SAAS,UAAU,WACrC,SAAS,iBAAiB;AAAA,EAC/B,IAAI,gBAAgB;AAAA,IAClB,MAAM,SAAS,eAAe,UAAU,IAAI;AAAA,IAC5C,IAAI,CAAC,OAAO,SAAS;AAAA,MACnB,MAAM,IAAI,gBAAgB,YAAY,OAAO,MAAM,MAAM,OAAO,KAAK;AAAA,IACvE;AAAA,EACF;AAAA,EAGA,MAAM,kBAAkB,IAAI,QAAQ,aAAa;AAAA,EAGjD,IAAI,WAAW,KAAK;AAAA,IAClB,OAAO,IAAI,SAAS,MAAM;AAAA,MACxB,QAAQ;AAAA,MACR,SAAS;AAAA,IACX,CAAC;AAAA,EACH;AAAA,EAGA,IAAI,CAAC,gBAAgB,IAAI,cAAc,GAAG;AAAA,IACxC,gBAAgB,IAAI,gBAAgB,kBAAkB;AAAA,EACxD;AAAA,EAEA,OAAO,IAAI,SAAS,KAAK,UAAU,IAAI,GAAG;AAAA,IACxC;AAAA,IACA,SAAS;AAAA,EACX,CAAC;AAAA;AAMH,SAAS,sBAA4D,CACnE,WACA,iBACU;AAAA,EACV,QAAQ,QAAQ,MAAM,SAAS,kBAAkB;AAAA,EACjD,MAAM,kBAAkB,IAAI,QAAQ,aAAa;AAAA,EAGjD,IAAI,WAAW,OAAO,gBAAgB,MAAM;AAAA,IAC1C,IAAI,CAAC,gBAAgB,IAAI,cAAc,GAAG;AAAA,MACxC,gBAAgB,IAAI,gBAAgB,KAAK,QAAQ,0BAA0B;AAAA,IAC7E;AAAA,IACA,gBAAgB,IAAI,kBAAkB,OAAO,KAAK,IAAI,CAAC;AAAA,IACvD,IAAI,CAAC,gBAAgB,IAAI,qBAAqB,GAAG;AAAA,MAC/C,MAAM,WAAW,mBAAmB,KAAK,IAAI;AAAA,MAC7C,gBAAgB,IACd,uBACA,yBAAyB,+BAA+B,UAC1D;AAAA,IACF;AAAA,IACA,OAAO,IAAI,SAAS,MAAM,EAAE,QAAQ,KAAK,SAAS,gBAAgB,CAAC;AAAA,EACrE;AAAA,EAGA,IAAI,CAAC,gBAAgB,IAAI,cAAc,GAAG;AAAA,IACxC,gBAAgB,IAAI,gBAAgB,kBAAkB;AAAA,EACxD;AAAA,EACA,OAAO,IAAI,SAAS,KAAK,UAAU,IAAI,GAAG;AAAA,IACxC;AAAA,IACA,SAAS;AAAA,EACX,CAAC;AAAA;AAMH,SAAS,uBAA8D,CACrE,SACU;AAAA,EACV,QAAQ,UAAU,aAAa,IAAI;AAAA,EACnC,MAAM,SAAS,SAAS,UAAU;AAAA,EAClC,MAAM,UAAU,IAAI;AAAA,EACpB,IAAI,SAAS;AAAA,EAEb,MAAM,SAA2B;AAAA,IAC/B,IAAI,CAAC,OAAO;AAAA,MACV,IAAI,CAAC,QAAQ;AAAA,QACX,OAAO,MAAM,QAAQ,OAAO,GAAG,KAAK,UAAU,KAAK;AAAA,CAAK,CAAC,EAAE,MAAM,MAAM;AAAA,UAErE,SAAS;AAAA,SACV;AAAA,MACH;AAAA;AAAA,IAEF,KAAK,CAAC,OAAO;AAAA,MACX,IAAI,CAAC,QAAQ;AAAA,QACX,SAAS;AAAA,QACT,MAAM,cAAc,MAAM;AAAA,UACxB,OAAO,MAAM,EAAE,MAAM,MAAM,EAE1B;AAAA;AAAA,QAEH,IAAI,UAAU,WAAW;AAAA,UACvB,OACG,MAAM,QAAQ,OAAO,GAAG,KAAK,UAAU,EAAE,WAAW,MAAM,MAAM,MAAM,CAAC;AAAA,CAAK,CAAC,EAC7E,MAAM,MAAM,EAEZ,EACA,QAAQ,WAAW;AAAA,QACxB,EAAO;AAAA,UACL,YAAY;AAAA;AAAA,MAEhB;AAAA;AAAA,QAEE,MAAM,GAAG;AAAA,MACX,OAAO,CAAC;AAAA;AAAA,EAEZ;AAAA,EAGA,QAAQ,QAAQ,QAAQ,MAAM,CAAC,EAAE,MAAM,CAAC,QAAQ;AAAA,IAC9C,QAAQ,MAAM,4BAA4B,GAAG;AAAA,IAC7C,OAAO,MAAM;AAAA,GACd;AAAA,EAED,OAAO,IAAI,SAAS,UAAU;AAAA,IAC5B,SAAS;AAAA,MACP,gBAAgB;AAAA,MAChB,iBAAiB;AAAA,MACjB,YAAY;AAAA,IACd;AAAA,EACF,CAAC;AAAA;AAMH,SAAS,iBAAkD,CACzD,SAIU;AAAA,EACV,QAAQ,UAAU,aAAa,IAAI;AAAA,EACnC,MAAM,SAAS,SAAS,UAAU;AAAA,EAClC,MAAM,UAAU,IAAI;AAAA,EACpB,MAAM,aAAa,IAAI;AAAA,EACvB,IAAI,SAAS;AAAA,EACb,IAAI;AAAA,EAEJ,MAAM,UAAyB;AAAA,IAC7B,IAAI,CAAC,OAAO,MAAM,SAAS;AAAA,MACzB,IAAI,CAAC,QAAQ;AAAA,QACX,IAAI,UAAU;AAAA,QACd,IAAI,SAAS,IAAI;AAAA,UACf,WAAW,OAAO,QAAQ;AAAA;AAAA,QAC5B;AAAA,QACA,WAAW,UAAU,OAAO,KAAK;AAAA;AAAA,QACjC,WAAW,SAAS,KAAK,UAAU,IAAI;AAAA;AAAA;AAAA,QACvC,OAAO,MAAM,QAAQ,OAAO,OAAO,CAAC,EAAE,MAAM,MAAM;AAAA,UAEhD,SAAS;AAAA,SACV;AAAA,MACH;AAAA;AAAA,IAEF,KAAK,GAAG;AAAA,MACN,IAAI,CAAC,QAAQ;AAAA,QACX,SAAS;AAAA,QACT,IAAI;AAAA,UAAS,QAAQ;AAAA,QACrB,OAAO,MAAM,EAAE,MAAM,MAAM,EAE1B;AAAA,MACH;AAAA;AAAA,QAEE,MAAM,GAAG;AAAA,MACX,OAAO,CAAC;AAAA;AAAA,EAEZ;AAAA,EAGA,QAAQ,QAAQ,QAAQ,SAAS,WAAW,MAAM,CAAC,EAChD,KAAK,CAAC,cAAc;AAAA,IACnB,IAAI,OAAO,cAAc,YAAY;AAAA,MACnC,UAAU;AAAA,IACZ;AAAA,GACD,EACA,MAAM,CAAC,QAAQ;AAAA,IACd,QAAQ,MAAM,sBAAsB,GAAG;AAAA,IACvC,QAAQ,MAAM;AAAA,GACf;AAAA,EAGH,OAAO,gBAAgB,gBAAgB,SAAS,IAAI;AAAA,EAGpD,aAAa,OAAO,IAAI,cAAgB,EAAE,MAAM,MAAM;AAAA,IACpD,WAAW,MAAM;AAAA,IACjB,QAAQ,MAAM;AAAA,GACf;AAAA,EAED,OAAO,IAAI,SAAS,gBAAgB;AAAA,IAClC,SAAS;AAAA,MACP,gBAAgB;AAAA,MAChB,iBAAiB;AAAA,MACjB,YAAY;AAAA,IACd;AAAA,EACF,CAAC;AAAA;AAAA;AAcI,MAAM,OAAwC;AAAA,EASzC;AAAA,EACA;AAAA,EATF;AAAA,EACA;AAAA,EAMR,WAAW,CACD,UACA,UACR,SACA;AAAA,IAHQ;AAAA,IACA;AAAA,IAIR,YAAY,MAAM,aAAa,OAAO,QAAQ,QAAQ,GAAG;AAAA,MACvD,IAAI;AAAA,QACF,gCAAoB,SAAS,IAAI;AAAA,QACjC,OAAO,OAAO;AAAA,QACd,MAAM,IAAI,MAAM,6BAA6B,UAAW,MAAgB,SAAS;AAAA;AAAA,IAErF;AAAA,IAGA,MAAM,KAAK,SAAS,YAAY;AAAA,IAChC,IAAI,IAAI;AAAA,MACN,KAAK,WAAW,GAAG,WAAW,GAAG,IAAI,KAAK,IAAI;AAAA,MAC9C,KAAK,WAAW,KAAK,SAAS,SAAS,GAAG,IAAI,KAAK,SAAS,MAAM,GAAG,EAAE,IAAI,KAAK;AAAA,IAClF,EAAO;AAAA,MACL,KAAK,WAAW;AAAA;AAAA,IAElB,KAAK,iBAAiB,SAAS;AAAA;AAAA,EAMzB,YAAY,CAClB,QACA,MAKO;AAAA,IACP,YAAY,MAAM,aAAa,OAAO,QAAQ,KAAK,QAAQ,GAAG;AAAA,MAC5D,IAAI,SAAS,WAAW,QAAQ;AAAA,QAC9B,MAAM,SAAS,sBAAU,SAAS,MAAM,IAAI;AAAA,QAC5C,IAAI,WAAW,MAAM;AAAA,UACnB,OAAO,EAAE,MAAM,UAAU,OAAO;AAAA,QAClC;AAAA,MACF;AAAA,IACF;AAAA,IACA,OAAO;AAAA;AAAA,OAMH,OAAM,CAAC,SAAqC;AAAA,IAChD,MAAM,MAAM,IAAI,IAAI,QAAQ,GAAG;AAAA,IAC/B,MAAM,SAAS,QAAQ;AAAA,IACvB,IAAI,OAAO,IAAI;AAAA,IAGf,IAAI,KAAK,YAAY,KAAK,WAAW,KAAK,QAAQ,GAAG;AAAA,MACnD,OAAO,KAAK,MAAM,KAAK,SAAS,MAAM,KAAK;AAAA,IAC7C;AAAA,IAEA,MAAM,QAAQ,KAAK,aAAa,QAAQ,IAAI;AAAA,IAC5C,IAAI,CAAC,OAAO;AAAA,MACV,MAAM,IAAI,mBAAmB,MAAM,MAAM;AAAA,IAC3C;AAAA,IAEA,QAAQ,MAAM,UAAU,WAAW;AAAA,IACnC,MAAM,UAAU,KAAK,SAAS;AAAA,IAG9B,MAAM,UAAU,KAAK,iBACjB,MAAM,KAAK,eAAe,SAAS,OAAO,IAAI,GAAG,QAAQ,IACxD;AAAA,IAGL,IAAI,SAAS,SAAS,aAAa;AAAA,MAEjC,MAAM,SAAQ,MAAM,sBAAsB,SAAS,UAAU,QAAQ,OAAO;AAAA,MAC5E,OAAO,wBAAwB,CAAC,WAAW;AAAA,QACzC,OAAQ,QAA6D;AAAA,aAChE;AAAA,UACH;AAAA,QACF,CAAC;AAAA,OACF;AAAA,IACH;AAAA,IAEA,IAAI,SAAS,SAAS,OAAO;AAAA,MAE3B,MAAM,SAAQ,MAAM,gBAAgB,SAAS,UAAU,QAAQ,OAAO;AAAA,MACtE,OAAO,kBAAkB,CAAC,SAAS,WAAW;AAAA,QAC5C,OAAQ,QAAiD;AAAA,aACpD;AAAA,UACH;AAAA,UACA;AAAA,QACF,CAAC;AAAA,OACF;AAAA,IACH;AAAA,IAEA,IAAI,SAAS,SAAS,YAAY;AAAA,MAEhC,MAAM,SAAQ,MAAM,qBAAqB,SAAS,UAAU,QAAQ,OAAO;AAAA,MAC3E,MAAM,kBAAkB;AAAA,MACxB,MAAM,WAAW,MAAM,gBACrB,MACF;AAAA,MACA,OAAO,uBACL,UACA,QACF;AAAA,IACF;AAAA,IAGA,MAAM,QAAQ,MAAM,aAAa,SAAS,UAAU,QAAQ,OAAO;AAAA,IACnE,MAAM,kBAAkB;AAAA,IACxB,MAAM,kBAAkB,MAAM,gBAC5B,KACF;AAAA,IACA,OAAO,eACL,UACA,eACF;AAAA;AAAA,MAME,KAAK,GAAG;AAAA,IACV,OAAO,CAAC,YAAqB,KAAK,OAAO,OAAO;AAAA;AAEpD;AAKO,SAAS,YAA6C,CAC3D,UACA,UACA,SACc;AAAA,EACd,OAAO,IAAI,OAAO,UAAU,UAAU,OAAO;AAAA;",
|
|
8
|
-
"debugId": "
|
|
7
|
+
"mappings": "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",
|
|
8
|
+
"debugId": "EE273BF08A26AE9264756E2164756E21",
|
|
9
9
|
"names": []
|
|
10
10
|
}
|
package/dist/cjs/package.json
CHANGED
package/dist/cjs/websocket.cjs
CHANGED
|
@@ -3,27 +3,37 @@
|
|
|
3
3
|
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
4
4
|
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
5
|
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
-
|
|
6
|
+
function __accessProp(key) {
|
|
7
|
+
return this[key];
|
|
8
|
+
}
|
|
7
9
|
var __toCommonJS = (from) => {
|
|
8
|
-
var entry = __moduleCache.get(from), desc;
|
|
10
|
+
var entry = (__moduleCache ??= new WeakMap).get(from), desc;
|
|
9
11
|
if (entry)
|
|
10
12
|
return entry;
|
|
11
13
|
entry = __defProp({}, "__esModule", { value: true });
|
|
12
|
-
if (from && typeof from === "object" || typeof from === "function")
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
14
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
15
|
+
for (var key of __getOwnPropNames(from))
|
|
16
|
+
if (!__hasOwnProp.call(entry, key))
|
|
17
|
+
__defProp(entry, key, {
|
|
18
|
+
get: __accessProp.bind(from, key),
|
|
19
|
+
enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
|
|
20
|
+
});
|
|
21
|
+
}
|
|
17
22
|
__moduleCache.set(from, entry);
|
|
18
23
|
return entry;
|
|
19
24
|
};
|
|
25
|
+
var __moduleCache;
|
|
26
|
+
var __returnValue = (v) => v;
|
|
27
|
+
function __exportSetter(name, newValue) {
|
|
28
|
+
this[name] = __returnValue.bind(null, newValue);
|
|
29
|
+
}
|
|
20
30
|
var __export = (target, all) => {
|
|
21
31
|
for (var name in all)
|
|
22
32
|
__defProp(target, name, {
|
|
23
33
|
get: all[name],
|
|
24
34
|
enumerable: true,
|
|
25
35
|
configurable: true,
|
|
26
|
-
set: (
|
|
36
|
+
set: __exportSetter.bind(all, name)
|
|
27
37
|
});
|
|
28
38
|
};
|
|
29
39
|
|
|
@@ -207,7 +217,7 @@ class WebSocketRouter {
|
|
|
207
217
|
const validatedData = this.validateData(data);
|
|
208
218
|
const typedWs = createTypedWebSocket(ws);
|
|
209
219
|
try {
|
|
210
|
-
const validatedMessage = this.validateMessage(upgradeData.endpoint,
|
|
220
|
+
const validatedMessage = this.validateMessage(upgradeData.endpoint, rawMessage);
|
|
211
221
|
await endpointHandlers.message({
|
|
212
222
|
ws: typedWs,
|
|
213
223
|
message: validatedMessage,
|
|
@@ -276,4 +286,4 @@ function createWebSocketRouter(contract, handlers, options) {
|
|
|
276
286
|
}
|
|
277
287
|
})
|
|
278
288
|
|
|
279
|
-
//# debugId=
|
|
289
|
+
//# debugId=CD0572859FC97E1C64756E2164756E21
|
|
@@ -2,9 +2,9 @@
|
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../websocket.ts"],
|
|
4
4
|
"sourcesContent": [
|
|
5
|
-
"import type {\n ExtractClientMessage,\n ExtractWSHeaders,\n ExtractWSParams,\n ExtractWSQuery,\n WebSocketContract,\n WebSocketContractDefinition,\n} from '@richie-rpc/core';\nimport { matchPath, parseQuery } from '@richie-rpc/core';\nimport { z } from 'zod';\n\n/**\n * Validation error for WebSocket messages\n */\nexport class WebSocketValidationError extends Error {\n constructor(\n public messageType: string,\n public zodError: z.ZodError<unknown>,\n ) {\n const pretty = z.prettifyError(zodError);\n super(`Validation failed for WebSocket message type: ${messageType}:\\n${pretty}`);\n this.name = 'WebSocketValidationError';\n }\n}\n\n/**\n * Typed WebSocket wrapper for sending messages\n */\nexport interface TypedServerWebSocket<\n T extends WebSocketContractDefinition,\n WS extends GenericWebSocket,\n> {\n /** The underlying Bun WebSocket */\n readonly raw: WS;\n /** Send a typed message to the client */\n send<K extends keyof T['serverMessages']>(\n type: K,\n payload: z.infer<T['serverMessages'][K]['payload']>,\n ): void;\n /** Close the connection */\n close(code?: number, reason?: string): void;\n}\n\nexport type GenericWebSocket = {\n send: (message: string) => void;\n close: (code?: number, reason?: string) => void;\n};\n\n/**\n * Arguments passed to handler functions\n */\nexport interface HandlerArgs<\n T extends WebSocketContractDefinition,\n WS extends GenericWebSocket,\n D = unknown,\n> {\n ws: TypedServerWebSocket<T, WS>;\n params: ExtractWSParams<T>;\n query: ExtractWSQuery<T>;\n headers: ExtractWSHeaders<T>;\n data: D;\n}\n\n/**\n * Handler functions for a WebSocket endpoint\n */\nexport interface WebSocketEndpointHandlers<\n T extends WebSocketContractDefinition,\n WS extends GenericWebSocket,\n D = unknown,\n> {\n /** Called when connection opens */\n open?(args: HandlerArgs<T, WS, D>): void | Promise<void>;\n /** Called for each validated message */\n message(args: HandlerArgs<T, WS, D> & { message: ExtractClientMessage<T> }): void | Promise<void>;\n /** Called when connection closes */\n close?(args: HandlerArgs<T, WS, D>): void;\n /** Called on backpressure drain */\n drain?(args: HandlerArgs<T, WS, D>): void;\n /** Called when message validation fails */\n validationError?(args: {\n ws: TypedServerWebSocket<T, WS>;\n error: WebSocketValidationError;\n data: D;\n }): void;\n}\n\n/**\n * Contract handlers mapping for WebSocket endpoints\n */\nexport type WebSocketContractHandlers<\n T extends WebSocketContract,\n WS extends GenericWebSocket,\n D = unknown,\n> = {\n [K in keyof T]: WebSocketEndpointHandlers<T[K], WS, D>;\n};\n\n/**\n * Upgrade data returned by matchAndPrepareUpgrade\n */\nexport interface UpgradeData {\n endpointName: string;\n endpoint: WebSocketContractDefinition;\n params: Record<string, string>;\n query: Record<string, string | string[]>;\n headers: Record<string, string>;\n}\n\n/**\n * Options for WebSocket router\n */\nexport interface WebSocketRouterOptions<WS extends GenericWebSocket, D = unknown> {\n basePath?: string;\n dataSchema?: z.ZodSchema<D>;\n rawWebSocket?: WS;\n}\n\ntype WithData<T, D> = T &\n (D extends object | string | number | boolean\n ? {\n data: D;\n }\n : {\n data?: never;\n });\n\n/**\n * WebSocket handler interface with context parameter\n */\nexport interface WebSocketHandler<WS extends GenericWebSocket, D = unknown> {\n open(\n args: WithData<\n {\n ws: WS;\n upgradeData: UpgradeData;\n },\n D\n >,\n ): void | Promise<void>;\n message(\n args: WithData<\n {\n ws: WS;\n rawMessage: string | Buffer<ArrayBuffer>;\n upgradeData: UpgradeData;\n },\n D\n >,\n ): void | Promise<void>;\n close(\n args: WithData<\n {\n ws: WS;\n code: number;\n reason: string;\n upgradeData: UpgradeData;\n },\n D\n >,\n ): void;\n drain(args: WithData<{ ws: WS; upgradeData: UpgradeData }, D>): void;\n}\n\n/**\n * Create a typed WebSocket wrapper\n */\nfunction createTypedWebSocket<T extends WebSocketContractDefinition, WS extends GenericWebSocket>(\n ws: WS,\n): TypedServerWebSocket<T, WS> {\n return {\n get raw() {\n return ws;\n },\n send(type, payload) {\n ws.send(JSON.stringify({ type, payload }));\n },\n close(code, reason) {\n ws.close(code, reason);\n },\n };\n}\n\n/**\n * WebSocket router for managing WebSocket contract endpoints\n */\nexport class WebSocketRouter<\n T extends WebSocketContract,\n WS extends GenericWebSocket,\n D = unknown,\n> {\n private basePath: string;\n private dataSchema?: z.ZodSchema<D>;\n\n constructor(\n private contract: T,\n private handlers: WebSocketContractHandlers<T, WS, D>,\n options?: WebSocketRouterOptions<WS, D>,\n ) {\n // Normalize basePath\n const bp = options?.basePath || '';\n if (bp) {\n this.basePath = bp.startsWith('/') ? bp : `/${bp}`;\n this.basePath = this.basePath.endsWith('/') ? this.basePath.slice(0, -1) : this.basePath;\n } else {\n this.basePath = '';\n }\n this.dataSchema = options?.dataSchema;\n }\n\n /**\n * Find matching endpoint for a path\n */\n private findEndpoint(path: string): {\n name: keyof T;\n endpoint: WebSocketContractDefinition;\n params: Record<string, string>;\n } | null {\n for (const [name, endpoint] of Object.entries(this.contract)) {\n const params = matchPath(endpoint.path, path);\n if (params !== null) {\n return {\n name,\n endpoint: endpoint as WebSocketContractDefinition,\n params,\n };\n }\n }\n return null;\n }\n\n /**\n * Parse and validate upgrade request parameters\n */\n private parseUpgradeParams(\n request: Request,\n endpoint: WebSocketContractDefinition,\n pathParams: Record<string, string>,\n ): { params: any; query: any; headers: any } {\n const url = new URL(request.url);\n\n // Parse and validate path params\n let params: any = pathParams;\n if (endpoint.params) {\n const result = endpoint.params.safeParse(pathParams);\n if (!result.success) {\n throw new Error(`Invalid path params: ${result.error.message}`);\n }\n params = result.data;\n }\n\n // Parse and validate query params\n let query: any = {};\n if (endpoint.query) {\n const queryData = parseQuery(url.searchParams);\n const result = endpoint.query.safeParse(queryData);\n if (!result.success) {\n throw new Error(`Invalid query params: ${result.error.message}`);\n }\n query = result.data;\n }\n\n // Parse and validate headers\n let headers: any = {};\n if (endpoint.headers) {\n const headersObj: Record<string, string> = {};\n request.headers.forEach((value, key) => {\n headersObj[key] = value;\n });\n const result = endpoint.headers.safeParse(headersObj);\n if (!result.success) {\n throw new Error(`Invalid headers: ${result.error.message}`);\n }\n headers = result.data;\n }\n\n return { params, query, headers };\n }\n\n /**\n * Match a request and prepare upgrade data\n * Returns null if no match, or UpgradeData for server.upgrade()\n */\n async matchAndPrepareUpgrade(request: Request): Promise<UpgradeData | null> {\n const url = new URL(request.url);\n let path = url.pathname;\n\n // Strip basePath if configured\n if (this.basePath && path.startsWith(this.basePath)) {\n path = path.slice(this.basePath.length) || '/';\n }\n\n const match = this.findEndpoint(path);\n if (!match) {\n return null;\n }\n\n const { name, endpoint, params: rawParams } = match;\n\n try {\n const { params, query, headers } = this.parseUpgradeParams(request, endpoint, rawParams);\n\n return {\n endpointName: String(name),\n endpoint,\n params,\n query,\n headers,\n };\n } catch (err) {\n // Validation failed during upgrade\n console.error('WebSocket upgrade validation failed:', err);\n return null;\n }\n }\n\n /**\n * Validate data against dataSchema if provided\n */\n private validateData(data: unknown): D {\n if (this.dataSchema) {\n const result = this.dataSchema.safeParse(data);\n if (!result.success) {\n throw new WebSocketValidationError('data', result.error);\n }\n return result.data;\n }\n return data as D;\n }\n\n /**\n * Validate an incoming client message\n */\n private validateMessage(\n endpoint: WebSocketContractDefinition,\n rawMessage: string | ArrayBuffer,\n ): ExtractClientMessage<typeof endpoint> {\n // Parse message\n const messageStr =\n typeof rawMessage === 'string' ? rawMessage : new TextDecoder().decode(rawMessage);\n const parsed = JSON.parse(messageStr) as { type: string; payload: unknown };\n\n const { type, payload } = parsed;\n\n // Find the schema for this message type\n const messageDef = endpoint.clientMessages[type];\n if (!messageDef) {\n throw new WebSocketValidationError(\n type,\n new z.ZodError([\n {\n code: 'custom',\n path: ['type'],\n message: `Unknown message type: ${type}`,\n },\n ]),\n );\n }\n\n // Validate payload\n const result = messageDef.payload.safeParse(payload);\n if (!result.success) {\n throw new WebSocketValidationError(type, result.error);\n }\n\n return { type, payload: result.data } as ExtractClientMessage<typeof endpoint>;\n }\n\n /**\n * Get WebSocket handler that accepts context as parameter\n */\n get websocketHandler(): WebSocketHandler<WS, D> {\n return {\n open: async ({ ws, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n // Validate data if schema provided\n const validatedData = this.validateData(data);\n\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n if (endpointHandlers.open) {\n await endpointHandlers.open({\n ws: typedWs,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n }\n },\n\n message: async ({ ws, rawMessage, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n const validatedData = this.validateData(data);\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n try {\n // Validate the message\n const validatedMessage = this.validateMessage(\n upgradeData.endpoint,\n typeof rawMessage === 'string' ? rawMessage : rawMessage.buffer,\n );\n\n // Call handler with validated message\n await endpointHandlers.message({\n ws: typedWs as any,\n message: validatedMessage as any,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n } catch (err) {\n if (err instanceof WebSocketValidationError) {\n // Call validation error handler if provided\n if (endpointHandlers.validationError) {\n endpointHandlers.validationError({\n ws: typedWs as any,\n error: err,\n data: validatedData,\n });\n } else {\n // Default: send error message back\n typedWs.send(\n 'error' as any,\n {\n code: 'VALIDATION_ERROR',\n message: err.message,\n issues: err.zodError.issues,\n } as any,\n );\n }\n } else {\n console.error('WebSocket message handler error:', err);\n }\n }\n },\n\n close: ({ ws, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n const validatedData = this.validateData(data);\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n if (endpointHandlers.close) {\n endpointHandlers.close({\n ws: typedWs as any,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n }\n },\n\n drain: ({ ws, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n const validatedData = this.validateData(data);\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n if (endpointHandlers.drain) {\n endpointHandlers.drain({\n ws: typedWs as any,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n }\n },\n };\n }\n}\n\n/**\n * Create a WebSocket router from a contract and handlers\n */\nexport function createWebSocketRouter<\n T extends WebSocketContract,\n WS extends GenericWebSocket,\n D = unknown,\n>(\n contract: T,\n handlers: WebSocketContractHandlers<T, WS, D>,\n options?: WebSocketRouterOptions<WS, D>,\n): WebSocketRouter<T, WS, D> {\n return new WebSocketRouter(contract, handlers, options);\n}\n"
|
|
5
|
+
"import type {\n ExtractClientMessage,\n ExtractWSHeaders,\n ExtractWSParams,\n ExtractWSQuery,\n WebSocketContract,\n WebSocketContractDefinition,\n} from '@richie-rpc/core';\nimport { matchPath, parseQuery } from '@richie-rpc/core';\nimport { z } from 'zod';\n\n/**\n * Validation error for WebSocket messages\n */\nexport class WebSocketValidationError extends Error {\n constructor(\n public messageType: string,\n public zodError: z.ZodError<unknown>,\n ) {\n const pretty = z.prettifyError(zodError);\n super(`Validation failed for WebSocket message type: ${messageType}:\\n${pretty}`);\n this.name = 'WebSocketValidationError';\n }\n}\n\n/**\n * Typed WebSocket wrapper for sending messages\n */\nexport interface TypedServerWebSocket<\n T extends WebSocketContractDefinition,\n WS extends GenericWebSocket,\n> {\n /** The underlying Bun WebSocket */\n readonly raw: WS;\n /** Send a typed message to the client */\n send<K extends keyof T['serverMessages']>(\n type: K,\n payload: z.infer<T['serverMessages'][K]['payload']>,\n ): void;\n /** Close the connection */\n close(code?: number, reason?: string): void;\n}\n\nexport type GenericWebSocket = {\n send: (message: string) => void;\n close: (code?: number, reason?: string) => void;\n};\n\n/**\n * Arguments passed to handler functions\n */\nexport interface HandlerArgs<\n T extends WebSocketContractDefinition,\n WS extends GenericWebSocket,\n D = unknown,\n> {\n ws: TypedServerWebSocket<T, WS>;\n params: ExtractWSParams<T>;\n query: ExtractWSQuery<T>;\n headers: ExtractWSHeaders<T>;\n data: D;\n}\n\n/**\n * Handler functions for a WebSocket endpoint\n */\nexport interface WebSocketEndpointHandlers<\n T extends WebSocketContractDefinition,\n WS extends GenericWebSocket,\n D = unknown,\n> {\n /** Called when connection opens */\n open?(args: HandlerArgs<T, WS, D>): void | Promise<void>;\n /** Called for each validated message */\n message(args: HandlerArgs<T, WS, D> & { message: ExtractClientMessage<T> }): void | Promise<void>;\n /** Called when connection closes */\n close?(args: HandlerArgs<T, WS, D>): void;\n /** Called on backpressure drain */\n drain?(args: HandlerArgs<T, WS, D>): void;\n /** Called when message validation fails */\n validationError?(args: {\n ws: TypedServerWebSocket<T, WS>;\n error: WebSocketValidationError;\n data: D;\n }): void;\n}\n\n/**\n * Contract handlers mapping for WebSocket endpoints\n */\nexport type WebSocketContractHandlers<\n T extends WebSocketContract,\n WS extends GenericWebSocket,\n D = unknown,\n> = {\n [K in keyof T]: WebSocketEndpointHandlers<T[K], WS, D>;\n};\n\n/**\n * Upgrade data returned by matchAndPrepareUpgrade\n */\nexport interface UpgradeData {\n endpointName: string;\n endpoint: WebSocketContractDefinition;\n params: Record<string, string>;\n query: Record<string, string | string[]>;\n headers: Record<string, string>;\n}\n\n/**\n * Options for WebSocket router\n */\nexport interface WebSocketRouterOptions<WS extends GenericWebSocket, D = unknown> {\n basePath?: string;\n dataSchema?: z.ZodSchema<D>;\n rawWebSocket?: WS;\n}\n\ntype WithData<T, D> = T &\n (D extends object | string | number | boolean\n ? {\n data: D;\n }\n : {\n data?: never;\n });\n\ntype RawWebSocketMessage = string | ArrayBuffer | ArrayBufferView;\n\n/**\n * WebSocket handler interface with context parameter\n */\nexport interface WebSocketHandler<WS extends GenericWebSocket, D = unknown> {\n open(\n args: WithData<\n {\n ws: WS;\n upgradeData: UpgradeData;\n },\n D\n >,\n ): void | Promise<void>;\n message(\n args: WithData<\n {\n ws: WS;\n rawMessage: RawWebSocketMessage;\n upgradeData: UpgradeData;\n },\n D\n >,\n ): void | Promise<void>;\n close(\n args: WithData<\n {\n ws: WS;\n code: number;\n reason: string;\n upgradeData: UpgradeData;\n },\n D\n >,\n ): void;\n drain(args: WithData<{ ws: WS; upgradeData: UpgradeData }, D>): void;\n}\n\n/**\n * Create a typed WebSocket wrapper\n */\nfunction createTypedWebSocket<T extends WebSocketContractDefinition, WS extends GenericWebSocket>(\n ws: WS,\n): TypedServerWebSocket<T, WS> {\n return {\n get raw() {\n return ws;\n },\n send(type, payload) {\n ws.send(JSON.stringify({ type, payload }));\n },\n close(code, reason) {\n ws.close(code, reason);\n },\n };\n}\n\n/**\n * WebSocket router for managing WebSocket contract endpoints\n */\nexport class WebSocketRouter<\n T extends WebSocketContract,\n WS extends GenericWebSocket,\n D = unknown,\n> {\n private basePath: string;\n private dataSchema?: z.ZodSchema<D>;\n\n constructor(\n private contract: T,\n private handlers: WebSocketContractHandlers<T, WS, D>,\n options?: WebSocketRouterOptions<WS, D>,\n ) {\n // Normalize basePath\n const bp = options?.basePath || '';\n if (bp) {\n this.basePath = bp.startsWith('/') ? bp : `/${bp}`;\n this.basePath = this.basePath.endsWith('/') ? this.basePath.slice(0, -1) : this.basePath;\n } else {\n this.basePath = '';\n }\n this.dataSchema = options?.dataSchema;\n }\n\n /**\n * Find matching endpoint for a path\n */\n private findEndpoint(path: string): {\n name: keyof T;\n endpoint: WebSocketContractDefinition;\n params: Record<string, string>;\n } | null {\n for (const [name, endpoint] of Object.entries(this.contract)) {\n const params = matchPath(endpoint.path, path);\n if (params !== null) {\n return {\n name,\n endpoint: endpoint as WebSocketContractDefinition,\n params,\n };\n }\n }\n return null;\n }\n\n /**\n * Parse and validate upgrade request parameters\n */\n private parseUpgradeParams(\n request: Request,\n endpoint: WebSocketContractDefinition,\n pathParams: Record<string, string>,\n ): { params: any; query: any; headers: any } {\n const url = new URL(request.url);\n\n // Parse and validate path params\n let params: any = pathParams;\n if (endpoint.params) {\n const result = endpoint.params.safeParse(pathParams);\n if (!result.success) {\n throw new Error(`Invalid path params: ${result.error.message}`);\n }\n params = result.data;\n }\n\n // Parse and validate query params\n let query: any = {};\n if (endpoint.query) {\n const queryData = parseQuery(url.searchParams);\n const result = endpoint.query.safeParse(queryData);\n if (!result.success) {\n throw new Error(`Invalid query params: ${result.error.message}`);\n }\n query = result.data;\n }\n\n // Parse and validate headers\n let headers: any = {};\n if (endpoint.headers) {\n const headersObj: Record<string, string> = {};\n request.headers.forEach((value, key) => {\n headersObj[key] = value;\n });\n const result = endpoint.headers.safeParse(headersObj);\n if (!result.success) {\n throw new Error(`Invalid headers: ${result.error.message}`);\n }\n headers = result.data;\n }\n\n return { params, query, headers };\n }\n\n /**\n * Match a request and prepare upgrade data\n * Returns null if no match, or UpgradeData for server.upgrade()\n */\n async matchAndPrepareUpgrade(request: Request): Promise<UpgradeData | null> {\n const url = new URL(request.url);\n let path = url.pathname;\n\n // Strip basePath if configured\n if (this.basePath && path.startsWith(this.basePath)) {\n path = path.slice(this.basePath.length) || '/';\n }\n\n const match = this.findEndpoint(path);\n if (!match) {\n return null;\n }\n\n const { name, endpoint, params: rawParams } = match;\n\n try {\n const { params, query, headers } = this.parseUpgradeParams(request, endpoint, rawParams);\n\n return {\n endpointName: String(name),\n endpoint,\n params,\n query,\n headers,\n };\n } catch (err) {\n // Validation failed during upgrade\n console.error('WebSocket upgrade validation failed:', err);\n return null;\n }\n }\n\n /**\n * Validate data against dataSchema if provided\n */\n private validateData(data: unknown): D {\n if (this.dataSchema) {\n const result = this.dataSchema.safeParse(data);\n if (!result.success) {\n throw new WebSocketValidationError('data', result.error);\n }\n return result.data;\n }\n return data as D;\n }\n\n /**\n * Validate an incoming client message\n */\n private validateMessage(\n endpoint: WebSocketContractDefinition,\n rawMessage: RawWebSocketMessage,\n ): ExtractClientMessage<typeof endpoint> {\n // Parse message\n const messageStr =\n typeof rawMessage === 'string' ? rawMessage : new TextDecoder().decode(rawMessage);\n const parsed = JSON.parse(messageStr) as { type: string; payload: unknown };\n\n const { type, payload } = parsed;\n\n // Find the schema for this message type\n const messageDef = endpoint.clientMessages[type];\n if (!messageDef) {\n throw new WebSocketValidationError(\n type,\n new z.ZodError([\n {\n code: 'custom',\n path: ['type'],\n message: `Unknown message type: ${type}`,\n },\n ]),\n );\n }\n\n // Validate payload\n const result = messageDef.payload.safeParse(payload);\n if (!result.success) {\n throw new WebSocketValidationError(type, result.error);\n }\n\n return { type, payload: result.data } as ExtractClientMessage<typeof endpoint>;\n }\n\n /**\n * Get WebSocket handler that accepts context as parameter\n */\n get websocketHandler(): WebSocketHandler<WS, D> {\n return {\n open: async ({ ws, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n // Validate data if schema provided\n const validatedData = this.validateData(data);\n\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n if (endpointHandlers.open) {\n await endpointHandlers.open({\n ws: typedWs,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n }\n },\n\n message: async ({ ws, rawMessage, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n const validatedData = this.validateData(data);\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n try {\n // Validate the message\n const validatedMessage = this.validateMessage(upgradeData.endpoint, rawMessage);\n\n // Call handler with validated message\n await endpointHandlers.message({\n ws: typedWs as any,\n message: validatedMessage as any,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n } catch (err) {\n if (err instanceof WebSocketValidationError) {\n // Call validation error handler if provided\n if (endpointHandlers.validationError) {\n endpointHandlers.validationError({\n ws: typedWs as any,\n error: err,\n data: validatedData,\n });\n } else {\n // Default: send error message back\n typedWs.send(\n 'error' as any,\n {\n code: 'VALIDATION_ERROR',\n message: err.message,\n issues: err.zodError.issues,\n } as any,\n );\n }\n } else {\n console.error('WebSocket message handler error:', err);\n }\n }\n },\n\n close: ({ ws, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n const validatedData = this.validateData(data);\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n if (endpointHandlers.close) {\n endpointHandlers.close({\n ws: typedWs as any,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n }\n },\n\n drain: ({ ws, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n const validatedData = this.validateData(data);\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n if (endpointHandlers.drain) {\n endpointHandlers.drain({\n ws: typedWs as any,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n }\n },\n };\n }\n}\n\n/**\n * Create a WebSocket router from a contract and handlers\n */\nexport function createWebSocketRouter<\n T extends WebSocketContract,\n WS extends GenericWebSocket,\n D = unknown,\n>(\n contract: T,\n handlers: WebSocketContractHandlers<T, WS, D>,\n options?: WebSocketRouterOptions<WS, D>,\n): WebSocketRouter<T, WS, D> {\n return new WebSocketRouter(contract, handlers, options);\n}\n"
|
|
6
6
|
],
|
|
7
|
-
"mappings": "
|
|
8
|
-
"debugId": "
|
|
7
|
+
"mappings": "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",
|
|
8
|
+
"debugId": "CD0572859FC97E1C64756E2164756E21",
|
|
9
9
|
"names": []
|
|
10
10
|
}
|
package/dist/mjs/package.json
CHANGED
package/dist/mjs/websocket.mjs
CHANGED
|
@@ -172,7 +172,7 @@ class WebSocketRouter {
|
|
|
172
172
|
const validatedData = this.validateData(data);
|
|
173
173
|
const typedWs = createTypedWebSocket(ws);
|
|
174
174
|
try {
|
|
175
|
-
const validatedMessage = this.validateMessage(upgradeData.endpoint,
|
|
175
|
+
const validatedMessage = this.validateMessage(upgradeData.endpoint, rawMessage);
|
|
176
176
|
await endpointHandlers.message({
|
|
177
177
|
ws: typedWs,
|
|
178
178
|
message: validatedMessage,
|
|
@@ -245,4 +245,4 @@ export {
|
|
|
245
245
|
WebSocketRouter
|
|
246
246
|
};
|
|
247
247
|
|
|
248
|
-
//# debugId=
|
|
248
|
+
//# debugId=33B03EDF351D69B064756E2164756E21
|
|
@@ -2,9 +2,9 @@
|
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../websocket.ts"],
|
|
4
4
|
"sourcesContent": [
|
|
5
|
-
"import type {\n ExtractClientMessage,\n ExtractWSHeaders,\n ExtractWSParams,\n ExtractWSQuery,\n WebSocketContract,\n WebSocketContractDefinition,\n} from '@richie-rpc/core';\nimport { matchPath, parseQuery } from '@richie-rpc/core';\nimport { z } from 'zod';\n\n/**\n * Validation error for WebSocket messages\n */\nexport class WebSocketValidationError extends Error {\n constructor(\n public messageType: string,\n public zodError: z.ZodError<unknown>,\n ) {\n const pretty = z.prettifyError(zodError);\n super(`Validation failed for WebSocket message type: ${messageType}:\\n${pretty}`);\n this.name = 'WebSocketValidationError';\n }\n}\n\n/**\n * Typed WebSocket wrapper for sending messages\n */\nexport interface TypedServerWebSocket<\n T extends WebSocketContractDefinition,\n WS extends GenericWebSocket,\n> {\n /** The underlying Bun WebSocket */\n readonly raw: WS;\n /** Send a typed message to the client */\n send<K extends keyof T['serverMessages']>(\n type: K,\n payload: z.infer<T['serverMessages'][K]['payload']>,\n ): void;\n /** Close the connection */\n close(code?: number, reason?: string): void;\n}\n\nexport type GenericWebSocket = {\n send: (message: string) => void;\n close: (code?: number, reason?: string) => void;\n};\n\n/**\n * Arguments passed to handler functions\n */\nexport interface HandlerArgs<\n T extends WebSocketContractDefinition,\n WS extends GenericWebSocket,\n D = unknown,\n> {\n ws: TypedServerWebSocket<T, WS>;\n params: ExtractWSParams<T>;\n query: ExtractWSQuery<T>;\n headers: ExtractWSHeaders<T>;\n data: D;\n}\n\n/**\n * Handler functions for a WebSocket endpoint\n */\nexport interface WebSocketEndpointHandlers<\n T extends WebSocketContractDefinition,\n WS extends GenericWebSocket,\n D = unknown,\n> {\n /** Called when connection opens */\n open?(args: HandlerArgs<T, WS, D>): void | Promise<void>;\n /** Called for each validated message */\n message(args: HandlerArgs<T, WS, D> & { message: ExtractClientMessage<T> }): void | Promise<void>;\n /** Called when connection closes */\n close?(args: HandlerArgs<T, WS, D>): void;\n /** Called on backpressure drain */\n drain?(args: HandlerArgs<T, WS, D>): void;\n /** Called when message validation fails */\n validationError?(args: {\n ws: TypedServerWebSocket<T, WS>;\n error: WebSocketValidationError;\n data: D;\n }): void;\n}\n\n/**\n * Contract handlers mapping for WebSocket endpoints\n */\nexport type WebSocketContractHandlers<\n T extends WebSocketContract,\n WS extends GenericWebSocket,\n D = unknown,\n> = {\n [K in keyof T]: WebSocketEndpointHandlers<T[K], WS, D>;\n};\n\n/**\n * Upgrade data returned by matchAndPrepareUpgrade\n */\nexport interface UpgradeData {\n endpointName: string;\n endpoint: WebSocketContractDefinition;\n params: Record<string, string>;\n query: Record<string, string | string[]>;\n headers: Record<string, string>;\n}\n\n/**\n * Options for WebSocket router\n */\nexport interface WebSocketRouterOptions<WS extends GenericWebSocket, D = unknown> {\n basePath?: string;\n dataSchema?: z.ZodSchema<D>;\n rawWebSocket?: WS;\n}\n\ntype WithData<T, D> = T &\n (D extends object | string | number | boolean\n ? {\n data: D;\n }\n : {\n data?: never;\n });\n\n/**\n * WebSocket handler interface with context parameter\n */\nexport interface WebSocketHandler<WS extends GenericWebSocket, D = unknown> {\n open(\n args: WithData<\n {\n ws: WS;\n upgradeData: UpgradeData;\n },\n D\n >,\n ): void | Promise<void>;\n message(\n args: WithData<\n {\n ws: WS;\n rawMessage: string | Buffer<ArrayBuffer>;\n upgradeData: UpgradeData;\n },\n D\n >,\n ): void | Promise<void>;\n close(\n args: WithData<\n {\n ws: WS;\n code: number;\n reason: string;\n upgradeData: UpgradeData;\n },\n D\n >,\n ): void;\n drain(args: WithData<{ ws: WS; upgradeData: UpgradeData }, D>): void;\n}\n\n/**\n * Create a typed WebSocket wrapper\n */\nfunction createTypedWebSocket<T extends WebSocketContractDefinition, WS extends GenericWebSocket>(\n ws: WS,\n): TypedServerWebSocket<T, WS> {\n return {\n get raw() {\n return ws;\n },\n send(type, payload) {\n ws.send(JSON.stringify({ type, payload }));\n },\n close(code, reason) {\n ws.close(code, reason);\n },\n };\n}\n\n/**\n * WebSocket router for managing WebSocket contract endpoints\n */\nexport class WebSocketRouter<\n T extends WebSocketContract,\n WS extends GenericWebSocket,\n D = unknown,\n> {\n private basePath: string;\n private dataSchema?: z.ZodSchema<D>;\n\n constructor(\n private contract: T,\n private handlers: WebSocketContractHandlers<T, WS, D>,\n options?: WebSocketRouterOptions<WS, D>,\n ) {\n // Normalize basePath\n const bp = options?.basePath || '';\n if (bp) {\n this.basePath = bp.startsWith('/') ? bp : `/${bp}`;\n this.basePath = this.basePath.endsWith('/') ? this.basePath.slice(0, -1) : this.basePath;\n } else {\n this.basePath = '';\n }\n this.dataSchema = options?.dataSchema;\n }\n\n /**\n * Find matching endpoint for a path\n */\n private findEndpoint(path: string): {\n name: keyof T;\n endpoint: WebSocketContractDefinition;\n params: Record<string, string>;\n } | null {\n for (const [name, endpoint] of Object.entries(this.contract)) {\n const params = matchPath(endpoint.path, path);\n if (params !== null) {\n return {\n name,\n endpoint: endpoint as WebSocketContractDefinition,\n params,\n };\n }\n }\n return null;\n }\n\n /**\n * Parse and validate upgrade request parameters\n */\n private parseUpgradeParams(\n request: Request,\n endpoint: WebSocketContractDefinition,\n pathParams: Record<string, string>,\n ): { params: any; query: any; headers: any } {\n const url = new URL(request.url);\n\n // Parse and validate path params\n let params: any = pathParams;\n if (endpoint.params) {\n const result = endpoint.params.safeParse(pathParams);\n if (!result.success) {\n throw new Error(`Invalid path params: ${result.error.message}`);\n }\n params = result.data;\n }\n\n // Parse and validate query params\n let query: any = {};\n if (endpoint.query) {\n const queryData = parseQuery(url.searchParams);\n const result = endpoint.query.safeParse(queryData);\n if (!result.success) {\n throw new Error(`Invalid query params: ${result.error.message}`);\n }\n query = result.data;\n }\n\n // Parse and validate headers\n let headers: any = {};\n if (endpoint.headers) {\n const headersObj: Record<string, string> = {};\n request.headers.forEach((value, key) => {\n headersObj[key] = value;\n });\n const result = endpoint.headers.safeParse(headersObj);\n if (!result.success) {\n throw new Error(`Invalid headers: ${result.error.message}`);\n }\n headers = result.data;\n }\n\n return { params, query, headers };\n }\n\n /**\n * Match a request and prepare upgrade data\n * Returns null if no match, or UpgradeData for server.upgrade()\n */\n async matchAndPrepareUpgrade(request: Request): Promise<UpgradeData | null> {\n const url = new URL(request.url);\n let path = url.pathname;\n\n // Strip basePath if configured\n if (this.basePath && path.startsWith(this.basePath)) {\n path = path.slice(this.basePath.length) || '/';\n }\n\n const match = this.findEndpoint(path);\n if (!match) {\n return null;\n }\n\n const { name, endpoint, params: rawParams } = match;\n\n try {\n const { params, query, headers } = this.parseUpgradeParams(request, endpoint, rawParams);\n\n return {\n endpointName: String(name),\n endpoint,\n params,\n query,\n headers,\n };\n } catch (err) {\n // Validation failed during upgrade\n console.error('WebSocket upgrade validation failed:', err);\n return null;\n }\n }\n\n /**\n * Validate data against dataSchema if provided\n */\n private validateData(data: unknown): D {\n if (this.dataSchema) {\n const result = this.dataSchema.safeParse(data);\n if (!result.success) {\n throw new WebSocketValidationError('data', result.error);\n }\n return result.data;\n }\n return data as D;\n }\n\n /**\n * Validate an incoming client message\n */\n private validateMessage(\n endpoint: WebSocketContractDefinition,\n rawMessage: string | ArrayBuffer,\n ): ExtractClientMessage<typeof endpoint> {\n // Parse message\n const messageStr =\n typeof rawMessage === 'string' ? rawMessage : new TextDecoder().decode(rawMessage);\n const parsed = JSON.parse(messageStr) as { type: string; payload: unknown };\n\n const { type, payload } = parsed;\n\n // Find the schema for this message type\n const messageDef = endpoint.clientMessages[type];\n if (!messageDef) {\n throw new WebSocketValidationError(\n type,\n new z.ZodError([\n {\n code: 'custom',\n path: ['type'],\n message: `Unknown message type: ${type}`,\n },\n ]),\n );\n }\n\n // Validate payload\n const result = messageDef.payload.safeParse(payload);\n if (!result.success) {\n throw new WebSocketValidationError(type, result.error);\n }\n\n return { type, payload: result.data } as ExtractClientMessage<typeof endpoint>;\n }\n\n /**\n * Get WebSocket handler that accepts context as parameter\n */\n get websocketHandler(): WebSocketHandler<WS, D> {\n return {\n open: async ({ ws, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n // Validate data if schema provided\n const validatedData = this.validateData(data);\n\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n if (endpointHandlers.open) {\n await endpointHandlers.open({\n ws: typedWs,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n }\n },\n\n message: async ({ ws, rawMessage, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n const validatedData = this.validateData(data);\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n try {\n // Validate the message\n const validatedMessage = this.validateMessage(\n upgradeData.endpoint,\n typeof rawMessage === 'string' ? rawMessage : rawMessage.buffer,\n );\n\n // Call handler with validated message\n await endpointHandlers.message({\n ws: typedWs as any,\n message: validatedMessage as any,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n } catch (err) {\n if (err instanceof WebSocketValidationError) {\n // Call validation error handler if provided\n if (endpointHandlers.validationError) {\n endpointHandlers.validationError({\n ws: typedWs as any,\n error: err,\n data: validatedData,\n });\n } else {\n // Default: send error message back\n typedWs.send(\n 'error' as any,\n {\n code: 'VALIDATION_ERROR',\n message: err.message,\n issues: err.zodError.issues,\n } as any,\n );\n }\n } else {\n console.error('WebSocket message handler error:', err);\n }\n }\n },\n\n close: ({ ws, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n const validatedData = this.validateData(data);\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n if (endpointHandlers.close) {\n endpointHandlers.close({\n ws: typedWs as any,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n }\n },\n\n drain: ({ ws, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n const validatedData = this.validateData(data);\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n if (endpointHandlers.drain) {\n endpointHandlers.drain({\n ws: typedWs as any,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n }\n },\n };\n }\n}\n\n/**\n * Create a WebSocket router from a contract and handlers\n */\nexport function createWebSocketRouter<\n T extends WebSocketContract,\n WS extends GenericWebSocket,\n D = unknown,\n>(\n contract: T,\n handlers: WebSocketContractHandlers<T, WS, D>,\n options?: WebSocketRouterOptions<WS, D>,\n): WebSocketRouter<T, WS, D> {\n return new WebSocketRouter(contract, handlers, options);\n}\n"
|
|
5
|
+
"import type {\n ExtractClientMessage,\n ExtractWSHeaders,\n ExtractWSParams,\n ExtractWSQuery,\n WebSocketContract,\n WebSocketContractDefinition,\n} from '@richie-rpc/core';\nimport { matchPath, parseQuery } from '@richie-rpc/core';\nimport { z } from 'zod';\n\n/**\n * Validation error for WebSocket messages\n */\nexport class WebSocketValidationError extends Error {\n constructor(\n public messageType: string,\n public zodError: z.ZodError<unknown>,\n ) {\n const pretty = z.prettifyError(zodError);\n super(`Validation failed for WebSocket message type: ${messageType}:\\n${pretty}`);\n this.name = 'WebSocketValidationError';\n }\n}\n\n/**\n * Typed WebSocket wrapper for sending messages\n */\nexport interface TypedServerWebSocket<\n T extends WebSocketContractDefinition,\n WS extends GenericWebSocket,\n> {\n /** The underlying Bun WebSocket */\n readonly raw: WS;\n /** Send a typed message to the client */\n send<K extends keyof T['serverMessages']>(\n type: K,\n payload: z.infer<T['serverMessages'][K]['payload']>,\n ): void;\n /** Close the connection */\n close(code?: number, reason?: string): void;\n}\n\nexport type GenericWebSocket = {\n send: (message: string) => void;\n close: (code?: number, reason?: string) => void;\n};\n\n/**\n * Arguments passed to handler functions\n */\nexport interface HandlerArgs<\n T extends WebSocketContractDefinition,\n WS extends GenericWebSocket,\n D = unknown,\n> {\n ws: TypedServerWebSocket<T, WS>;\n params: ExtractWSParams<T>;\n query: ExtractWSQuery<T>;\n headers: ExtractWSHeaders<T>;\n data: D;\n}\n\n/**\n * Handler functions for a WebSocket endpoint\n */\nexport interface WebSocketEndpointHandlers<\n T extends WebSocketContractDefinition,\n WS extends GenericWebSocket,\n D = unknown,\n> {\n /** Called when connection opens */\n open?(args: HandlerArgs<T, WS, D>): void | Promise<void>;\n /** Called for each validated message */\n message(args: HandlerArgs<T, WS, D> & { message: ExtractClientMessage<T> }): void | Promise<void>;\n /** Called when connection closes */\n close?(args: HandlerArgs<T, WS, D>): void;\n /** Called on backpressure drain */\n drain?(args: HandlerArgs<T, WS, D>): void;\n /** Called when message validation fails */\n validationError?(args: {\n ws: TypedServerWebSocket<T, WS>;\n error: WebSocketValidationError;\n data: D;\n }): void;\n}\n\n/**\n * Contract handlers mapping for WebSocket endpoints\n */\nexport type WebSocketContractHandlers<\n T extends WebSocketContract,\n WS extends GenericWebSocket,\n D = unknown,\n> = {\n [K in keyof T]: WebSocketEndpointHandlers<T[K], WS, D>;\n};\n\n/**\n * Upgrade data returned by matchAndPrepareUpgrade\n */\nexport interface UpgradeData {\n endpointName: string;\n endpoint: WebSocketContractDefinition;\n params: Record<string, string>;\n query: Record<string, string | string[]>;\n headers: Record<string, string>;\n}\n\n/**\n * Options for WebSocket router\n */\nexport interface WebSocketRouterOptions<WS extends GenericWebSocket, D = unknown> {\n basePath?: string;\n dataSchema?: z.ZodSchema<D>;\n rawWebSocket?: WS;\n}\n\ntype WithData<T, D> = T &\n (D extends object | string | number | boolean\n ? {\n data: D;\n }\n : {\n data?: never;\n });\n\ntype RawWebSocketMessage = string | ArrayBuffer | ArrayBufferView;\n\n/**\n * WebSocket handler interface with context parameter\n */\nexport interface WebSocketHandler<WS extends GenericWebSocket, D = unknown> {\n open(\n args: WithData<\n {\n ws: WS;\n upgradeData: UpgradeData;\n },\n D\n >,\n ): void | Promise<void>;\n message(\n args: WithData<\n {\n ws: WS;\n rawMessage: RawWebSocketMessage;\n upgradeData: UpgradeData;\n },\n D\n >,\n ): void | Promise<void>;\n close(\n args: WithData<\n {\n ws: WS;\n code: number;\n reason: string;\n upgradeData: UpgradeData;\n },\n D\n >,\n ): void;\n drain(args: WithData<{ ws: WS; upgradeData: UpgradeData }, D>): void;\n}\n\n/**\n * Create a typed WebSocket wrapper\n */\nfunction createTypedWebSocket<T extends WebSocketContractDefinition, WS extends GenericWebSocket>(\n ws: WS,\n): TypedServerWebSocket<T, WS> {\n return {\n get raw() {\n return ws;\n },\n send(type, payload) {\n ws.send(JSON.stringify({ type, payload }));\n },\n close(code, reason) {\n ws.close(code, reason);\n },\n };\n}\n\n/**\n * WebSocket router for managing WebSocket contract endpoints\n */\nexport class WebSocketRouter<\n T extends WebSocketContract,\n WS extends GenericWebSocket,\n D = unknown,\n> {\n private basePath: string;\n private dataSchema?: z.ZodSchema<D>;\n\n constructor(\n private contract: T,\n private handlers: WebSocketContractHandlers<T, WS, D>,\n options?: WebSocketRouterOptions<WS, D>,\n ) {\n // Normalize basePath\n const bp = options?.basePath || '';\n if (bp) {\n this.basePath = bp.startsWith('/') ? bp : `/${bp}`;\n this.basePath = this.basePath.endsWith('/') ? this.basePath.slice(0, -1) : this.basePath;\n } else {\n this.basePath = '';\n }\n this.dataSchema = options?.dataSchema;\n }\n\n /**\n * Find matching endpoint for a path\n */\n private findEndpoint(path: string): {\n name: keyof T;\n endpoint: WebSocketContractDefinition;\n params: Record<string, string>;\n } | null {\n for (const [name, endpoint] of Object.entries(this.contract)) {\n const params = matchPath(endpoint.path, path);\n if (params !== null) {\n return {\n name,\n endpoint: endpoint as WebSocketContractDefinition,\n params,\n };\n }\n }\n return null;\n }\n\n /**\n * Parse and validate upgrade request parameters\n */\n private parseUpgradeParams(\n request: Request,\n endpoint: WebSocketContractDefinition,\n pathParams: Record<string, string>,\n ): { params: any; query: any; headers: any } {\n const url = new URL(request.url);\n\n // Parse and validate path params\n let params: any = pathParams;\n if (endpoint.params) {\n const result = endpoint.params.safeParse(pathParams);\n if (!result.success) {\n throw new Error(`Invalid path params: ${result.error.message}`);\n }\n params = result.data;\n }\n\n // Parse and validate query params\n let query: any = {};\n if (endpoint.query) {\n const queryData = parseQuery(url.searchParams);\n const result = endpoint.query.safeParse(queryData);\n if (!result.success) {\n throw new Error(`Invalid query params: ${result.error.message}`);\n }\n query = result.data;\n }\n\n // Parse and validate headers\n let headers: any = {};\n if (endpoint.headers) {\n const headersObj: Record<string, string> = {};\n request.headers.forEach((value, key) => {\n headersObj[key] = value;\n });\n const result = endpoint.headers.safeParse(headersObj);\n if (!result.success) {\n throw new Error(`Invalid headers: ${result.error.message}`);\n }\n headers = result.data;\n }\n\n return { params, query, headers };\n }\n\n /**\n * Match a request and prepare upgrade data\n * Returns null if no match, or UpgradeData for server.upgrade()\n */\n async matchAndPrepareUpgrade(request: Request): Promise<UpgradeData | null> {\n const url = new URL(request.url);\n let path = url.pathname;\n\n // Strip basePath if configured\n if (this.basePath && path.startsWith(this.basePath)) {\n path = path.slice(this.basePath.length) || '/';\n }\n\n const match = this.findEndpoint(path);\n if (!match) {\n return null;\n }\n\n const { name, endpoint, params: rawParams } = match;\n\n try {\n const { params, query, headers } = this.parseUpgradeParams(request, endpoint, rawParams);\n\n return {\n endpointName: String(name),\n endpoint,\n params,\n query,\n headers,\n };\n } catch (err) {\n // Validation failed during upgrade\n console.error('WebSocket upgrade validation failed:', err);\n return null;\n }\n }\n\n /**\n * Validate data against dataSchema if provided\n */\n private validateData(data: unknown): D {\n if (this.dataSchema) {\n const result = this.dataSchema.safeParse(data);\n if (!result.success) {\n throw new WebSocketValidationError('data', result.error);\n }\n return result.data;\n }\n return data as D;\n }\n\n /**\n * Validate an incoming client message\n */\n private validateMessage(\n endpoint: WebSocketContractDefinition,\n rawMessage: RawWebSocketMessage,\n ): ExtractClientMessage<typeof endpoint> {\n // Parse message\n const messageStr =\n typeof rawMessage === 'string' ? rawMessage : new TextDecoder().decode(rawMessage);\n const parsed = JSON.parse(messageStr) as { type: string; payload: unknown };\n\n const { type, payload } = parsed;\n\n // Find the schema for this message type\n const messageDef = endpoint.clientMessages[type];\n if (!messageDef) {\n throw new WebSocketValidationError(\n type,\n new z.ZodError([\n {\n code: 'custom',\n path: ['type'],\n message: `Unknown message type: ${type}`,\n },\n ]),\n );\n }\n\n // Validate payload\n const result = messageDef.payload.safeParse(payload);\n if (!result.success) {\n throw new WebSocketValidationError(type, result.error);\n }\n\n return { type, payload: result.data } as ExtractClientMessage<typeof endpoint>;\n }\n\n /**\n * Get WebSocket handler that accepts context as parameter\n */\n get websocketHandler(): WebSocketHandler<WS, D> {\n return {\n open: async ({ ws, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n // Validate data if schema provided\n const validatedData = this.validateData(data);\n\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n if (endpointHandlers.open) {\n await endpointHandlers.open({\n ws: typedWs,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n }\n },\n\n message: async ({ ws, rawMessage, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n const validatedData = this.validateData(data);\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n try {\n // Validate the message\n const validatedMessage = this.validateMessage(upgradeData.endpoint, rawMessage);\n\n // Call handler with validated message\n await endpointHandlers.message({\n ws: typedWs as any,\n message: validatedMessage as any,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n } catch (err) {\n if (err instanceof WebSocketValidationError) {\n // Call validation error handler if provided\n if (endpointHandlers.validationError) {\n endpointHandlers.validationError({\n ws: typedWs as any,\n error: err,\n data: validatedData,\n });\n } else {\n // Default: send error message back\n typedWs.send(\n 'error' as any,\n {\n code: 'VALIDATION_ERROR',\n message: err.message,\n issues: err.zodError.issues,\n } as any,\n );\n }\n } else {\n console.error('WebSocket message handler error:', err);\n }\n }\n },\n\n close: ({ ws, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n const validatedData = this.validateData(data);\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n if (endpointHandlers.close) {\n endpointHandlers.close({\n ws: typedWs as any,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n }\n },\n\n drain: ({ ws, upgradeData, data }) => {\n const endpointHandlers = this.handlers[upgradeData.endpointName as keyof T];\n if (!endpointHandlers) return;\n\n const validatedData = this.validateData(data);\n const typedWs = createTypedWebSocket<WebSocketContractDefinition, WS>(ws);\n\n if (endpointHandlers.drain) {\n endpointHandlers.drain({\n ws: typedWs as any,\n params: upgradeData.params as any,\n query: upgradeData.query as any,\n headers: upgradeData.headers as any,\n data: validatedData,\n });\n }\n },\n };\n }\n}\n\n/**\n * Create a WebSocket router from a contract and handlers\n */\nexport function createWebSocketRouter<\n T extends WebSocketContract,\n WS extends GenericWebSocket,\n D = unknown,\n>(\n contract: T,\n handlers: WebSocketContractHandlers<T, WS, D>,\n options?: WebSocketRouterOptions<WS, D>,\n): WebSocketRouter<T, WS, D> {\n return new WebSocketRouter(contract, handlers, options);\n}\n"
|
|
6
6
|
],
|
|
7
|
-
"mappings": ";;AAQA;AACA;AAAA;AAKO,MAAM,iCAAiC,MAAM;AAAA,EAEzC;AAAA,EACA;AAAA,EAFT,WAAW,CACF,aACA,UACP;AAAA,IACA,MAAM,SAAS,EAAE,cAAc,QAAQ;AAAA,IACvC,MAAM,iDAAiD;AAAA,EAAiB,QAAQ;AAAA,IAJzE;AAAA,IACA;AAAA,IAIP,KAAK,OAAO;AAAA;AAEhB;
|
|
8
|
-
"debugId": "
|
|
7
|
+
"mappings": "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",
|
|
8
|
+
"debugId": "33B03EDF351D69B064756E2164756E21",
|
|
9
9
|
"names": []
|
|
10
10
|
}
|
|
@@ -83,6 +83,7 @@ type WithData<T, D> = T & (D extends object | string | number | boolean ? {
|
|
|
83
83
|
} : {
|
|
84
84
|
data?: never;
|
|
85
85
|
});
|
|
86
|
+
type RawWebSocketMessage = string | ArrayBuffer | ArrayBufferView;
|
|
86
87
|
/**
|
|
87
88
|
* WebSocket handler interface with context parameter
|
|
88
89
|
*/
|
|
@@ -93,7 +94,7 @@ export interface WebSocketHandler<WS extends GenericWebSocket, D = unknown> {
|
|
|
93
94
|
}, D>): void | Promise<void>;
|
|
94
95
|
message(args: WithData<{
|
|
95
96
|
ws: WS;
|
|
96
|
-
rawMessage:
|
|
97
|
+
rawMessage: RawWebSocketMessage;
|
|
97
98
|
upgradeData: UpgradeData;
|
|
98
99
|
}, D>): void | Promise<void>;
|
|
99
100
|
close(args: WithData<{
|