UNPKG

@minisylar/express-typed-router

Version:

A strongly-typed Express router with Zod validation and automatic type inference for params, body, query, and middleware

587 lines (586 loc) 38.8 kB
import express, { NextFunction, Request, Response } from "express"; import { StandardSchemaV1 } from "@standard-schema/spec"; //#region src/schema-router.d.ts type AnyStandardSchema = StandardSchemaV1<any, any>; type InferOutput<T> = T extends StandardSchemaV1 ? StandardSchemaV1.InferOutput<T> : unknown; type InferInput<T> = T extends StandardSchemaV1 ? StandardSchemaV1.InferInput<T> : unknown; type InferFromSafeParse<T> = T extends { safeParse: (...args: any[]) => infer R; } ? R extends { success: true; data: infer O; } ? O : R extends Promise<infer PR> ? PR extends { success: true; data: infer O; } ? O : never : never : T extends { parse: (...args: any[]) => infer R; } ? R : never; type InferSchemaOutput<T> = T extends StandardSchemaV1 ? StandardSchemaV1.InferOutput<T> : InferFromSafeParse<T>; declare function parseSchema<T>(schema: T, data: unknown): InferSchemaOutput<T>; type SafeParseResult<T> = StandardSchemaV1.Result<InferSchemaOutput<T>> | { value: InferSchemaOutput<T>; } | { issues: any[]; } | Promise<StandardSchemaV1.Result<InferSchemaOutput<T>> | { value: InferSchemaOutput<T>; } | { issues: any[]; }>; declare function safeParseSchema<T>(schema: T, data: unknown): SafeParseResult<T>; declare function isSchemaError(error: unknown): error is { issues: any[]; }; /** * Extract route parameters from Express.js route patterns. * * Supports all Express.js routing patterns: * - Named parameters: /users/:userId → { userId: string } * - Multiple parameters: /users/:userId/books/:bookId → { userId: string; bookId: string } * - Parameters with separators: /flights/:from-:to → { from: string; to: string } * - Dot notation: /plantae/:genus.:species → { genus: string; species: string } * - Regex constraints: /user/:id(\d+) → { id: string } * - Optional parameters: /posts/:year/:month? → { year: string; month?: string } * - Wildcard parameters: /files/* → { "0": string } * - Multiple wildcards: /a/star/b/star → { "0": string; "1": string } */ type ExtractRouteParams<Path extends string> = string extends Path ? Record<string, string> : ExtractParams<Path>; /** * Main parameter extraction logic - enhanced for Express 5 support with recursion depth limit */ type ExtractParams<Path extends string> = Path extends `${infer Before}{${infer OptionalContent}}${infer After}` ? ExtractOptionalSegment<OptionalContent> & ExtractParams<`${Before}${After}`> : Path extends `${infer _Before}:${infer Rest}` ? ExtractSingleParam<Rest> & ExtractParams<RemoveFirstParam<Path>> : Path extends `${infer _Before}*${infer Name}/${infer After}` ? Name extends "" ? { [K in CountWildcards<_Before, "0">]: string } & ExtractParams<`/${After}`> : { [K in Name]: string[] } & ExtractParams<`/${After}`> : Path extends `${infer _Before}*${infer Name}-${infer After}` ? Name extends "" ? { [K in CountWildcards<_Before, "0">]: string } & ExtractParams<`-${After}`> : { [K in Name]: string[] } & ExtractParams<`-${After}`> : Path extends `${infer _Before}*${infer Name}.${infer After}` ? Name extends "" ? { [K in CountWildcards<_Before, "0">]: string } & ExtractParams<`.${After}`> : { [K in Name]: string[] } & ExtractParams<`.${After}`> : Path extends `${infer _Before}*${infer Name}#${infer After}` ? Name extends "" ? { [K in CountWildcards<_Before, "0">]: string } & ExtractParams<`#${After}`> : { [K in Name]: string[] } & ExtractParams<`#${After}`> : Path extends `${infer _Before}*${infer Name}:${infer After}` ? Name extends "" ? { [K in CountWildcards<_Before, "0">]: string } & ExtractParams<`:${After}`> : { [K in Name]: string[] } & ExtractParams<`:${After}`> : Path extends `${infer _Before}*${infer Name}` ? Name extends "" ? { [K in CountWildcards<_Before, "0">]: string } & ExtractParams<``> : { [K in Name]: string[] } & ExtractParams<``> : Path extends `${infer _Before}*${infer After}` ? { [K in CountWildcards<_Before, "0">]: string } & ExtractParams<After> : {}; /** * Extract parameters from Express 5 optional segments in braces * Handles patterns like {/:param}, {.:ext}, {/optional/:param} */ type ExtractOptionalSegment<Content extends string> = Content extends `*${infer Name}` ? Name extends "" ? {} : { [K in Name]?: string[] } : Content extends `/${infer Rest}` ? Rest extends `*${infer Name}` ? Name extends "" ? {} : { [K in Name]?: string[] } : Rest extends `:${infer R}` ? ExtractOptionalParam<R> : {} : Content extends `/:${infer Rest}` ? ExtractOptionalParam<Rest> : Content extends `.:${infer Rest}` ? ExtractOptionalParam<Rest> : Content extends `${infer _Path}:${infer Rest}` ? ExtractOptionalParam<Rest> : {}; /** * Extract a single optional parameter from brace content */ type ExtractOptionalParam<Rest extends string> = Rest extends `${infer ParamName}/${infer _After}` ? { [K in ParamName]?: string } : Rest extends `${infer ParamName}-${infer _After}` ? { [K in ParamName]?: string } : Rest extends `${infer ParamName}.${infer _After}` ? { [K in ParamName]?: string } : Rest extends `${infer ParamName}` ? { [K in ParamName]?: string } : {}; /** * Extract a single parameter name from the rest of the path * Enhanced to handle Express 5 patterns and optional parameters correctly * Special handling for consecutive parameters like :from-:to * Order matters: regex constraints must be handled before repeating parameters */ type ExtractSingleParam<Rest extends string> = Rest extends `${infer ParamName}(${infer _Constraint})${infer _After}` ? { [K in ParamName]: string } : Rest extends `${infer ParamName}-:${infer _NextParam}` ? { [K in ParamName]: string } : Rest extends `${infer ParamName}.:${infer _NextParam}` ? { [K in ParamName]: string } : Rest extends `${infer ParamName}?/${infer _After}` ? { [K in ParamName]?: string } : Rest extends `${infer ParamName}?-${infer _After}` ? { [K in ParamName]?: string } : Rest extends `${infer ParamName}?.${infer _After}` ? { [K in ParamName]?: string } : Rest extends `${infer ParamName}?#${infer _After}` ? { [K in ParamName]?: string } : Rest extends `${infer ParamName}?:${infer _After}` ? { [K in ParamName]?: string } : Rest extends `${infer ParamName}/${infer _After}` ? { [K in ParamName]: string } : Rest extends `${infer ParamName}-${infer _After}` ? { [K in ParamName]: string } : Rest extends `${infer ParamName}.${infer _After}` ? { [K in ParamName]: string } : Rest extends `${infer ParamName}#${infer _After}` ? { [K in ParamName]: string } : Rest extends `${infer ParamName}:${infer _After}` ? { [K in ParamName]: string } : Rest extends `${infer ParamName}+${infer _After}` ? { [K in ParamName]: string[] } : Rest extends `${infer ParamName}*${infer _After}` ? { [K in ParamName]?: string[] } : Rest extends `${infer ParamName}?${infer _After}` ? { [K in ParamName]?: string } : Rest extends string ? Rest extends "" ? {} : Rest extends `${infer ParamName}?` ? { [K in ParamName]?: string } : Rest extends `${infer ParamName}+` ? { [K in ParamName]: string[] } : Rest extends `${infer ParamName}*` ? { [K in ParamName]?: string[] } : { [K in Rest]: string } : {}; /** * Remove the first parameter from path to continue parsing * Enhanced to handle Express 5 patterns and optional parameters * Handles patterns like :from-:to by removing just :from and keeping -:to * Order matters: regex constraints must be handled before repeating parameters */ type RemoveFirstParam<Path extends string> = Path extends `${infer Before}:${infer Rest}` ? Rest extends `${infer _ParamName}(${infer _Constraint})${infer After}` ? `${Before}${After}` : Rest extends `${infer _ParamName}-:${infer After}` ? `${Before}-:${After}` : Rest extends `${infer _ParamName}.:${infer After}` ? `${Before}.:${After}` : Rest extends `${infer _ParamName}?/${infer After}` ? `${Before}/${After}` : Rest extends `${infer _ParamName}?-${infer After}` ? `${Before}${After}` : Rest extends `${infer _ParamName}?.${infer After}` ? `${Before}${After}` : Rest extends `${infer _ParamName}?#${infer After}` ? `${Before}${After}` : Rest extends `${infer _ParamName}?:${infer After}` ? `${Before}:${After}` : Rest extends `${infer _ParamName}/${infer After}` ? `${Before}/${After}` : Rest extends `${infer _ParamName}-${infer After}` ? `${Before}${After}` : Rest extends `${infer _ParamName}.${infer After}` ? `${Before}${After}` : Rest extends `${infer _ParamName}#${infer After}` ? `${Before}${After}` : Rest extends `${infer _ParamName}:${infer After}` ? `${Before}:${After}` : Rest extends `${infer _ParamName}+${infer After}` ? `${Before}${After}` : Rest extends `${infer _ParamName}*${infer After}` ? `${Before}${After}` : Rest extends `${infer _ParamName}?${infer After}` ? `${Before}${After}` : Before : Path; /** * Count wildcards to assign proper numeric indices with recursion depth limit */ type CountWildcards<Path extends string, Count extends string = "0"> = Path extends `${infer _Before}*${infer Rest}` ? CountWildcards<Rest, IncrementWildcard<Count>> : Count; /** * Helper type to increment wildcard count as string */ type IncrementWildcard<T extends string> = T extends "0" ? "1" : T extends "1" ? "2" : T extends "2" ? "3" : T extends "3" ? "4" : T extends "4" ? "5" : T extends "5" ? "6" : T extends "6" ? "7" : T extends "7" ? "8" : T extends "8" ? "9" : "10"; /** * Express middleware that adds custom properties to the request object and/or response locals. * * @template TReq - The shape of the properties added to the request object. * @template TLocals - The shape of the properties added to response.locals. * @param req - The Express request object, extended with TReq. * @param res - The Express response object with typed locals. * @param next - The next middleware function. */ type TypedMiddleware<TReq extends Record<string, any> = {}, TLocals extends Record<string, any> = {}> = (req: Request & TReq, res: Response<any, TLocals>, next: NextFunction) => void | Promise<void>; /** * Simplified TypedMiddleware for request-only extensions (backward compatibility) */ type RequestOnlyMiddleware<TReq extends Record<string, any>> = TypedMiddleware<TReq, {}>; /** * Simplified TypedMiddleware for response locals-only extensions */ type LocalsOnlyMiddleware<TLocals extends Record<string, any>> = TypedMiddleware<{}, TLocals>; type InferMiddlewareProps<T extends readonly TypedMiddleware<any, any>[]> = T extends readonly [infer First, ...infer Rest] ? First extends TypedMiddleware<infer FirstReq, any> ? Rest extends readonly TypedMiddleware<any, any>[] ? FirstReq & InferMiddlewareProps<Rest> : FirstReq : {} : {}; type InferMiddlewareLocals<T extends readonly TypedMiddleware<any, any>[]> = T extends readonly [infer First, ...infer Rest] ? First extends TypedMiddleware<any, infer FirstLocals> ? Rest extends readonly TypedMiddleware<any, any>[] ? FirstLocals & InferMiddlewareLocals<Rest> : FirstLocals : {} : {}; type SchemaRequest<Path extends string = string, BodySchema extends AnyStandardSchema | unknown = unknown, QuerySchema extends AnyStandardSchema | unknown = unknown, MiddlewareProps extends Record<string, any> = {}> = Omit<Request, "params" | "query" | "body"> & { params: ExtractRouteParams<Path>; body: BodySchema extends unknown ? InferSchemaOutput<BodySchema> : unknown; query: QuerySchema extends unknown ? InferSchemaOutput<QuerySchema> : unknown; } & MiddlewareProps; type SchemaRouteHandler<Path extends string = string, BodySchema extends AnyStandardSchema | unknown = unknown, QuerySchema extends AnyStandardSchema | unknown = unknown, MiddlewareProps extends Record<string, any> = {}, ResponseLocals extends Record<string, any> = {}> = (req: SchemaRequest<Path, BodySchema, QuerySchema, MiddlewareProps>, res: Response<any, ResponseLocals>, next?: NextFunction) => void | undefined | Promise<void | undefined> | Response | Promise<Response> | Promise<Response | undefined>; /** * Options for defining a typed route, including schemas and middleware. * * @template BodySchema - Schema for request body validation. * @template QuerySchema - Schema for query parameter validation. * @property bodySchema - Optional schema for validating the request body. * @property querySchema - Optional schema for validating the query string. * @property middleware - Optional array of TypedMiddleware for this route. */ interface RouteOptions<BodySchema extends AnyStandardSchema | unknown = unknown, QuerySchema extends AnyStandardSchema | unknown = unknown> { bodySchema?: BodySchema; querySchema?: QuerySchema; middleware?: TypedMiddleware<any, any>[]; tags?: string[]; description?: string; summary?: string; deprecated?: boolean; responseSchema?: AnyStandardSchema; /** Exclude this route from the generated OpenAPI spec entirely. */ hidden?: boolean; } type DocMeta = Pick<RouteOptions<unknown, unknown>, "tags" | "summary" | "description" | "deprecated" | "responseSchema" | "hidden">; type HttpMethod = "get" | "post" | "put" | "delete" | "patch" | "options" | "head" | "all"; interface DocsOptions { title?: string; version?: string; description?: string; servers?: Array<{ url: string; description?: string; }>; /** * Override the Scalar CDN URL. Use this to pin a specific version or * self-host the Scalar bundle to avoid the external CDN dependency. * @default "https://cdn.jsdelivr.net/npm/@scalar/api-reference" */ cdnUrl?: string; /** * File path to write the OpenAPI spec to whenever it is generated. * Enables `openapi-typescript --watch` in development — the tool watches * the file and regenerates your client types automatically as routes change. * * The file is written once at startup, so it contains route **schemas only** * — never captured response examples. This keeps real response data (which * may include PII) out of any file you might commit or share. * * @example * // docs options * { specOutputPath: './openapi.json' } * * // then in a separate terminal (or via concurrently in package.json): * // npx openapi-typescript ./openapi.json -o ./src/client.d.ts --watch */ specOutputPath?: string; /** * Learn response shapes from live traffic and add them to the docs. The * library observes real responses and **infers a JSON Schema** from them * (field names, types, nullability, required vs optional) — so the docs and * generated client types reflect what your API actually returns. * * Modes: * - `true` (default) — **redacted**: infer the schema only. Real values are * discarded at capture time, so no user data is ever stored or shown. Safe * to expose. * - `"live"` — infer the schema **and** attach a real captured response as an * example. ⚠️ Examples contain actual data (emails, tokens, IDs). Only use * for trusted/internal docs. * - `false` — don't observe responses at all. * * Use the per-route `hidden: true` option to exclude individual sensitive * routes regardless of mode. * * @default true */ sampleResponses?: boolean | "live"; } interface RouteMetadata { method: HttpMethod; path: string; bodySchema?: AnyStandardSchema; querySchema?: AnyStandardSchema; tags?: string[]; description?: string; summary?: string; deprecated?: boolean; responseSchema?: AnyStandardSchema; hidden?: boolean; responseSamples: Map<number, { schema: Record<string, any>; example?: unknown; }>; } declare function inferJsonSchema(value: unknown): Record<string, any>; /** * Extra properties that middleware has added to the Express `req` object. * * Starts as `{}` (nothing added yet) and widens automatically with every * `.useMiddleware()` call. After `router.useMiddleware(authMiddleware)` where * `authMiddleware` contributes `{ userId: string }`, this becomes `{ userId: string }`. * * You will see this type in router hover text — it is the accumulating "req additions" slot. */ type AdditionalReqProps = {}; /** * Extra properties that middleware has added to `res.locals`. * * Starts as `{}` and widens automatically with every `.useMiddleware()` call, * mirroring the `TLocals` parameter of each `TypedMiddleware` you attach. */ type AdditionalLocals = {}; /** * A strongly-typed Express router. The two generic params accumulate as * middleware is added via `.useMiddleware()`. * * @typeParam Req - Extra properties on `req` contributed by middleware. Starts as {@link AdditionalReqProps}. * @typeParam Locals - Extra properties on `res.locals` contributed by middleware. Starts as {@link AdditionalLocals}. */ declare class TypedRouter<Req extends Record<string, any> = AdditionalReqProps, Locals extends Record<string, any> = AdditionalLocals> { private router; private routes; private mountedRouters; private sampleMode; private scheduleSpecWrite?; constructor(); /** * Add typed middleware that extends the request with additional properties * and/or adds properties to response.locals */ /** * Add typed middleware to the router. * This middleware will apply to all routes defined after this call. * * @template TReq - Type extensions for the request object * @template TLocals - Type extensions for response.locals * @param middleware - The typed middleware function * @returns A new router instance with updated types */ useMiddleware<TReq extends Record<string, any> = {}, TLocals extends Record<string, any> = {}>(middleware: TypedMiddleware<TReq, TLocals>): TypedRouter<Req & TReq, Locals & TLocals>; /** * Get the underlying Express router, typed as a RequestHandler so it can be * passed directly to app.use() without a cast in Express 5. */ getRouter(): express.Router & express.RequestHandler; /** * Mount middleware or a sub-router at an optional path prefix. * * When passed the result of another TypedRouter's .getRouter(), it is * automatically recognised and tracked for .docs() — no extra wiring needed. * * @example * // v1.routes.ts — pass TypedRouter instances directly, no .getRouter() needed * export const v1Routes = createTypedRouter() * * v1Routes.use('/products', productRoutes) // tracked ✓ * v1Routes.use('/profile', profileRoutes) // tracked ✓ * v1Routes.use('/', callbackRouter) // plain Express, also works * * app.use('/v1', v1Routes.getRouter()) * app.use('/docs', v1Routes.docs({ title: 'My API' })) // just works */ use(path: string, ...handlers: Array<express.RequestHandler | express.Router | TypedRouter<any, any>>): TypedRouter<Req, Locals>; use(...handlers: Array<express.RequestHandler | express.Router | TypedRouter<any, any>>): TypedRouter<Req, Locals>; /** * Mount a TypedRouter at a path prefix, registering it both on the Express * router and in the docs registry so .docs() picks it up automatically. * * @example * const v1 = createTypedRouter() * .mount('/products', productRoutes) * .mount('/profile', profileRoutes) * .mount('/supplier', supplierRoutes) * * app.use('/v1', v1.getRouter()) * app.use('/docs', v1.docs({ title: 'My API' })) */ mount(prefix: string, router: TypedRouter<any, any>): TypedRouter<Req, Locals>; mount(router: TypedRouter<any, any>): TypedRouter<Req, Locals>; /** * Returns the collected route metadata for this router, including all * sub-routers registered via .mount() with their prefixes applied. * Used internally by .docs() and by createDocs() for multi-router merging. */ getRouteMetadata(): RouteMetadata[]; /** * Turn on response observation for this router and every router mounted under * it. Called by .docs() and createDocs() so it happens only when docs are * actually generated. The visited set guards against mount cycles. * @internal Public only so createDocs() can reach it; not part of the API. */ enableSampling(mode?: "redacted" | "live", writer?: () => void, visited?: Set<TypedRouter<any, any>>): void; /** * Record a sub-router for docs, de-duplicating identical (prefix, router) * pairs and propagating the sample mode if docs were already requested. */ private trackMounted; /** * Seed in-memory response schemas from a previously written spec, so a server * restart doesn't reset the docs/spec file to empty. Only fills statuses we * haven't already observed this process, and never overwrites fresher data. * @internal */ hydrateResponses(spec: any, prefix?: string, visited?: Set<TypedRouter<any, any>>): void; /** * Returns an Express router that serves OpenAPI docs. * Mount it anywhere on your app — routes are auto-discovered. * * @example * app.use('/docs', router.docs({ title: 'My API', version: '1.0.0' })) * // GET /docs → Scalar UI * // GET /docs/openapi.json → raw OpenAPI 3.1 spec */ docs(options?: DocsOptions): express.Router & express.RequestHandler; get<Path extends string>(path: Path, handler: SchemaRouteHandler<Path, unknown, unknown, Req, Locals>): TypedRouter<Req, Locals>; get<Path extends string, BodySchema extends AnyStandardSchema | unknown, QuerySchema extends AnyStandardSchema | unknown>(path: Path, options: RouteOptions<BodySchema, QuerySchema>, handler: SchemaRouteHandler<Path, BodySchema, QuerySchema, Req, Locals>): TypedRouter<Req, Locals>; get<Path extends string, Middleware extends readonly TypedMiddleware<any, any>[]>(path: Path, options: { middleware: Middleware; }, handler: SchemaRouteHandler<Path, unknown, unknown, Req & InferMiddlewareProps<Middleware>, Locals & InferMiddlewareLocals<Middleware>>): TypedRouter<Req, Locals>; get<Path extends string, BodySchema extends AnyStandardSchema | unknown, QuerySchema extends AnyStandardSchema | unknown, M extends TypedMiddleware<any, any>[]>(path: Path, options: RouteOptions<BodySchema, QuerySchema> & { middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, BodySchema, QuerySchema, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; post<Path extends string, BodySchema extends AnyStandardSchema, QuerySchema extends AnyStandardSchema | unknown, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { bodySchema: BodySchema; querySchema?: QuerySchema; middleware: [...M]; }, handler: SchemaRouteHandler<Path, BodySchema, QuerySchema, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; post<Path extends string, BodySchema extends AnyStandardSchema, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { bodySchema: BodySchema; middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, BodySchema, unknown, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; post<Path extends string, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, unknown, unknown, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; post<Path extends string, BodySchema extends AnyStandardSchema | unknown, QuerySchema extends AnyStandardSchema | unknown>(path: Path, options: RouteOptions<BodySchema, QuerySchema>, handler: SchemaRouteHandler<Path, BodySchema, QuerySchema, Req, Locals>): TypedRouter<Req, Locals>; post<Path extends string>(path: Path, handler: SchemaRouteHandler<Path, unknown, unknown, Req, Locals>): TypedRouter<Req, Locals>; put<Path extends string, BodySchema extends AnyStandardSchema, QuerySchema extends AnyStandardSchema | unknown, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { bodySchema: BodySchema; querySchema?: QuerySchema; middleware: [...M]; }, handler: SchemaRouteHandler<Path, BodySchema, QuerySchema, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; put<Path extends string, BodySchema extends AnyStandardSchema, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { bodySchema: BodySchema; middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, BodySchema, unknown, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; put<Path extends string, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, unknown, unknown, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; put<Path extends string, BodySchema extends AnyStandardSchema | unknown, QuerySchema extends AnyStandardSchema | unknown>(path: Path, options: RouteOptions<BodySchema, QuerySchema>, handler: SchemaRouteHandler<Path, BodySchema, QuerySchema, Req, Locals>): TypedRouter<Req, Locals>; put<Path extends string>(path: Path, handler: SchemaRouteHandler<Path, unknown, unknown, Req, Locals>): TypedRouter<Req, Locals>; patch<Path extends string, BodySchema extends AnyStandardSchema, QuerySchema extends AnyStandardSchema | unknown, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { bodySchema: BodySchema; querySchema?: QuerySchema; middleware: [...M]; }, handler: SchemaRouteHandler<Path, BodySchema, QuerySchema, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; patch<Path extends string, BodySchema extends AnyStandardSchema, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { bodySchema: BodySchema; middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, BodySchema, unknown, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; patch<Path extends string, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, unknown, unknown, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; patch<Path extends string, BodySchema extends AnyStandardSchema | unknown, QuerySchema extends AnyStandardSchema | unknown>(path: Path, options: RouteOptions<BodySchema, QuerySchema>, handler: SchemaRouteHandler<Path, BodySchema, QuerySchema, Req, Locals>): TypedRouter<Req, Locals>; patch<Path extends string>(path: Path, handler: SchemaRouteHandler<Path, unknown, unknown, Req, Locals>): TypedRouter<Req, Locals>; delete<Path extends string, QuerySchema extends AnyStandardSchema | unknown, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { querySchema: QuerySchema; middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, unknown, QuerySchema, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; delete<Path extends string, QuerySchema extends AnyStandardSchema | unknown>(path: Path, options: { querySchema: QuerySchema; }, handler: SchemaRouteHandler<Path, unknown, QuerySchema, Req, Locals>): TypedRouter<Req, Locals>; delete<Path extends string>(path: Path, options: DocMeta, handler: SchemaRouteHandler<Path, unknown, unknown, Req, Locals>): TypedRouter<Req, Locals>; delete<Path extends string, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, unknown, unknown, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; delete<Path extends string>(path: Path, handler: SchemaRouteHandler<Path, unknown, unknown, Req, Locals>): TypedRouter<Req, Locals>; options<Path extends string, QuerySchema extends AnyStandardSchema | unknown, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { querySchema: QuerySchema; middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, unknown, QuerySchema, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; options<Path extends string, QuerySchema extends AnyStandardSchema | unknown>(path: Path, options: { querySchema: QuerySchema; }, handler: SchemaRouteHandler<Path, unknown, QuerySchema, Req, Locals>): TypedRouter<Req, Locals>; options<Path extends string>(path: Path, options: DocMeta, handler: SchemaRouteHandler<Path, unknown, unknown, Req, Locals>): TypedRouter<Req, Locals>; options<Path extends string, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, unknown, unknown, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; options<Path extends string>(path: Path, handler: SchemaRouteHandler<Path, unknown, unknown, Req, Locals>): TypedRouter<Req, Locals>; head<Path extends string, QuerySchema extends AnyStandardSchema | unknown, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { querySchema: QuerySchema; middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, unknown, QuerySchema, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; head<Path extends string, QuerySchema extends AnyStandardSchema | unknown>(path: Path, options: { querySchema: QuerySchema; }, handler: SchemaRouteHandler<Path, unknown, QuerySchema, Req, Locals>): TypedRouter<Req, Locals>; head<Path extends string>(path: Path, options: DocMeta, handler: SchemaRouteHandler<Path, unknown, unknown, Req, Locals>): TypedRouter<Req, Locals>; head<Path extends string, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, unknown, unknown, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; head<Path extends string>(path: Path, handler: SchemaRouteHandler<Path, unknown, unknown, Req, Locals>): TypedRouter<Req, Locals>; all<Path extends string, BodySchema extends AnyStandardSchema, QuerySchema extends AnyStandardSchema | unknown, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { bodySchema: BodySchema; querySchema?: QuerySchema; middleware: [...M]; }, handler: SchemaRouteHandler<Path, BodySchema, QuerySchema, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; all<Path extends string, BodySchema extends AnyStandardSchema, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { bodySchema: BodySchema; middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, BodySchema, unknown, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; all<Path extends string, QuerySchema extends AnyStandardSchema | unknown, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { querySchema: QuerySchema; middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, unknown, QuerySchema, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; all<Path extends string, BodySchema extends AnyStandardSchema | unknown, QuerySchema extends AnyStandardSchema | unknown>(path: Path, options: RouteOptions<BodySchema, QuerySchema>, handler: SchemaRouteHandler<Path, BodySchema, QuerySchema, Req, Locals>): TypedRouter<Req, Locals>; all<Path extends string, M extends TypedMiddleware<any, any>[]>(path: Path, options: DocMeta & { middleware: [...M]; }, // Using tuple spread pattern handler: SchemaRouteHandler<Path, unknown, unknown, Req & InferMiddlewareProps<readonly [...M]>, // Make it readonly for type inference // Make it readonly for type inference Locals & InferMiddlewareLocals<readonly [...M]>>): TypedRouter<Req, Locals>; all<Path extends string>(path: Path, handler: SchemaRouteHandler<Path, unknown, unknown, Req, Locals>): TypedRouter<Req, Locals>; private registerRoute; private createBodyValidationMiddleware; private createQueryValidationMiddleware; } /** * Create a new strongly-typed Express router instance. * * This is the simplest way to get started with @minisylar/express-typed-router. * * @example * import { createTypedRouter } from '@minisylar/express-typed-router'; * * // Create a router and add a typed GET route * const router = createTypedRouter(); * router.get('/hello/:name', (req, res) => { * // req.params.name is typed as string * res.json({ message: `Hello, ${req.params.name}!` }); * }); * * // Use with Express * import express from 'express'; * const app = express(); * app.use('/api', router.getRouter()); */ declare function createTypedRouter<Req extends Record<string, any> = AdditionalReqProps, Locals extends Record<string, any> = AdditionalLocals>(): TypedRouter<Req, Locals>; /** * Configuration options for createTypedRouterWithConfig. * * @property validateInput - (Future) Whether to enable global input validation. * @property errorHandler - Optional global error handler middleware for the router. */ interface RouterConfig { validateInput?: boolean; errorHandler?: (error: any, req: Request, res: Response, next: NextFunction) => void; } /** * Create a new typed router with optional configuration. * * Use this if you want to add a global error handler or future global options. * * @param config - Optional configuration for the router (e.g. error handler). * @returns A new TypedRouter instance. * * @example * import { createTypedRouterWithConfig } from '@minisylar/express-typed-router'; * * const router = createTypedRouterWithConfig({ * errorHandler: (err, req, res, next) => { * res.status(500).json({ error: 'Something went wrong', details: err }); * } * }); */ declare function createTypedRouterWithConfig<Req extends Record<string, any> = AdditionalReqProps, Locals extends Record<string, any> = AdditionalLocals>(config?: RouterConfig): TypedRouter<Req, Locals>; /** * Create a new typed router with pre-configured middleware. * * This is useful for setting up router-level middleware in a single call. * * @param middleware - One or more TypedMiddleware functions to apply to all routes. * @returns A new TypedRouter instance with the middleware applied. * * @example * import { createTypedRouterWithMiddleware } from '@minisylar/express-typed-router'; * * const router = createTypedRouterWithMiddleware(authMiddleware, loggingMiddleware); */ declare function createTypedRouterWithMiddleware<T extends Record<string, any>>(...middleware: TypedMiddleware<any, any>[]): TypedRouter<T>; /** * An entry for createDocs(). Either a bare TypedRouter (no prefix prepended) * or an object with an explicit prefix matching the mount point in app.use(). * * @example * // Routes defined as /users/:id — mount prefix prepends /api * { prefix: '/api', router: usersRouter } * * // Routes already include the full path — no prefix needed * authRouter */ type RouterDocEntry = TypedRouter<any, any> | { prefix: string; router: TypedRouter<any, any>; }; /** * Create a unified OpenAPI docs endpoint that merges routes from multiple * TypedRouter instances. Use this when routes are split across files. * * @example * // users.router.ts — routes like /users, /users/:id * export const usersRouter = createTypedRouter(); * * // auth.router.ts — routes like /login, /logout * export const authRouter = createTypedRouter(); * * // app.ts * app.use('/api', usersRouter.getRouter()); * app.use('/api', authRouter.getRouter()); * app.use('/docs', createDocs( * [ * { prefix: '/api', router: usersRouter }, * { prefix: '/api', router: authRouter }, * ], * { title: 'My API', version: '1.0.0' } * )); */ declare function createDocs(routers: RouterDocEntry | RouterDocEntry[], options?: DocsOptions): express.Router & express.RequestHandler; //#endregion export { AdditionalLocals, AdditionalReqProps, AnyStandardSchema, DocsOptions, ExtractRouteParams, HttpMethod, InferInput, InferOutput, InferSchemaOutput, LocalsOnlyMiddleware, RequestOnlyMiddleware, RouteOptions, RouterConfig, RouterDocEntry, SafeParseResult, SchemaRequest, SchemaRouteHandler, TypedMiddleware, TypedRouter, createDocs, createTypedRouter, createTypedRouterWithConfig, createTypedRouterWithMiddleware, inferJsonSchema, isSchemaError, parseSchema, safeParseSchema };