@tyravel/http 0.1.0

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.
@@ -0,0 +1,10 @@
1
+ export { TyravelRequest } from './request.js';
2
+ export type { SessionContract } from './session-contract.js';
3
+ export { Response, ResponseFactory } from './response.js';
4
+ export { createRouter, RouteNotFoundException, Router, } from './router.js';
5
+ export type { Groupable, MiddlewareGroupable, Routable, RouteScope, ScopedRouteRegistrar, } from './router.js';
6
+ export { joinRoutePaths, RouteGroupBuilder, } from './route-group.js';
7
+ export { MiddlewareNotFoundException, MiddlewareRegistry, } from './middleware-registry.js';
8
+ export type { MiddlewareInput } from './middleware-registry.js';
9
+ export type { HttpMethod, Middleware, RouteDefinition, RouteHandler, RouteParams, } from './types.js';
10
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,cAAc,CAAC;AAC9C,YAAY,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC;AAC7D,OAAO,EAAE,QAAQ,EAAE,eAAe,EAAE,MAAM,eAAe,CAAC;AAC1D,OAAO,EACL,YAAY,EACZ,sBAAsB,EACtB,MAAM,GACP,MAAM,aAAa,CAAC;AACrB,YAAY,EACV,SAAS,EACT,mBAAmB,EACnB,QAAQ,EACR,UAAU,EACV,oBAAoB,GACrB,MAAM,aAAa,CAAC;AACrB,OAAO,EACL,cAAc,EACd,iBAAiB,GAClB,MAAM,kBAAkB,CAAC;AAC1B,OAAO,EACL,2BAA2B,EAC3B,kBAAkB,GACnB,MAAM,0BAA0B,CAAC;AAClC,YAAY,EAAE,eAAe,EAAE,MAAM,0BAA0B,CAAC;AAChE,YAAY,EACV,UAAU,EACV,UAAU,EACV,eAAe,EACf,YAAY,EACZ,WAAW,GACZ,MAAM,YAAY,CAAC"}
package/dist/index.js ADDED
@@ -0,0 +1,6 @@
1
+ export { TyravelRequest } from './request.js';
2
+ export { Response, ResponseFactory } from './response.js';
3
+ export { createRouter, RouteNotFoundException, Router, } from './router.js';
4
+ export { joinRoutePaths, RouteGroupBuilder, } from './route-group.js';
5
+ export { MiddlewareNotFoundException, MiddlewareRegistry, } from './middleware-registry.js';
6
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,cAAc,CAAC;AAE9C,OAAO,EAAE,QAAQ,EAAE,eAAe,EAAE,MAAM,eAAe,CAAC;AAC1D,OAAO,EACL,YAAY,EACZ,sBAAsB,EACtB,MAAM,GACP,MAAM,aAAa,CAAC;AAQrB,OAAO,EACL,cAAc,EACd,iBAAiB,GAClB,MAAM,kBAAkB,CAAC;AAC1B,OAAO,EACL,2BAA2B,EAC3B,kBAAkB,GACnB,MAAM,0BAA0B,CAAC"}
@@ -0,0 +1,13 @@
1
+ import type { Middleware } from './types.js';
2
+ export declare class MiddlewareNotFoundException extends Error {
3
+ constructor(name: string);
4
+ }
5
+ export type MiddlewareInput = Middleware | string;
6
+ export declare class MiddlewareRegistry {
7
+ private aliases;
8
+ alias(name: string, middleware: Middleware): this;
9
+ has(name: string): boolean;
10
+ resolve(input: MiddlewareInput): Middleware;
11
+ resolveMany(inputs: MiddlewareInput[]): Middleware[];
12
+ }
13
+ //# sourceMappingURL=middleware-registry.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"middleware-registry.d.ts","sourceRoot":"","sources":["../src/middleware-registry.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,YAAY,CAAC;AAE7C,qBAAa,2BAA4B,SAAQ,KAAK;gBACxC,IAAI,EAAE,MAAM;CAIzB;AAED,MAAM,MAAM,eAAe,GAAG,UAAU,GAAG,MAAM,CAAC;AAElD,qBAAa,kBAAkB;IAC7B,OAAO,CAAC,OAAO,CAAiC;IAEhD,KAAK,CAAC,IAAI,EAAE,MAAM,EAAE,UAAU,EAAE,UAAU,GAAG,IAAI;IAKjD,GAAG,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO;IAI1B,OAAO,CAAC,KAAK,EAAE,eAAe,GAAG,UAAU;IAa3C,WAAW,CAAC,MAAM,EAAE,eAAe,EAAE,GAAG,UAAU,EAAE;CAGrD"}
@@ -0,0 +1,30 @@
1
+ export class MiddlewareNotFoundException extends Error {
2
+ constructor(name) {
3
+ super(`Middleware not found: ${name}`);
4
+ this.name = 'MiddlewareNotFoundException';
5
+ }
6
+ }
7
+ export class MiddlewareRegistry {
8
+ aliases = new Map();
9
+ alias(name, middleware) {
10
+ this.aliases.set(name, middleware);
11
+ return this;
12
+ }
13
+ has(name) {
14
+ return this.aliases.has(name);
15
+ }
16
+ resolve(input) {
17
+ if (typeof input !== 'string') {
18
+ return input;
19
+ }
20
+ const middleware = this.aliases.get(input);
21
+ if (!middleware) {
22
+ throw new MiddlewareNotFoundException(input);
23
+ }
24
+ return middleware;
25
+ }
26
+ resolveMany(inputs) {
27
+ return inputs.map((input) => this.resolve(input));
28
+ }
29
+ }
30
+ //# sourceMappingURL=middleware-registry.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"middleware-registry.js","sourceRoot":"","sources":["../src/middleware-registry.ts"],"names":[],"mappings":"AAEA,MAAM,OAAO,2BAA4B,SAAQ,KAAK;IACpD,YAAY,IAAY;QACtB,KAAK,CAAC,yBAAyB,IAAI,EAAE,CAAC,CAAC;QACvC,IAAI,CAAC,IAAI,GAAG,6BAA6B,CAAC;IAC5C,CAAC;CACF;AAID,MAAM,OAAO,kBAAkB;IACrB,OAAO,GAAG,IAAI,GAAG,EAAsB,CAAC;IAEhD,KAAK,CAAC,IAAY,EAAE,UAAsB;QACxC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;QACnC,OAAO,IAAI,CAAC;IACd,CAAC;IAED,GAAG,CAAC,IAAY;QACd,OAAO,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IAChC,CAAC;IAED,OAAO,CAAC,KAAsB;QAC5B,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;YAC9B,OAAO,KAAK,CAAC;QACf,CAAC;QAED,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC3C,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,MAAM,IAAI,2BAA2B,CAAC,KAAK,CAAC,CAAC;QAC/C,CAAC;QAED,OAAO,UAAU,CAAC;IACpB,CAAC;IAED,WAAW,CAAC,MAAyB;QACnC,OAAO,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC;IACpD,CAAC;CACF"}
@@ -0,0 +1,23 @@
1
+ import type { RouteParams } from './types.js';
2
+ import type { SessionContract } from './session-contract.js';
3
+ export declare class TyravelRequest {
4
+ readonly raw: Request;
5
+ readonly params: RouteParams;
6
+ readonly routeName?: string | undefined;
7
+ constructor(raw: Request, params?: RouteParams, routeName?: string | undefined);
8
+ session?: SessionContract;
9
+ user: unknown;
10
+ get method(): string;
11
+ get url(): URL;
12
+ get path(): string;
13
+ get headers(): Headers;
14
+ param(name: string, fallback?: string): string | undefined;
15
+ query(name: string, fallback?: string): string | undefined;
16
+ json<T = unknown>(): Promise<T>;
17
+ text(): Promise<string>;
18
+ formData(): Promise<FormData>;
19
+ input<T = string>(key: string): Promise<T | undefined>;
20
+ header(name: string, fallback?: string): string | undefined;
21
+ bearerToken(): string | undefined;
22
+ }
23
+ //# sourceMappingURL=request.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"request.d.ts","sourceRoot":"","sources":["../src/request.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAC9C,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC;AAE7D,qBAAa,cAAc;aAEP,GAAG,EAAE,OAAO;aACZ,MAAM,EAAE,WAAW;aACnB,SAAS,CAAC,EAAE,MAAM;gBAFlB,GAAG,EAAE,OAAO,EACZ,MAAM,GAAE,WAAgB,EACxB,SAAS,CAAC,EAAE,MAAM,YAAA;IAGpC,OAAO,CAAC,EAAE,eAAe,CAAC;IAC1B,IAAI,EAAE,OAAO,CAAQ;IAErB,IAAI,MAAM,IAAI,MAAM,CAEnB;IAED,IAAI,GAAG,IAAI,GAAG,CAEb;IAED,IAAI,IAAI,IAAI,MAAM,CAEjB;IAED,IAAI,OAAO,IAAI,OAAO,CAErB;IAED,KAAK,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS;IAI1D,KAAK,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS;IAIpD,IAAI,CAAC,CAAC,GAAG,OAAO,KAAK,OAAO,CAAC,CAAC,CAAC;IAI/B,IAAI,IAAI,OAAO,CAAC,MAAM,CAAC;IAIvB,QAAQ,IAAI,OAAO,CAAC,QAAQ,CAAC;IAInC,KAAK,CAAC,CAAC,GAAG,MAAM,EAAE,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,CAAC,GAAG,SAAS,CAAC;IAiBtD,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS;IAI3D,WAAW,IAAI,MAAM,GAAG,SAAS;CAOlC"}
@@ -0,0 +1,61 @@
1
+ export class TyravelRequest {
2
+ raw;
3
+ params;
4
+ routeName;
5
+ constructor(raw, params = {}, routeName) {
6
+ this.raw = raw;
7
+ this.params = params;
8
+ this.routeName = routeName;
9
+ }
10
+ session;
11
+ user = null;
12
+ get method() {
13
+ return this.raw.method;
14
+ }
15
+ get url() {
16
+ return new URL(this.raw.url);
17
+ }
18
+ get path() {
19
+ return this.url.pathname;
20
+ }
21
+ get headers() {
22
+ return this.raw.headers;
23
+ }
24
+ param(name, fallback) {
25
+ return this.params[name] ?? fallback;
26
+ }
27
+ query(name, fallback) {
28
+ return this.url.searchParams.get(name) ?? fallback;
29
+ }
30
+ async json() {
31
+ return this.raw.json();
32
+ }
33
+ async text() {
34
+ return this.raw.text();
35
+ }
36
+ async formData() {
37
+ return this.raw.formData();
38
+ }
39
+ input(key) {
40
+ const contentType = this.headers.get('content-type') ?? '';
41
+ if (contentType.includes('application/json')) {
42
+ return this.json().then((body) => body[key]);
43
+ }
44
+ if (contentType.includes('application/x-www-form-urlencoded') ||
45
+ contentType.includes('multipart/form-data')) {
46
+ return this.formData().then((body) => body.get(key));
47
+ }
48
+ return Promise.resolve(undefined);
49
+ }
50
+ header(name, fallback) {
51
+ return this.raw.headers.get(name) ?? fallback;
52
+ }
53
+ bearerToken() {
54
+ const authorization = this.header('authorization');
55
+ if (!authorization?.startsWith('Bearer ')) {
56
+ return undefined;
57
+ }
58
+ return authorization.slice('Bearer '.length);
59
+ }
60
+ }
61
+ //# sourceMappingURL=request.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"request.js","sourceRoot":"","sources":["../src/request.ts"],"names":[],"mappings":"AAGA,MAAM,OAAO,cAAc;IAEP;IACA;IACA;IAHlB,YACkB,GAAY,EACZ,SAAsB,EAAE,EACxB,SAAkB;QAFlB,QAAG,GAAH,GAAG,CAAS;QACZ,WAAM,GAAN,MAAM,CAAkB;QACxB,cAAS,GAAT,SAAS,CAAS;IACjC,CAAC;IAEJ,OAAO,CAAmB;IAC1B,IAAI,GAAY,IAAI,CAAC;IAErB,IAAI,MAAM;QACR,OAAO,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC;IACzB,CAAC;IAED,IAAI,GAAG;QACL,OAAO,IAAI,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;IAC/B,CAAC;IAED,IAAI,IAAI;QACN,OAAO,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC;IAC3B,CAAC;IAED,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC;IAC1B,CAAC;IAED,KAAK,CAAC,IAAY,EAAE,QAAiB;QACnC,OAAO,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,QAAQ,CAAC;IACvC,CAAC;IAED,KAAK,CAAC,IAAY,EAAE,QAAiB;QACnC,OAAO,IAAI,CAAC,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,QAAQ,CAAC;IACrD,CAAC;IAED,KAAK,CAAC,IAAI;QACR,OAAO,IAAI,CAAC,GAAG,CAAC,IAAI,EAAgB,CAAC;IACvC,CAAC;IAED,KAAK,CAAC,IAAI;QACR,OAAO,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC;IACzB,CAAC;IAED,KAAK,CAAC,QAAQ;QACZ,OAAO,IAAI,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC;IAC7B,CAAC;IAED,KAAK,CAAa,GAAW;QAC3B,MAAM,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,IAAI,EAAE,CAAC;QAE3D,IAAI,WAAW,CAAC,QAAQ,CAAC,kBAAkB,CAAC,EAAE,CAAC;YAC7C,OAAO,IAAI,CAAC,IAAI,EAAqB,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;QAClE,CAAC;QAED,IACE,WAAW,CAAC,QAAQ,CAAC,mCAAmC,CAAC;YACzD,WAAW,CAAC,QAAQ,CAAC,qBAAqB,CAAC,EAC3C,CAAC;YACD,OAAO,IAAI,CAAC,QAAQ,EAAE,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAkB,CAAC,CAAC;QACxE,CAAC;QAED,OAAO,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;IACpC,CAAC;IAED,MAAM,CAAC,IAAY,EAAE,QAAiB;QACpC,OAAO,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,QAAQ,CAAC;IAChD,CAAC;IAED,WAAW;QACT,MAAM,aAAa,GAAG,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;QACnD,IAAI,CAAC,aAAa,EAAE,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;YAC1C,OAAO,SAAS,CAAC;QACnB,CAAC;QACD,OAAO,aAAa,CAAC,KAAK,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;IAC/C,CAAC;CACF"}
@@ -0,0 +1,9 @@
1
+ export declare class ResponseFactory {
2
+ json(data: unknown, init?: ResponseInit): Response;
3
+ html(body: string, init?: ResponseInit): Response;
4
+ text(body: string, init?: ResponseInit): Response;
5
+ redirect(location: string, status?: number): Response;
6
+ noContent(status?: number): Response;
7
+ }
8
+ export declare const Response: ResponseFactory;
9
+ //# sourceMappingURL=response.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"response.d.ts","sourceRoot":"","sources":["../src/response.ts"],"names":[],"mappings":"AAEA,qBAAa,eAAe;IAC1B,IAAI,CAAC,IAAI,EAAE,OAAO,EAAE,IAAI,GAAE,YAAiB,GAAG,QAAQ;IAYtD,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,GAAE,YAAiB,GAAG,QAAQ;IAYrD,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,GAAE,YAAiB,GAAG,QAAQ;IAYrD,QAAQ,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,SAAM,GAAG,QAAQ;IAIlD,SAAS,CAAC,MAAM,SAAM,GAAG,QAAQ;CAGlC;AAED,eAAO,MAAM,QAAQ,iBAAwB,CAAC"}
@@ -0,0 +1,41 @@
1
+ const WebResponse = globalThis.Response;
2
+ export class ResponseFactory {
3
+ json(data, init = {}) {
4
+ const headers = new Headers(init.headers);
5
+ if (!headers.has('content-type')) {
6
+ headers.set('content-type', 'application/json; charset=utf-8');
7
+ }
8
+ return new WebResponse(JSON.stringify(data), {
9
+ ...init,
10
+ headers,
11
+ });
12
+ }
13
+ html(body, init = {}) {
14
+ const headers = new Headers(init.headers);
15
+ if (!headers.has('content-type')) {
16
+ headers.set('content-type', 'text/html; charset=utf-8');
17
+ }
18
+ return new WebResponse(body, {
19
+ ...init,
20
+ headers,
21
+ });
22
+ }
23
+ text(body, init = {}) {
24
+ const headers = new Headers(init.headers);
25
+ if (!headers.has('content-type')) {
26
+ headers.set('content-type', 'text/plain; charset=utf-8');
27
+ }
28
+ return new WebResponse(body, {
29
+ ...init,
30
+ headers,
31
+ });
32
+ }
33
+ redirect(location, status = 302) {
34
+ return WebResponse.redirect(location, status);
35
+ }
36
+ noContent(status = 204) {
37
+ return new WebResponse(null, { status });
38
+ }
39
+ }
40
+ export const Response = new ResponseFactory();
41
+ //# sourceMappingURL=response.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"response.js","sourceRoot":"","sources":["../src/response.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,GAAG,UAAU,CAAC,QAAQ,CAAC;AAExC,MAAM,OAAO,eAAe;IAC1B,IAAI,CAAC,IAAa,EAAE,OAAqB,EAAE;QACzC,MAAM,OAAO,GAAG,IAAI,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC1C,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,EAAE,CAAC;YACjC,OAAO,CAAC,GAAG,CAAC,cAAc,EAAE,iCAAiC,CAAC,CAAC;QACjE,CAAC;QAED,OAAO,IAAI,WAAW,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE;YAC3C,GAAG,IAAI;YACP,OAAO;SACR,CAAC,CAAC;IACL,CAAC;IAED,IAAI,CAAC,IAAY,EAAE,OAAqB,EAAE;QACxC,MAAM,OAAO,GAAG,IAAI,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC1C,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,EAAE,CAAC;YACjC,OAAO,CAAC,GAAG,CAAC,cAAc,EAAE,0BAA0B,CAAC,CAAC;QAC1D,CAAC;QAED,OAAO,IAAI,WAAW,CAAC,IAAI,EAAE;YAC3B,GAAG,IAAI;YACP,OAAO;SACR,CAAC,CAAC;IACL,CAAC;IAED,IAAI,CAAC,IAAY,EAAE,OAAqB,EAAE;QACxC,MAAM,OAAO,GAAG,IAAI,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC1C,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,EAAE,CAAC;YACjC,OAAO,CAAC,GAAG,CAAC,cAAc,EAAE,2BAA2B,CAAC,CAAC;QAC3D,CAAC;QAED,OAAO,IAAI,WAAW,CAAC,IAAI,EAAE;YAC3B,GAAG,IAAI;YACP,OAAO;SACR,CAAC,CAAC;IACL,CAAC;IAED,QAAQ,CAAC,QAAgB,EAAE,MAAM,GAAG,GAAG;QACrC,OAAO,WAAW,CAAC,QAAQ,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;IAChD,CAAC;IAED,SAAS,CAAC,MAAM,GAAG,GAAG;QACpB,OAAO,IAAI,WAAW,CAAC,IAAI,EAAE,EAAE,MAAM,EAAE,CAAC,CAAC;IAC3C,CAAC;CACF;AAED,MAAM,CAAC,MAAM,QAAQ,GAAG,IAAI,eAAe,EAAE,CAAC"}
@@ -0,0 +1,53 @@
1
+ import type { MiddlewareInput } from './middleware-registry.js';
2
+ import type { Router } from './router.js';
3
+ import type { RouteHandler, RouteParams } from './types.js';
4
+ export interface RouteScope {
5
+ prefix: string;
6
+ middleware: MiddlewareInput[];
7
+ namePrefix?: string;
8
+ }
9
+ export interface Routable {
10
+ get(pattern: string, handler: RouteHandler): Routable;
11
+ post(pattern: string, handler: RouteHandler): Routable;
12
+ put(pattern: string, handler: RouteHandler): Routable;
13
+ patch(pattern: string, handler: RouteHandler): Routable;
14
+ delete(pattern: string, handler: RouteHandler): Routable;
15
+ use(...middleware: MiddlewareInput[]): Routable;
16
+ name(name: string): Routable;
17
+ url(name: string, params?: RouteParams): string;
18
+ }
19
+ export interface ScopedRouteRegistrar {
20
+ get(pattern: string, handler: RouteHandler): Routable;
21
+ post(pattern: string, handler: RouteHandler): Routable;
22
+ put(pattern: string, handler: RouteHandler): Routable;
23
+ patch(pattern: string, handler: RouteHandler): Routable;
24
+ delete(pattern: string, handler: RouteHandler): Routable;
25
+ }
26
+ export interface Groupable extends Routable {
27
+ prefix(prefix: string): MiddlewareGroupable;
28
+ namePrefix(prefix: string): MiddlewareGroupable;
29
+ group(callback: (routes: Groupable) => void): Routable;
30
+ }
31
+ export interface MiddlewareGroupable extends Groupable {
32
+ middleware(...middleware: MiddlewareInput[]): MiddlewareGroupable;
33
+ }
34
+ export declare function joinRoutePaths(...segments: string[]): string;
35
+ export declare class RouteGroupBuilder implements MiddlewareGroupable {
36
+ private readonly router;
37
+ private readonly scope;
38
+ constructor(router: Router, scope?: RouteScope);
39
+ prefix(prefix: string): MiddlewareGroupable;
40
+ middleware(...middleware: MiddlewareInput[]): MiddlewareGroupable;
41
+ namePrefix(prefix: string): MiddlewareGroupable;
42
+ group(callback: (routes: Groupable) => void): Routable;
43
+ get(pattern: string, handler: RouteHandler): Routable;
44
+ post(pattern: string, handler: RouteHandler): Routable;
45
+ put(pattern: string, handler: RouteHandler): Routable;
46
+ patch(pattern: string, handler: RouteHandler): Routable;
47
+ delete(pattern: string, handler: RouteHandler): Routable;
48
+ use(...middleware: MiddlewareInput[]): Routable;
49
+ name(name: string): Routable;
50
+ url(name: string, params?: RouteParams): string;
51
+ private add;
52
+ }
53
+ //# sourceMappingURL=route-group.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"route-group.d.ts","sourceRoot":"","sources":["../src/route-group.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,71 @@
1
+ export function joinRoutePaths(...segments) {
2
+ const parts = segments.flatMap((segment) => {
3
+ const trimmed = segment.trim();
4
+ if (!trimmed || trimmed === '/') {
5
+ return [];
6
+ }
7
+ return [trimmed.replace(/^\/+|\/+$/g, '')];
8
+ });
9
+ if (parts.length === 0) {
10
+ return '';
11
+ }
12
+ return `/${parts.join('/')}`;
13
+ }
14
+ export class RouteGroupBuilder {
15
+ router;
16
+ scope;
17
+ constructor(router, scope = { prefix: '', middleware: [] }) {
18
+ this.router = router;
19
+ this.scope = scope;
20
+ }
21
+ prefix(prefix) {
22
+ return new RouteGroupBuilder(this.router, {
23
+ ...this.scope,
24
+ prefix: joinRoutePaths(this.scope.prefix, prefix),
25
+ });
26
+ }
27
+ middleware(...middleware) {
28
+ return new RouteGroupBuilder(this.router, {
29
+ ...this.scope,
30
+ middleware: [...this.scope.middleware, ...middleware],
31
+ });
32
+ }
33
+ namePrefix(prefix) {
34
+ return new RouteGroupBuilder(this.router, {
35
+ ...this.scope,
36
+ namePrefix: prefix,
37
+ });
38
+ }
39
+ group(callback) {
40
+ this.router.runInScope(this.scope, () => callback(this));
41
+ return this.router;
42
+ }
43
+ get(pattern, handler) {
44
+ return this.add('GET', pattern, handler);
45
+ }
46
+ post(pattern, handler) {
47
+ return this.add('POST', pattern, handler);
48
+ }
49
+ put(pattern, handler) {
50
+ return this.add('PUT', pattern, handler);
51
+ }
52
+ patch(pattern, handler) {
53
+ return this.add('PATCH', pattern, handler);
54
+ }
55
+ delete(pattern, handler) {
56
+ return this.add('DELETE', pattern, handler);
57
+ }
58
+ use(...middleware) {
59
+ return this.router.use(...middleware);
60
+ }
61
+ name(name) {
62
+ return this.router.name(name);
63
+ }
64
+ url(name, params) {
65
+ return this.router.url(name, params);
66
+ }
67
+ add(method, pattern, handler) {
68
+ return this.router.addScoped(this.scope, method, pattern, handler);
69
+ }
70
+ }
71
+ //# sourceMappingURL=route-group.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"route-group.js","sourceRoot":"","sources":["../src/route-group.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,42 @@
1
+ import { type Groupable, type MiddlewareGroupable, type Routable, type RouteScope, type ScopedRouteRegistrar } from './route-group.js';
2
+ import { MiddlewareRegistry, type MiddlewareInput } from './middleware-registry.js';
3
+ import type { HttpMethod, RouteHandler, RouteParams } from './types.js';
4
+ export declare class RouteNotFoundException extends Error {
5
+ constructor(method: string, path: string);
6
+ }
7
+ export declare class Router implements Routable {
8
+ private routes;
9
+ private namedRoutes;
10
+ private globalMiddleware;
11
+ private scopeStack;
12
+ private readonly middlewareRegistry;
13
+ private handlerNormalizer;
14
+ constructor(middlewareRegistry?: MiddlewareRegistry);
15
+ getMiddlewareRegistry(): MiddlewareRegistry;
16
+ setHandlerNormalizer(normalizer: (handler: RouteHandler) => RouteHandler): this;
17
+ prefix(prefix: string): MiddlewareGroupable;
18
+ namePrefix(prefix: string): MiddlewareGroupable;
19
+ group(callback: (routes: Groupable) => void): Routable;
20
+ middleware(...middleware: MiddlewareInput[]): ScopedRouteRegistrar;
21
+ runInScope(scope: RouteScope, callback: () => void): void;
22
+ get(pattern: string, handler: RouteHandler): Routable;
23
+ post(pattern: string, handler: RouteHandler): Routable;
24
+ put(pattern: string, handler: RouteHandler): Routable;
25
+ patch(pattern: string, handler: RouteHandler): Routable;
26
+ delete(pattern: string, handler: RouteHandler): Routable;
27
+ use(...middleware: MiddlewareInput[]): Routable;
28
+ add(method: HttpMethod, pattern: string, handler: RouteHandler, middleware?: MiddlewareInput[]): Routable;
29
+ addScoped(scope: RouteScope, method: HttpMethod, pattern: string, handler: RouteHandler, middleware?: MiddlewareInput[]): Routable;
30
+ name(name: string): Routable;
31
+ url(name: string, params?: RouteParams): string;
32
+ dispatch(request: Request): Promise<Response>;
33
+ private runPipeline;
34
+ private normalizeResponse;
35
+ private resolveMiddleware;
36
+ private mergeScopes;
37
+ private compile;
38
+ private extractParams;
39
+ }
40
+ export declare function createRouter(middlewareRegistry?: MiddlewareRegistry): Router;
41
+ export type { Groupable, MiddlewareGroupable, Routable, RouteScope, ScopedRouteRegistrar, } from './route-group.js';
42
+ //# sourceMappingURL=router.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"router.d.ts","sourceRoot":"","sources":["../src/router.ts"],"names":[],"mappings":"AAEA,OAAO,EAGL,KAAK,SAAS,EACd,KAAK,mBAAmB,EACxB,KAAK,QAAQ,EACb,KAAK,UAAU,EACf,KAAK,oBAAoB,EAC1B,MAAM,kBAAkB,CAAC;AAC1B,OAAO,EACL,kBAAkB,EAClB,KAAK,eAAe,EACrB,MAAM,0BAA0B,CAAC;AAClC,OAAO,KAAK,EACV,UAAU,EAGV,YAAY,EACZ,WAAW,EACZ,MAAM,YAAY,CAAC;AAEpB,qBAAa,sBAAuB,SAAQ,KAAK;gBACnC,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM;CAIzC;AAoCD,qBAAa,MAAO,YAAW,QAAQ;IACrC,OAAO,CAAC,MAAM,CAAyB;IACvC,OAAO,CAAC,WAAW,CAAsC;IACzD,OAAO,CAAC,gBAAgB,CAAyB;IACjD,OAAO,CAAC,UAAU,CAAoB;IACtC,OAAO,CAAC,QAAQ,CAAC,kBAAkB,CAAqB;IACxD,OAAO,CAAC,iBAAiB,CAAiE;gBAE9E,kBAAkB,qBAA2B;IAIzD,qBAAqB,IAAI,kBAAkB;IAI3C,oBAAoB,CAAC,UAAU,EAAE,CAAC,OAAO,EAAE,YAAY,KAAK,YAAY,GAAG,IAAI;IAK/E,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,mBAAmB;IAI3C,UAAU,CAAC,MAAM,EAAE,MAAM,GAAG,mBAAmB;IAI/C,KAAK,CAAC,QAAQ,EAAE,CAAC,MAAM,EAAE,SAAS,KAAK,IAAI,GAAG,QAAQ;IAItD,UAAU,CAAC,GAAG,UAAU,EAAE,eAAe,EAAE,GAAG,oBAAoB;IAIlE,UAAU,CAAC,KAAK,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,IAAI,GAAG,IAAI;IASzD,GAAG,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,YAAY,GAAG,QAAQ;IAIrD,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,YAAY,GAAG,QAAQ;IAItD,GAAG,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,YAAY,GAAG,QAAQ;IAIrD,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,YAAY,GAAG,QAAQ;IAIvD,MAAM,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,YAAY,GAAG,QAAQ;IAIxD,GAAG,CAAC,GAAG,UAAU,EAAE,eAAe,EAAE,GAAG,QAAQ;IAK/C,GAAG,CACD,MAAM,EAAE,UAAU,EAClB,OAAO,EAAE,MAAM,EACf,OAAO,EAAE,YAAY,EACrB,UAAU,GAAE,eAAe,EAAO,GACjC,QAAQ;IAIX,SAAS,CACP,KAAK,EAAE,UAAU,EACjB,MAAM,EAAE,UAAU,EAClB,OAAO,EAAE,MAAM,EACf,OAAO,EAAE,YAAY,EACrB,UAAU,GAAE,eAAe,EAAO,GACjC,QAAQ;IAoBX,IAAI,CAAC,IAAI,EAAE,MAAM,GAAG,QAAQ;IAa5B,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,GAAE,WAAgB,GAAG,MAAM;IAe7C,QAAQ,CAAC,OAAO,EAAE,OAAO,GAAG,OAAO,CAAC,QAAQ,CAAC;YA4BrC,WAAW;IAyBzB,OAAO,CAAC,iBAAiB;IAIzB,OAAO,CAAC,iBAAiB;IAIzB,OAAO,CAAC,WAAW;IAWnB,OAAO,CAAC,OAAO;IAqBf,OAAO,CAAC,aAAa;CAYtB;AAED,wBAAgB,YAAY,CAAC,kBAAkB,CAAC,EAAE,kBAAkB,GAAG,MAAM,CAE5E;AAED,YAAY,EACV,SAAS,EACT,mBAAmB,EACnB,QAAQ,EACR,UAAU,EACV,oBAAoB,GACrB,MAAM,kBAAkB,CAAC"}
package/dist/router.js ADDED
@@ -0,0 +1,215 @@
1
+ import { TyravelRequest } from './request.js';
2
+ import { joinRoutePaths, RouteGroupBuilder, } from './route-group.js';
3
+ import { MiddlewareRegistry, } from './middleware-registry.js';
4
+ export class RouteNotFoundException extends Error {
5
+ constructor(method, path) {
6
+ super(`Route not found: ${method} ${path}`);
7
+ this.name = 'RouteNotFoundException';
8
+ }
9
+ }
10
+ class RouteRegistrar {
11
+ router;
12
+ method;
13
+ middleware;
14
+ constructor(router, method, middleware = []) {
15
+ this.router = router;
16
+ this.method = method;
17
+ this.middleware = middleware;
18
+ }
19
+ get(pattern, handler) {
20
+ return this.router.add(this.method, pattern, handler, this.middleware);
21
+ }
22
+ post(pattern, handler) {
23
+ return this.router.add('POST', pattern, handler, this.middleware);
24
+ }
25
+ put(pattern, handler) {
26
+ return this.router.add('PUT', pattern, handler, this.middleware);
27
+ }
28
+ patch(pattern, handler) {
29
+ return this.router.add('PATCH', pattern, handler, this.middleware);
30
+ }
31
+ delete(pattern, handler) {
32
+ return this.router.add('DELETE', pattern, handler, this.middleware);
33
+ }
34
+ }
35
+ export class Router {
36
+ routes = [];
37
+ namedRoutes = new Map();
38
+ globalMiddleware = [];
39
+ scopeStack = [];
40
+ middlewareRegistry;
41
+ handlerNormalizer = (handler) => handler;
42
+ constructor(middlewareRegistry = new MiddlewareRegistry()) {
43
+ this.middlewareRegistry = middlewareRegistry;
44
+ }
45
+ getMiddlewareRegistry() {
46
+ return this.middlewareRegistry;
47
+ }
48
+ setHandlerNormalizer(normalizer) {
49
+ this.handlerNormalizer = normalizer;
50
+ return this;
51
+ }
52
+ prefix(prefix) {
53
+ return new RouteGroupBuilder(this).prefix(prefix);
54
+ }
55
+ namePrefix(prefix) {
56
+ return new RouteGroupBuilder(this).namePrefix(prefix);
57
+ }
58
+ group(callback) {
59
+ return new RouteGroupBuilder(this).group(callback);
60
+ }
61
+ middleware(...middleware) {
62
+ return new RouteRegistrar(this, 'GET', middleware);
63
+ }
64
+ runInScope(scope, callback) {
65
+ this.scopeStack.push(scope);
66
+ try {
67
+ callback();
68
+ }
69
+ finally {
70
+ this.scopeStack.pop();
71
+ }
72
+ }
73
+ get(pattern, handler) {
74
+ return this.add('GET', pattern, handler);
75
+ }
76
+ post(pattern, handler) {
77
+ return this.add('POST', pattern, handler);
78
+ }
79
+ put(pattern, handler) {
80
+ return this.add('PUT', pattern, handler);
81
+ }
82
+ patch(pattern, handler) {
83
+ return this.add('PATCH', pattern, handler);
84
+ }
85
+ delete(pattern, handler) {
86
+ return this.add('DELETE', pattern, handler);
87
+ }
88
+ use(...middleware) {
89
+ this.globalMiddleware.push(...middleware);
90
+ return this;
91
+ }
92
+ add(method, pattern, handler, middleware = []) {
93
+ return this.addScoped({ prefix: '', middleware: [] }, method, pattern, handler, middleware);
94
+ }
95
+ addScoped(scope, method, pattern, handler, middleware = []) {
96
+ const activeScope = this.mergeScopes([...this.scopeStack, scope]);
97
+ const fullPattern = joinRoutePaths(activeScope.prefix, pattern);
98
+ this.routes.push({
99
+ method,
100
+ pattern: fullPattern,
101
+ handler: this.handlerNormalizer(handler),
102
+ middleware: this.resolveMiddleware([
103
+ ...this.globalMiddleware,
104
+ ...activeScope.middleware,
105
+ ...middleware,
106
+ ]),
107
+ name: undefined,
108
+ namePrefix: activeScope.namePrefix,
109
+ });
110
+ return this;
111
+ }
112
+ name(name) {
113
+ const route = this.routes.at(-1);
114
+ if (!route) {
115
+ throw new Error('Cannot name a route before defining one.');
116
+ }
117
+ const scopedName = route.namePrefix ? `${route.namePrefix}${name}` : name;
118
+ route.name = scopedName;
119
+ this.namedRoutes.set(scopedName, route);
120
+ return this;
121
+ }
122
+ url(name, params = {}) {
123
+ const route = this.namedRoutes.get(name);
124
+ if (!route) {
125
+ throw new Error(`Named route not found: ${name}`);
126
+ }
127
+ return route.pattern.replace(/:([A-Za-z0-9_]+)/g, (_, key) => {
128
+ const value = params[key];
129
+ if (value === undefined) {
130
+ throw new Error(`Missing route parameter: ${key}`);
131
+ }
132
+ return encodeURIComponent(value);
133
+ });
134
+ }
135
+ async dispatch(request) {
136
+ const compiled = this.compile();
137
+ const url = new URL(request.url);
138
+ const method = request.method.toUpperCase();
139
+ for (const route of compiled) {
140
+ if (route.definition.method !== method) {
141
+ continue;
142
+ }
143
+ const match = route.regex.exec(url.pathname);
144
+ if (!match) {
145
+ continue;
146
+ }
147
+ const params = this.extractParams(route.paramNames, match);
148
+ const tyravelRequest = new TyravelRequest(request, params, route.definition.name);
149
+ return this.runPipeline(tyravelRequest, route.definition);
150
+ }
151
+ throw new RouteNotFoundException(method, url.pathname);
152
+ }
153
+ async runPipeline(request, route) {
154
+ const middleware = route.middleware;
155
+ let index = -1;
156
+ const next = async () => {
157
+ index += 1;
158
+ if (index < middleware.length) {
159
+ const current = middleware[index];
160
+ if (!current) {
161
+ return next();
162
+ }
163
+ return current(request, next);
164
+ }
165
+ const result = await route.handler(request);
166
+ return this.normalizeResponse(result);
167
+ };
168
+ return next();
169
+ }
170
+ normalizeResponse(result) {
171
+ return result;
172
+ }
173
+ resolveMiddleware(inputs) {
174
+ return this.middlewareRegistry.resolveMany(inputs);
175
+ }
176
+ mergeScopes(scopes) {
177
+ return scopes.reduce((merged, scope) => ({
178
+ prefix: joinRoutePaths(merged.prefix, scope.prefix),
179
+ middleware: [...merged.middleware, ...scope.middleware],
180
+ namePrefix: scope.namePrefix ?? merged.namePrefix,
181
+ }), { prefix: '', middleware: [] });
182
+ }
183
+ compile() {
184
+ return this.routes.map((definition) => {
185
+ const paramNames = [];
186
+ const pattern = definition.pattern
187
+ .replace(/\/+$/, '')
188
+ .replace(/:([A-Za-z0-9_]+)/g, (_, name) => {
189
+ paramNames.push(name);
190
+ return '([^/]+)';
191
+ });
192
+ const normalizedPattern = pattern === '' ? '/' : pattern;
193
+ const regex = new RegExp(`^${normalizedPattern}/?$`);
194
+ return {
195
+ definition,
196
+ regex,
197
+ paramNames,
198
+ };
199
+ });
200
+ }
201
+ extractParams(names, match) {
202
+ const params = {};
203
+ for (const [index, name] of names.entries()) {
204
+ const value = match[index + 1];
205
+ if (value !== undefined) {
206
+ params[name] = decodeURIComponent(value);
207
+ }
208
+ }
209
+ return params;
210
+ }
211
+ }
212
+ export function createRouter(middlewareRegistry) {
213
+ return new Router(middlewareRegistry);
214
+ }
215
+ //# sourceMappingURL=router.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"router.js","sourceRoot":"","sources":["../src/router.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=router.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"router.test.d.ts","sourceRoot":"","sources":["../src/router.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,80 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { MiddlewareRegistry } from './middleware-registry.js';
3
+ import { Response } from './response.js';
4
+ import { RouteNotFoundException, Router } from './router.js';
5
+ describe('Router', () => {
6
+ it('matches static routes', async () => {
7
+ const router = new Router();
8
+ router.get('/', () => Response.text('home'));
9
+ const response = await router.dispatch(new Request('http://localhost/'));
10
+ expect(await response.text()).toBe('home');
11
+ });
12
+ it('matches parameterized routes', async () => {
13
+ const router = new Router();
14
+ router.get('/users/:id', (request) => Response.json({ id: request.param('id') }));
15
+ const response = await router.dispatch(new Request('http://localhost/users/42'));
16
+ expect(await response.json()).toEqual({ id: '42' });
17
+ });
18
+ it('runs middleware in order', async () => {
19
+ const router = new Router();
20
+ const order = [];
21
+ router.use(async (_request, next) => {
22
+ order.push('global');
23
+ return next();
24
+ });
25
+ router.middleware(async (_request, next) => {
26
+ order.push('route');
27
+ return next();
28
+ }).get('/ping', () => {
29
+ order.push('handler');
30
+ return Response.text('pong');
31
+ });
32
+ await router.dispatch(new Request('http://localhost/ping'));
33
+ expect(order).toEqual(['global', 'route', 'handler']);
34
+ });
35
+ it('resolves named middleware aliases', async () => {
36
+ const registry = new MiddlewareRegistry();
37
+ const router = new Router(registry);
38
+ const order = [];
39
+ registry.alias('auth', async (_request, next) => {
40
+ order.push('auth');
41
+ return next();
42
+ });
43
+ router.middleware('auth').get('/secure', () => Response.text('ok'));
44
+ await router.dispatch(new Request('http://localhost/secure'));
45
+ expect(order).toEqual(['auth']);
46
+ });
47
+ it('applies route groups with prefix and middleware', async () => {
48
+ const router = new Router();
49
+ const order = [];
50
+ router.prefix('api').middleware(async (_request, next) => {
51
+ order.push('api');
52
+ return next();
53
+ }).group(() => {
54
+ router.get('/users', () => {
55
+ order.push('handler');
56
+ return Response.text('ok');
57
+ });
58
+ });
59
+ const response = await router.dispatch(new Request('http://localhost/api/users'));
60
+ expect(await response.text()).toBe('ok');
61
+ expect(order).toEqual(['api', 'handler']);
62
+ });
63
+ it('prefixes named routes inside groups', () => {
64
+ const router = new Router();
65
+ router.prefix('api').namePrefix('api.').group(() => {
66
+ router.get('/users', () => Response.text('ok')).name('users.index');
67
+ });
68
+ expect(router.url('api.users.index')).toBe('/api/users');
69
+ });
70
+ it('generates urls for named routes', () => {
71
+ const router = new Router();
72
+ router.get('/posts/:slug', () => Response.text('ok')).name('posts.show');
73
+ expect(router.url('posts.show', { slug: 'hello-world' })).toBe('/posts/hello-world');
74
+ });
75
+ it('throws when no route matches', async () => {
76
+ const router = new Router();
77
+ await expect(router.dispatch(new Request('http://localhost/missing'))).rejects.toThrow(RouteNotFoundException);
78
+ });
79
+ });
80
+ //# sourceMappingURL=router.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"router.test.js","sourceRoot":"","sources":["../src/router.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ export interface SessionContract {
2
+ readonly id: string;
3
+ get<T = unknown>(key: string, fallback?: T): T | undefined;
4
+ put(key: string, value: unknown): SessionContract;
5
+ forget(key: string): SessionContract;
6
+ }
7
+ //# sourceMappingURL=session-contract.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-contract.d.ts","sourceRoot":"","sources":["../src/session-contract.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,eAAe;IAC9B,QAAQ,CAAC,EAAE,EAAE,MAAM,CAAC;IACpB,GAAG,CAAC,CAAC,GAAG,OAAO,EAAE,GAAG,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,SAAS,CAAC;IAC3D,GAAG,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,GAAG,eAAe,CAAC;IAClD,MAAM,CAAC,GAAG,EAAE,MAAM,GAAG,eAAe,CAAC;CACtC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=session-contract.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-contract.js","sourceRoot":"","sources":["../src/session-contract.ts"],"names":[],"mappings":""}
@@ -0,0 +1,14 @@
1
+ import type { TyravelRequest } from './request.js';
2
+ export type HttpMethod = 'GET' | 'POST' | 'PUT' | 'PATCH' | 'DELETE' | 'OPTIONS' | 'HEAD';
3
+ export type RouteParams = Record<string, string>;
4
+ export type RouteHandler = (request: TyravelRequest) => Response | Promise<Response>;
5
+ export type Middleware = (request: TyravelRequest, next: () => Promise<Response>) => Promise<Response>;
6
+ export interface RouteDefinition {
7
+ method: HttpMethod;
8
+ pattern: string;
9
+ handler: RouteHandler;
10
+ name?: string;
11
+ namePrefix?: string;
12
+ middleware: Middleware[];
13
+ }
14
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,cAAc,CAAC;AAEnD,MAAM,MAAM,UAAU,GAClB,KAAK,GACL,MAAM,GACN,KAAK,GACL,OAAO,GACP,QAAQ,GACR,SAAS,GACT,MAAM,CAAC;AAEX,MAAM,MAAM,WAAW,GAAG,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;AAEjD,MAAM,MAAM,YAAY,GAAG,CACzB,OAAO,EAAE,cAAc,KACpB,QAAQ,GAAG,OAAO,CAAC,QAAQ,CAAC,CAAC;AAElC,MAAM,MAAM,UAAU,GAAG,CACvB,OAAO,EAAE,cAAc,EACvB,IAAI,EAAE,MAAM,OAAO,CAAC,QAAQ,CAAC,KAC1B,OAAO,CAAC,QAAQ,CAAC,CAAC;AAEvB,MAAM,WAAW,eAAe;IAC9B,MAAM,EAAE,UAAU,CAAC;IACnB,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,YAAY,CAAC;IACtB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,UAAU,EAAE,UAAU,EAAE,CAAC;CAC1B"}
package/dist/types.js ADDED
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":""}
package/package.json ADDED
@@ -0,0 +1,33 @@
1
+ {
2
+ "name": "@tyravel/http",
3
+ "version": "0.1.0",
4
+ "type": "module",
5
+ "main": "./dist/index.js",
6
+ "types": "./dist/index.d.ts",
7
+ "exports": {
8
+ ".": {
9
+ "types": "./dist/index.d.ts",
10
+ "import": "./dist/index.js"
11
+ }
12
+ },
13
+ "scripts": {
14
+ "build": "tsc -p tsconfig.json",
15
+ "typecheck": "tsc -p tsconfig.json --noEmit"
16
+ },
17
+ "files": [
18
+ "dist"
19
+ ],
20
+ "description": "HTTP router and middleware for Tyravel",
21
+ "license": "MIT",
22
+ "repository": {
23
+ "type": "git",
24
+ "url": "git+https://github.com/thesimonharms/tyravel.git",
25
+ "directory": "packages/http"
26
+ },
27
+ "publishConfig": {
28
+ "access": "public"
29
+ },
30
+ "engines": {
31
+ "node": ">=22"
32
+ }
33
+ }