@flight-framework/core 0.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/LICENSE +21 -0
- package/dist/actions/index.d.ts +108 -0
- package/dist/actions/index.js +3 -0
- package/dist/actions/index.js.map +1 -0
- package/dist/adapters/index.d.ts +243 -0
- package/dist/adapters/index.js +3 -0
- package/dist/adapters/index.js.map +1 -0
- package/dist/cache/index.d.ts +76 -0
- package/dist/cache/index.js +3 -0
- package/dist/cache/index.js.map +1 -0
- package/dist/chunk-3AIQVGTM.js +120 -0
- package/dist/chunk-3AIQVGTM.js.map +1 -0
- package/dist/chunk-AFSKXC6V.js +188 -0
- package/dist/chunk-AFSKXC6V.js.map +1 -0
- package/dist/chunk-AJ3IBYXT.js +47 -0
- package/dist/chunk-AJ3IBYXT.js.map +1 -0
- package/dist/chunk-GCQZ4FHI.js +245 -0
- package/dist/chunk-GCQZ4FHI.js.map +1 -0
- package/dist/chunk-I5RHYGX6.js +167 -0
- package/dist/chunk-I5RHYGX6.js.map +1 -0
- package/dist/chunk-KWFX6WHG.js +311 -0
- package/dist/chunk-KWFX6WHG.js.map +1 -0
- package/dist/chunk-Q4C5CCHK.js +13 -0
- package/dist/chunk-Q4C5CCHK.js.map +1 -0
- package/dist/chunk-QEFGUHYD.js +221 -0
- package/dist/chunk-QEFGUHYD.js.map +1 -0
- package/dist/chunk-TKXN7KGE.js +145 -0
- package/dist/chunk-TKXN7KGE.js.map +1 -0
- package/dist/chunk-WAGCTWGY.js +93 -0
- package/dist/chunk-WAGCTWGY.js.map +1 -0
- package/dist/chunk-Y22KEW2F.js +223 -0
- package/dist/chunk-Y22KEW2F.js.map +1 -0
- package/dist/chunk-ZVC3ZWLM.js +52 -0
- package/dist/chunk-ZVC3ZWLM.js.map +1 -0
- package/dist/config/index.d.ts +146 -0
- package/dist/config/index.js +3 -0
- package/dist/config/index.js.map +1 -0
- package/dist/file-router/index.d.ts +71 -0
- package/dist/file-router/index.js +3 -0
- package/dist/file-router/index.js.map +1 -0
- package/dist/handlers/index.d.ts +59 -0
- package/dist/handlers/index.js +3 -0
- package/dist/handlers/index.js.map +1 -0
- package/dist/index.d.ts +23 -0
- package/dist/index.js +19 -0
- package/dist/index.js.map +1 -0
- package/dist/middleware/index.d.ts +152 -0
- package/dist/middleware/index.js +3 -0
- package/dist/middleware/index.js.map +1 -0
- package/dist/render/index.d.ts +131 -0
- package/dist/render/index.js +3 -0
- package/dist/render/index.js.map +1 -0
- package/dist/router/index.d.ts +65 -0
- package/dist/router/index.js +3 -0
- package/dist/router/index.js.map +1 -0
- package/dist/rsc/index.d.ts +131 -0
- package/dist/rsc/index.js +3 -0
- package/dist/rsc/index.js.map +1 -0
- package/dist/server/index.d.ts +135 -0
- package/dist/server/index.js +6 -0
- package/dist/server/index.js.map +1 -0
- package/dist/streaming/index.d.ts +169 -0
- package/dist/streaming/index.js +3 -0
- package/dist/streaming/index.js.map +1 -0
- package/package.json +100 -0
|
@@ -0,0 +1,167 @@
|
|
|
1
|
+
import { readdir } from 'fs/promises';
|
|
2
|
+
import { join, extname, basename } from 'path';
|
|
3
|
+
|
|
4
|
+
// src/file-router/index.ts
|
|
5
|
+
async function scanRoutes(options) {
|
|
6
|
+
const {
|
|
7
|
+
directory,
|
|
8
|
+
extensions = [".ts", ".js"]
|
|
9
|
+
} = options;
|
|
10
|
+
const routes = [];
|
|
11
|
+
const errors = [];
|
|
12
|
+
async function scanDir(dir, basePath = "") {
|
|
13
|
+
try {
|
|
14
|
+
const entries = await readdir(dir, { withFileTypes: true });
|
|
15
|
+
for (const entry of entries) {
|
|
16
|
+
const fullPath = join(dir, entry.name);
|
|
17
|
+
if (entry.isDirectory()) {
|
|
18
|
+
if (entry.name.startsWith(".") || entry.name === "node_modules") {
|
|
19
|
+
continue;
|
|
20
|
+
}
|
|
21
|
+
await scanDir(fullPath, `${basePath}/${entry.name}`);
|
|
22
|
+
} else if (entry.isFile()) {
|
|
23
|
+
const ext = extname(entry.name);
|
|
24
|
+
if (!extensions.includes(ext)) {
|
|
25
|
+
continue;
|
|
26
|
+
}
|
|
27
|
+
const routeInfo = parseRouteFile(entry.name, basePath);
|
|
28
|
+
if (routeInfo && routeInfo.method && routeInfo.path) {
|
|
29
|
+
routes.push({
|
|
30
|
+
method: routeInfo.method,
|
|
31
|
+
path: routeInfo.path,
|
|
32
|
+
filePath: fullPath,
|
|
33
|
+
type: routeInfo.type
|
|
34
|
+
});
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
} catch (error) {
|
|
39
|
+
errors.push(`Failed to scan ${dir}: ${error}`);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
await scanDir(directory);
|
|
43
|
+
return { routes, errors };
|
|
44
|
+
}
|
|
45
|
+
function parseRouteFile(filename, basePath) {
|
|
46
|
+
const ext = extname(filename);
|
|
47
|
+
const nameWithoutExt = basename(filename, ext);
|
|
48
|
+
if (nameWithoutExt.startsWith("_")) {
|
|
49
|
+
return null;
|
|
50
|
+
}
|
|
51
|
+
let type = "api";
|
|
52
|
+
let routeName = nameWithoutExt;
|
|
53
|
+
let method = "ALL";
|
|
54
|
+
const pageMatch = nameWithoutExt.match(/^(.+)?\.page$/i);
|
|
55
|
+
if (pageMatch) {
|
|
56
|
+
type = "page";
|
|
57
|
+
method = "GET";
|
|
58
|
+
routeName = pageMatch[1] || "index";
|
|
59
|
+
} else {
|
|
60
|
+
const methodMatch = nameWithoutExt.match(/^(.+)\.(get|post|put|delete|patch|head|options)$/i);
|
|
61
|
+
if (methodMatch) {
|
|
62
|
+
routeName = methodMatch[1] || nameWithoutExt;
|
|
63
|
+
method = (methodMatch[2] || "ALL").toUpperCase();
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
if (basePath.startsWith("/api") || basePath.includes("/api/")) {
|
|
67
|
+
type = "api";
|
|
68
|
+
}
|
|
69
|
+
let path = basePath;
|
|
70
|
+
if (routeName !== "index") {
|
|
71
|
+
path = `${basePath}/${convertToRoutePath(routeName)}`;
|
|
72
|
+
}
|
|
73
|
+
if (!path.startsWith("/")) {
|
|
74
|
+
path = "/" + path;
|
|
75
|
+
}
|
|
76
|
+
if (path !== "/" && path.endsWith("/")) {
|
|
77
|
+
path = path.slice(0, -1);
|
|
78
|
+
}
|
|
79
|
+
return { method, path, type };
|
|
80
|
+
}
|
|
81
|
+
function convertToRoutePath(name) {
|
|
82
|
+
if (name.startsWith("[[...") && name.endsWith("]]")) {
|
|
83
|
+
const paramName = name.slice(5, -2);
|
|
84
|
+
return `:${paramName}*`;
|
|
85
|
+
}
|
|
86
|
+
if (name.startsWith("[...") && name.endsWith("]")) {
|
|
87
|
+
const paramName = name.slice(4, -1);
|
|
88
|
+
return `:${paramName}+`;
|
|
89
|
+
}
|
|
90
|
+
if (name.startsWith("[") && name.endsWith("]")) {
|
|
91
|
+
const paramName = name.slice(1, -1);
|
|
92
|
+
return `:${paramName}`;
|
|
93
|
+
}
|
|
94
|
+
return name;
|
|
95
|
+
}
|
|
96
|
+
async function loadRoutes(scanResult) {
|
|
97
|
+
const loadedRoutes = [];
|
|
98
|
+
for (const route of scanResult.routes) {
|
|
99
|
+
try {
|
|
100
|
+
const module = await import(route.filePath);
|
|
101
|
+
if (route.type === "page") {
|
|
102
|
+
const component = module.default;
|
|
103
|
+
const meta = module.meta || module.metadata || {};
|
|
104
|
+
if (component) {
|
|
105
|
+
loadedRoutes.push({
|
|
106
|
+
...route,
|
|
107
|
+
component,
|
|
108
|
+
meta
|
|
109
|
+
});
|
|
110
|
+
}
|
|
111
|
+
continue;
|
|
112
|
+
}
|
|
113
|
+
if (route.method === "ALL") {
|
|
114
|
+
const methods = ["GET", "POST", "PUT", "DELETE", "PATCH", "HEAD", "OPTIONS"];
|
|
115
|
+
for (const method of methods) {
|
|
116
|
+
if (typeof module[method] === "function") {
|
|
117
|
+
loadedRoutes.push({
|
|
118
|
+
...route,
|
|
119
|
+
method,
|
|
120
|
+
handler: module[method]
|
|
121
|
+
});
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
if (typeof module.default === "function") {
|
|
125
|
+
loadedRoutes.push({
|
|
126
|
+
...route,
|
|
127
|
+
method: "ALL",
|
|
128
|
+
handler: module.default
|
|
129
|
+
});
|
|
130
|
+
}
|
|
131
|
+
} else {
|
|
132
|
+
const handler = module[route.method] || module.default;
|
|
133
|
+
if (typeof handler === "function") {
|
|
134
|
+
loadedRoutes.push({
|
|
135
|
+
...route,
|
|
136
|
+
handler
|
|
137
|
+
});
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
} catch (error) {
|
|
141
|
+
console.error(`[Flight] Failed to load route ${route.filePath}:`, error);
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
return loadedRoutes;
|
|
145
|
+
}
|
|
146
|
+
async function createFileRouter(options) {
|
|
147
|
+
let routes = [];
|
|
148
|
+
async function refresh() {
|
|
149
|
+
const scanResult = await scanRoutes(options);
|
|
150
|
+
if (scanResult.errors.length > 0) {
|
|
151
|
+
console.warn("[Flight] Route scan errors:", scanResult.errors);
|
|
152
|
+
}
|
|
153
|
+
routes = await loadRoutes(scanResult);
|
|
154
|
+
console.log(`[Flight] Loaded ${routes.length} routes from ${options.directory}`);
|
|
155
|
+
}
|
|
156
|
+
await refresh();
|
|
157
|
+
return {
|
|
158
|
+
get routes() {
|
|
159
|
+
return routes;
|
|
160
|
+
},
|
|
161
|
+
refresh
|
|
162
|
+
};
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
export { createFileRouter, loadRoutes, scanRoutes };
|
|
166
|
+
//# sourceMappingURL=chunk-I5RHYGX6.js.map
|
|
167
|
+
//# sourceMappingURL=chunk-I5RHYGX6.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/file-router/index.ts"],"names":[],"mappings":"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","file":"chunk-I5RHYGX6.js","sourcesContent":["/**\r\n * @flight/core - File Router\r\n * \r\n * Auto-discovery of routes from file system.\r\n * Similar to Next.js App Router and Nuxt server/api patterns.\r\n */\r\n\r\nimport { readdir } from 'node:fs/promises';\r\nimport { join, basename, extname } from 'node:path';\r\n\r\n// ============================================================================\r\n// Types\r\n// ============================================================================\r\n\r\nexport type HttpMethod = 'GET' | 'POST' | 'PUT' | 'DELETE' | 'PATCH' | 'HEAD' | 'OPTIONS';\r\n\r\nexport type Handler = (context: unknown) => Response | Promise<Response>;\r\nexport type Middleware = (context: unknown, next: () => Promise<Response>) => Response | Promise<Response>;\r\n\r\nexport interface FileRoute {\r\n /** HTTP method (GET, POST, etc) or 'ALL' */\r\n method: HttpMethod | 'ALL';\r\n /** Route path with params (e.g., /users/:id) */\r\n path: string;\r\n /** Original file path */\r\n filePath: string;\r\n /** Handler function (for APIs) */\r\n handler?: Handler;\r\n /** Route-specific middleware */\r\n middleware?: Middleware[];\r\n /** Route type: 'page' for SSR pages, 'api' for API endpoints */\r\n type: 'page' | 'api';\r\n /** Component function (for pages) */\r\n component?: () => unknown;\r\n /** Page metadata (title, description, etc) */\r\n meta?: Record<string, unknown>;\r\n}\r\n\r\nexport interface FileRouterOptions {\r\n /** Root directory to scan (default: src/routes) */\r\n directory: string;\r\n /** File extensions to consider (default: ['.ts', '.js']) */\r\n extensions?: string[];\r\n /** Whether to watch for changes (default: false in prod) */\r\n watch?: boolean;\r\n}\r\n\r\nexport interface ScanResult {\r\n routes: FileRoute[];\r\n errors: string[];\r\n}\r\n\r\n// ============================================================================\r\n// File Scanner\r\n// ============================================================================\r\n\r\n/**\r\n * Scan a directory for route files\r\n */\r\nexport async function scanRoutes(options: FileRouterOptions): Promise<ScanResult> {\r\n const {\r\n directory,\r\n extensions = ['.ts', '.js'],\r\n } = options;\r\n\r\n const routes: FileRoute[] = [];\r\n const errors: string[] = [];\r\n\r\n async function scanDir(dir: string, basePath: string = ''): Promise<void> {\r\n try {\r\n const entries = await readdir(dir, { withFileTypes: true });\r\n\r\n for (const entry of entries) {\r\n const fullPath = join(dir, entry.name);\r\n\r\n if (entry.isDirectory()) {\r\n // Skip hidden directories and node_modules\r\n if (entry.name.startsWith('.') || entry.name === 'node_modules') {\r\n continue;\r\n }\r\n\r\n // Recurse into subdirectory\r\n await scanDir(fullPath, `${basePath}/${entry.name}`);\r\n } else if (entry.isFile()) {\r\n const ext = extname(entry.name);\r\n\r\n if (!extensions.includes(ext)) {\r\n continue;\r\n }\r\n\r\n // Parse route from filename\r\n const routeInfo = parseRouteFile(entry.name, basePath);\r\n\r\n if (routeInfo && routeInfo.method && routeInfo.path) {\r\n routes.push({\r\n method: routeInfo.method,\r\n path: routeInfo.path,\r\n filePath: fullPath,\r\n type: routeInfo.type,\r\n });\r\n }\r\n }\r\n }\r\n } catch (error) {\r\n errors.push(`Failed to scan ${dir}: ${error}`);\r\n }\r\n }\r\n\r\n await scanDir(directory);\r\n\r\n return { routes, errors };\r\n}\r\n\r\n// ============================================================================\r\n// Route Parser\r\n// ============================================================================\r\n\r\ninterface ParsedRoute {\r\n method: HttpMethod | 'ALL';\r\n path: string;\r\n type: 'page' | 'api';\r\n}\r\n\r\n/**\r\n * Parse route information from filename and path\r\n * \r\n * Patterns:\r\n * - index.ts → /\r\n * - users.ts → /users\r\n * - users.get.ts → GET /users (API)\r\n * - users.post.ts → POST /users (API)\r\n * - about.page.tsx → GET /about (Page)\r\n * - blog/[slug].page.tsx → GET /blog/:slug (Page)\r\n * - [id].ts → /:id\r\n * - [...slug].ts → /* (catch-all)\r\n * - [[...slug]].ts → /* (optional catch-all)\r\n */\r\nfunction parseRouteFile(filename: string, basePath: string): ParsedRoute | null {\r\n const ext = extname(filename);\r\n const nameWithoutExt = basename(filename, ext);\r\n\r\n // Skip files starting with underscore (private)\r\n if (nameWithoutExt.startsWith('_')) {\r\n return null;\r\n }\r\n\r\n // Detect route type\r\n let type: 'page' | 'api' = 'api';\r\n let routeName = nameWithoutExt;\r\n let method: HttpMethod | 'ALL' = 'ALL';\r\n\r\n // Check for page suffix (e.g., about.page.tsx, index.page.tsx)\r\n const pageMatch = nameWithoutExt.match(/^(.+)?\\.page$/i);\r\n if (pageMatch) {\r\n type = 'page';\r\n method = 'GET'; // Pages are always GET\r\n routeName = pageMatch[1] || 'index';\r\n } else {\r\n // Check for method suffix (e.g., users.get.ts)\r\n const methodMatch = nameWithoutExt.match(/^(.+)\\.(get|post|put|delete|patch|head|options)$/i);\r\n if (methodMatch) {\r\n routeName = methodMatch[1] || nameWithoutExt;\r\n method = (methodMatch[2] || 'ALL').toUpperCase() as HttpMethod;\r\n }\r\n }\r\n\r\n // Also check if file is in /api/ directory\r\n if (basePath.startsWith('/api') || basePath.includes('/api/')) {\r\n type = 'api';\r\n }\r\n\r\n // Build route path\r\n let path = basePath;\r\n\r\n if (routeName !== 'index') {\r\n path = `${basePath}/${convertToRoutePath(routeName)}`;\r\n }\r\n\r\n // Ensure path starts with /\r\n if (!path.startsWith('/')) {\r\n path = '/' + path;\r\n }\r\n\r\n // Remove trailing slash (except for root)\r\n if (path !== '/' && path.endsWith('/')) {\r\n path = path.slice(0, -1);\r\n }\r\n\r\n return { method, path, type };\r\n}\r\n\r\n/**\r\n * Convert filename segment to route path segment\r\n * \r\n * - [id] → :id\r\n * - [...slug] → *\r\n * - [[...slug]] → *?\r\n */\r\nfunction convertToRoutePath(name: string): string {\r\n // Optional catch-all: [[...slug]]\r\n if (name.startsWith('[[...') && name.endsWith(']]')) {\r\n const paramName = name.slice(5, -2);\r\n return `:${paramName}*`;\r\n }\r\n\r\n // Catch-all: [...slug]\r\n if (name.startsWith('[...') && name.endsWith(']')) {\r\n const paramName = name.slice(4, -1);\r\n return `:${paramName}+`;\r\n }\r\n\r\n // Dynamic param: [id]\r\n if (name.startsWith('[') && name.endsWith(']')) {\r\n const paramName = name.slice(1, -1);\r\n return `:${paramName}`;\r\n }\r\n\r\n return name;\r\n}\r\n\r\n// ============================================================================\r\n// Route Loader\r\n// ============================================================================\r\n\r\n/**\r\n * Load routes with their handlers or components\r\n */\r\nexport async function loadRoutes(scanResult: ScanResult): Promise<FileRoute[]> {\r\n const loadedRoutes: FileRoute[] = [];\r\n\r\n for (const route of scanResult.routes) {\r\n try {\r\n // Dynamic import of route file\r\n const module = await import(route.filePath);\r\n\r\n // Handle PAGE routes\r\n if (route.type === 'page') {\r\n // Pages export default component\r\n const component = module.default;\r\n const meta = module.meta || module.metadata || {};\r\n\r\n if (component) {\r\n loadedRoutes.push({\r\n ...route,\r\n component,\r\n meta,\r\n });\r\n }\r\n continue;\r\n }\r\n\r\n // Handle API routes\r\n if (route.method === 'ALL') {\r\n // Look for named exports: GET, POST, PUT, DELETE, etc\r\n const methods: HttpMethod[] = ['GET', 'POST', 'PUT', 'DELETE', 'PATCH', 'HEAD', 'OPTIONS'];\r\n\r\n for (const method of methods) {\r\n if (typeof module[method] === 'function') {\r\n loadedRoutes.push({\r\n ...route,\r\n method,\r\n handler: module[method],\r\n });\r\n }\r\n }\r\n\r\n // Also check for default export as handler\r\n if (typeof module.default === 'function') {\r\n loadedRoutes.push({\r\n ...route,\r\n method: 'ALL',\r\n handler: module.default,\r\n });\r\n }\r\n } else {\r\n // Specific method from filename suffix\r\n const handler = module[route.method] || module.default;\r\n\r\n if (typeof handler === 'function') {\r\n loadedRoutes.push({\r\n ...route,\r\n handler,\r\n });\r\n }\r\n }\r\n } catch (error) {\r\n console.error(`[Flight] Failed to load route ${route.filePath}:`, error);\r\n }\r\n }\r\n\r\n return loadedRoutes;\r\n}\r\n\r\n// ============================================================================\r\n// File Router Factory\r\n// ============================================================================\r\n\r\nexport interface FileRouter {\r\n routes: FileRoute[];\r\n refresh: () => Promise<void>;\r\n}\r\n\r\n/**\r\n * Create a file-based router\r\n * \r\n * @example\r\n * ```typescript\r\n * import { createFileRouter } from '@flight/core/file-router';\r\n * import { createServer } from '@flight/http';\r\n * \r\n * const router = await createFileRouter({ directory: './src/routes' });\r\n * const app = createServer();\r\n * \r\n * // Register all discovered routes\r\n * for (const route of router.routes) {\r\n * app[route.method.toLowerCase()](route.path, route.handler);\r\n * }\r\n * ```\r\n */\r\nexport async function createFileRouter(options: FileRouterOptions): Promise<FileRouter> {\r\n let routes: FileRoute[] = [];\r\n\r\n async function refresh(): Promise<void> {\r\n const scanResult = await scanRoutes(options);\r\n\r\n if (scanResult.errors.length > 0) {\r\n console.warn('[Flight] Route scan errors:', scanResult.errors);\r\n }\r\n\r\n routes = await loadRoutes(scanResult);\r\n\r\n console.log(`[Flight] Loaded ${routes.length} routes from ${options.directory}`);\r\n }\r\n\r\n // Initial load\r\n await refresh();\r\n\r\n return {\r\n get routes() {\r\n return routes;\r\n },\r\n refresh,\r\n };\r\n}\r\n"]}
|
|
@@ -0,0 +1,311 @@
|
|
|
1
|
+
// src/middleware/index.ts
|
|
2
|
+
function createMiddlewareChain() {
|
|
3
|
+
const stack = [];
|
|
4
|
+
function use(pathOrMiddleware, maybeMiddleware) {
|
|
5
|
+
if (typeof pathOrMiddleware === "function") {
|
|
6
|
+
stack.push({ handler: pathOrMiddleware });
|
|
7
|
+
} else if (typeof pathOrMiddleware === "string" && maybeMiddleware) {
|
|
8
|
+
stack.push({ path: pathOrMiddleware, handler: maybeMiddleware });
|
|
9
|
+
} else if (typeof pathOrMiddleware === "object") {
|
|
10
|
+
stack.push(pathOrMiddleware);
|
|
11
|
+
}
|
|
12
|
+
return chain;
|
|
13
|
+
}
|
|
14
|
+
async function execute(ctx) {
|
|
15
|
+
let index = -1;
|
|
16
|
+
async function dispatch(i) {
|
|
17
|
+
if (i <= index) {
|
|
18
|
+
throw new Error("next() called multiple times");
|
|
19
|
+
}
|
|
20
|
+
index = i;
|
|
21
|
+
if (i >= stack.length) {
|
|
22
|
+
return;
|
|
23
|
+
}
|
|
24
|
+
const definition = stack[i];
|
|
25
|
+
if (!definition) {
|
|
26
|
+
return dispatch(i + 1);
|
|
27
|
+
}
|
|
28
|
+
if (!shouldRun(definition, ctx)) {
|
|
29
|
+
return dispatch(i + 1);
|
|
30
|
+
}
|
|
31
|
+
await definition.handler(ctx, () => dispatch(i + 1));
|
|
32
|
+
}
|
|
33
|
+
await dispatch(0);
|
|
34
|
+
}
|
|
35
|
+
const chain = {
|
|
36
|
+
use,
|
|
37
|
+
execute,
|
|
38
|
+
middlewares: () => [...stack]
|
|
39
|
+
};
|
|
40
|
+
return chain;
|
|
41
|
+
}
|
|
42
|
+
function shouldRun(def, ctx) {
|
|
43
|
+
if (def.methods && !def.methods.includes(ctx.method.toUpperCase())) {
|
|
44
|
+
return false;
|
|
45
|
+
}
|
|
46
|
+
if (def.path) {
|
|
47
|
+
const path = ctx.url.pathname;
|
|
48
|
+
if (typeof def.path === "string") {
|
|
49
|
+
if (def.path.endsWith("*")) {
|
|
50
|
+
const prefix = def.path.slice(0, -1);
|
|
51
|
+
if (!path.startsWith(prefix)) {
|
|
52
|
+
return false;
|
|
53
|
+
}
|
|
54
|
+
} else if (def.path !== path) {
|
|
55
|
+
return false;
|
|
56
|
+
}
|
|
57
|
+
} else if (def.path instanceof RegExp) {
|
|
58
|
+
if (!def.path.test(path)) {
|
|
59
|
+
return false;
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
return true;
|
|
64
|
+
}
|
|
65
|
+
function cors(options = {}) {
|
|
66
|
+
const {
|
|
67
|
+
origin = "*",
|
|
68
|
+
methods = ["GET", "POST", "PUT", "DELETE", "PATCH", "OPTIONS"],
|
|
69
|
+
headers = ["Content-Type", "Authorization"],
|
|
70
|
+
credentials = false,
|
|
71
|
+
maxAge = 86400
|
|
72
|
+
} = options;
|
|
73
|
+
return async (ctx, next) => {
|
|
74
|
+
const requestOrigin = ctx.headers.get("Origin") || "";
|
|
75
|
+
let allowedOrigin = null;
|
|
76
|
+
if (origin === "*") {
|
|
77
|
+
allowedOrigin = "*";
|
|
78
|
+
} else if (typeof origin === "string") {
|
|
79
|
+
allowedOrigin = origin;
|
|
80
|
+
} else if (Array.isArray(origin)) {
|
|
81
|
+
if (origin.includes(requestOrigin)) {
|
|
82
|
+
allowedOrigin = requestOrigin;
|
|
83
|
+
}
|
|
84
|
+
} else if (typeof origin === "function") {
|
|
85
|
+
if (origin(requestOrigin)) {
|
|
86
|
+
allowedOrigin = requestOrigin;
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
if (allowedOrigin) {
|
|
90
|
+
ctx.responseHeaders.set("Access-Control-Allow-Origin", allowedOrigin);
|
|
91
|
+
ctx.responseHeaders.set("Access-Control-Allow-Methods", methods.join(", "));
|
|
92
|
+
ctx.responseHeaders.set("Access-Control-Allow-Headers", headers.join(", "));
|
|
93
|
+
if (credentials) {
|
|
94
|
+
ctx.responseHeaders.set("Access-Control-Allow-Credentials", "true");
|
|
95
|
+
}
|
|
96
|
+
if (maxAge) {
|
|
97
|
+
ctx.responseHeaders.set("Access-Control-Max-Age", String(maxAge));
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
if (ctx.method === "OPTIONS") {
|
|
101
|
+
ctx.status = 204;
|
|
102
|
+
ctx.responseBody = null;
|
|
103
|
+
return;
|
|
104
|
+
}
|
|
105
|
+
await next();
|
|
106
|
+
};
|
|
107
|
+
}
|
|
108
|
+
var LOG_LEVEL_PRIORITY = {
|
|
109
|
+
debug: 0,
|
|
110
|
+
info: 1,
|
|
111
|
+
warn: 2,
|
|
112
|
+
error: 3,
|
|
113
|
+
silent: 4
|
|
114
|
+
};
|
|
115
|
+
var LOG_LEVEL_COLORS = {
|
|
116
|
+
debug: "\x1B[36m",
|
|
117
|
+
// Cyan
|
|
118
|
+
info: "\x1B[32m",
|
|
119
|
+
// Green
|
|
120
|
+
warn: "\x1B[33m",
|
|
121
|
+
// Yellow
|
|
122
|
+
error: "\x1B[31m",
|
|
123
|
+
// Red
|
|
124
|
+
silent: ""
|
|
125
|
+
};
|
|
126
|
+
var STATUS_COLORS = {
|
|
127
|
+
"2": "\x1B[32m",
|
|
128
|
+
// Green for 2xx
|
|
129
|
+
"3": "\x1B[36m",
|
|
130
|
+
// Cyan for 3xx
|
|
131
|
+
"4": "\x1B[33m",
|
|
132
|
+
// Yellow for 4xx
|
|
133
|
+
"5": "\x1B[31m"
|
|
134
|
+
// Red for 5xx
|
|
135
|
+
};
|
|
136
|
+
var RESET = "\x1B[0m";
|
|
137
|
+
var DIM = "\x1B[2m";
|
|
138
|
+
function logger(options = {}) {
|
|
139
|
+
const {
|
|
140
|
+
level = "info",
|
|
141
|
+
format = "pretty",
|
|
142
|
+
skip,
|
|
143
|
+
writer,
|
|
144
|
+
includeHeaders = false,
|
|
145
|
+
colorize = true
|
|
146
|
+
} = options;
|
|
147
|
+
return async (ctx, next) => {
|
|
148
|
+
if (skip?.(ctx)) {
|
|
149
|
+
return next();
|
|
150
|
+
}
|
|
151
|
+
const start = Date.now();
|
|
152
|
+
const startTime = (/* @__PURE__ */ new Date()).toISOString();
|
|
153
|
+
await next();
|
|
154
|
+
const duration = Date.now() - start;
|
|
155
|
+
const status = ctx.status;
|
|
156
|
+
let logLevel = "info";
|
|
157
|
+
if (status >= 500) {
|
|
158
|
+
logLevel = "error";
|
|
159
|
+
} else if (status >= 400) {
|
|
160
|
+
logLevel = "warn";
|
|
161
|
+
} else if (status >= 300) {
|
|
162
|
+
logLevel = "info";
|
|
163
|
+
} else {
|
|
164
|
+
logLevel = "debug";
|
|
165
|
+
}
|
|
166
|
+
if (LOG_LEVEL_PRIORITY[logLevel] < LOG_LEVEL_PRIORITY[level]) {
|
|
167
|
+
return;
|
|
168
|
+
}
|
|
169
|
+
const entry = {
|
|
170
|
+
timestamp: startTime,
|
|
171
|
+
level: logLevel,
|
|
172
|
+
method: ctx.method,
|
|
173
|
+
path: ctx.url.pathname,
|
|
174
|
+
status,
|
|
175
|
+
duration
|
|
176
|
+
};
|
|
177
|
+
const userAgent = ctx.headers.get("user-agent");
|
|
178
|
+
if (userAgent) entry.userAgent = userAgent;
|
|
179
|
+
const forwardedFor = ctx.headers.get("x-forwarded-for");
|
|
180
|
+
const ip = forwardedFor || "unknown";
|
|
181
|
+
entry.ip = ip;
|
|
182
|
+
if (includeHeaders) {
|
|
183
|
+
const headers = {};
|
|
184
|
+
ctx.headers.forEach((value, key) => {
|
|
185
|
+
if (!key.toLowerCase().includes("authorization")) {
|
|
186
|
+
headers[key] = value;
|
|
187
|
+
}
|
|
188
|
+
});
|
|
189
|
+
entry.headers = headers;
|
|
190
|
+
}
|
|
191
|
+
let formatted;
|
|
192
|
+
switch (format) {
|
|
193
|
+
case "json":
|
|
194
|
+
formatted = JSON.stringify(entry);
|
|
195
|
+
break;
|
|
196
|
+
case "combined":
|
|
197
|
+
formatted = `${ip} - - [${startTime}] "${ctx.method} ${ctx.url.pathname}" ${status} - "${userAgent || "-"}"`;
|
|
198
|
+
break;
|
|
199
|
+
case "common":
|
|
200
|
+
formatted = `${ip} - - [${startTime}] "${ctx.method} ${ctx.url.pathname}" ${status}`;
|
|
201
|
+
break;
|
|
202
|
+
case "short":
|
|
203
|
+
formatted = `${ctx.method} ${ctx.url.pathname} ${status} ${duration}ms`;
|
|
204
|
+
break;
|
|
205
|
+
case "tiny":
|
|
206
|
+
formatted = `${ctx.method} ${ctx.url.pathname} ${status}`;
|
|
207
|
+
break;
|
|
208
|
+
case "pretty":
|
|
209
|
+
default:
|
|
210
|
+
if (colorize) {
|
|
211
|
+
const statusFirstChar = String(status).charAt(0) || "2";
|
|
212
|
+
const statusColor = STATUS_COLORS[statusFirstChar] ?? "";
|
|
213
|
+
const levelColor = LOG_LEVEL_COLORS[logLevel];
|
|
214
|
+
formatted = `${DIM}${startTime}${RESET} ${levelColor}${logLevel.toUpperCase().padEnd(5)}${RESET} ${ctx.method.padEnd(7)} ${ctx.url.pathname} ${statusColor}${status}${RESET} ${DIM}${duration}ms${RESET}`;
|
|
215
|
+
} else {
|
|
216
|
+
formatted = `${startTime} ${logLevel.toUpperCase().padEnd(5)} ${ctx.method.padEnd(7)} ${ctx.url.pathname} ${status} ${duration}ms`;
|
|
217
|
+
}
|
|
218
|
+
break;
|
|
219
|
+
}
|
|
220
|
+
if (writer) {
|
|
221
|
+
writer(entry, formatted);
|
|
222
|
+
} else {
|
|
223
|
+
switch (logLevel) {
|
|
224
|
+
case "error":
|
|
225
|
+
console.error(formatted);
|
|
226
|
+
break;
|
|
227
|
+
case "warn":
|
|
228
|
+
console.warn(formatted);
|
|
229
|
+
break;
|
|
230
|
+
case "debug":
|
|
231
|
+
console.debug(formatted);
|
|
232
|
+
break;
|
|
233
|
+
default:
|
|
234
|
+
console.log(formatted);
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
};
|
|
238
|
+
}
|
|
239
|
+
function securityHeaders(options = {}) {
|
|
240
|
+
const {
|
|
241
|
+
contentSecurityPolicy = "default-src 'self'",
|
|
242
|
+
xFrameOptions = "DENY",
|
|
243
|
+
xContentTypeOptions = true,
|
|
244
|
+
referrerPolicy = "strict-origin-when-cross-origin",
|
|
245
|
+
permissionsPolicy
|
|
246
|
+
} = options;
|
|
247
|
+
return async (ctx, next) => {
|
|
248
|
+
if (contentSecurityPolicy) {
|
|
249
|
+
ctx.responseHeaders.set("Content-Security-Policy", contentSecurityPolicy);
|
|
250
|
+
}
|
|
251
|
+
if (xFrameOptions) {
|
|
252
|
+
ctx.responseHeaders.set("X-Frame-Options", xFrameOptions);
|
|
253
|
+
}
|
|
254
|
+
if (xContentTypeOptions) {
|
|
255
|
+
ctx.responseHeaders.set("X-Content-Type-Options", "nosniff");
|
|
256
|
+
}
|
|
257
|
+
if (referrerPolicy) {
|
|
258
|
+
ctx.responseHeaders.set("Referrer-Policy", referrerPolicy);
|
|
259
|
+
}
|
|
260
|
+
if (permissionsPolicy) {
|
|
261
|
+
ctx.responseHeaders.set("Permissions-Policy", permissionsPolicy);
|
|
262
|
+
}
|
|
263
|
+
await next();
|
|
264
|
+
};
|
|
265
|
+
}
|
|
266
|
+
function compress() {
|
|
267
|
+
return async (ctx, next) => {
|
|
268
|
+
ctx.locals["flight:compress"] = true;
|
|
269
|
+
await next();
|
|
270
|
+
};
|
|
271
|
+
}
|
|
272
|
+
function createContextFromRequest(request, params = {}) {
|
|
273
|
+
const url = new URL(request.url);
|
|
274
|
+
return {
|
|
275
|
+
url,
|
|
276
|
+
method: request.method,
|
|
277
|
+
headers: request.headers,
|
|
278
|
+
params,
|
|
279
|
+
query: url.searchParams,
|
|
280
|
+
locals: {},
|
|
281
|
+
request,
|
|
282
|
+
status: 200,
|
|
283
|
+
responseHeaders: new Headers()
|
|
284
|
+
};
|
|
285
|
+
}
|
|
286
|
+
function createResponseFromContext(ctx) {
|
|
287
|
+
return new Response(ctx.responseBody, {
|
|
288
|
+
status: ctx.status,
|
|
289
|
+
headers: ctx.responseHeaders
|
|
290
|
+
});
|
|
291
|
+
}
|
|
292
|
+
function compose(...middlewares) {
|
|
293
|
+
return async (ctx, next) => {
|
|
294
|
+
let index = -1;
|
|
295
|
+
async function dispatch(i) {
|
|
296
|
+
if (i <= index) {
|
|
297
|
+
throw new Error("next() called multiple times");
|
|
298
|
+
}
|
|
299
|
+
index = i;
|
|
300
|
+
const fn = i === middlewares.length ? next : middlewares[i];
|
|
301
|
+
if (fn) {
|
|
302
|
+
await fn(ctx, () => dispatch(i + 1));
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
await dispatch(0);
|
|
306
|
+
};
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
export { compose, compress, cors, createContextFromRequest, createMiddlewareChain, createResponseFromContext, logger, securityHeaders };
|
|
310
|
+
//# sourceMappingURL=chunk-KWFX6WHG.js.map
|
|
311
|
+
//# sourceMappingURL=chunk-KWFX6WHG.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/middleware/index.ts"],"names":[],"mappings":"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","file":"chunk-KWFX6WHG.js","sourcesContent":["/**\r\n * Flight Middleware - Composable request/response handlers\r\n * \r\n * Framework-agnostic middleware system inspired by Koa/Hono.\r\n */\r\n\r\n// ============================================================================\r\n// Types\r\n// ============================================================================\r\n\r\n/** Request context passed through middleware chain */\r\nexport interface MiddlewareContext {\r\n /** Request URL */\r\n url: URL;\r\n /** Request method */\r\n method: string;\r\n /** Request headers */\r\n headers: Headers;\r\n /** URL parameters from routing */\r\n params: Record<string, string | string[]>;\r\n /** Query parameters */\r\n query: URLSearchParams;\r\n /** Parsed request body (if any) */\r\n body?: unknown;\r\n /** Local data shared between middleware */\r\n locals: Record<string, unknown>;\r\n /** Original request (platform-specific) */\r\n request?: Request;\r\n\r\n // Response building\r\n /** Response status code */\r\n status: number;\r\n /** Response headers */\r\n responseHeaders: Headers;\r\n /** Response body */\r\n responseBody?: BodyInit | null;\r\n}\r\n\r\n/** Next function to call the next middleware */\r\nexport type MiddlewareNext = () => Promise<void>;\r\n\r\n/** Middleware function signature */\r\nexport type Middleware = (\r\n ctx: MiddlewareContext,\r\n next: MiddlewareNext\r\n) => Promise<void> | void;\r\n\r\n/** Middleware with optional path matching */\r\nexport interface MiddlewareDefinition {\r\n /** Path pattern to match (undefined = match all) */\r\n path?: string | RegExp;\r\n /** HTTP methods to match (undefined = match all) */\r\n methods?: string[];\r\n /** The middleware function */\r\n handler: Middleware;\r\n}\r\n\r\n// ============================================================================\r\n// Middleware Chain\r\n// ============================================================================\r\n\r\nexport interface MiddlewareChain {\r\n /** Add middleware to the chain */\r\n use(middleware: Middleware): MiddlewareChain;\r\n use(path: string, middleware: Middleware): MiddlewareChain;\r\n use(definition: MiddlewareDefinition): MiddlewareChain;\r\n\r\n /** Execute the middleware chain */\r\n execute(ctx: MiddlewareContext): Promise<void>;\r\n\r\n /** Get all middleware definitions */\r\n middlewares(): MiddlewareDefinition[];\r\n}\r\n\r\n/**\r\n * Create a new middleware chain\r\n */\r\nexport function createMiddlewareChain(): MiddlewareChain {\r\n const stack: MiddlewareDefinition[] = [];\r\n\r\n function use(\r\n pathOrMiddleware: string | Middleware | MiddlewareDefinition,\r\n maybeMiddleware?: Middleware\r\n ): MiddlewareChain {\r\n if (typeof pathOrMiddleware === 'function') {\r\n // use(middleware)\r\n stack.push({ handler: pathOrMiddleware });\r\n } else if (typeof pathOrMiddleware === 'string' && maybeMiddleware) {\r\n // use(path, middleware)\r\n stack.push({ path: pathOrMiddleware, handler: maybeMiddleware });\r\n } else if (typeof pathOrMiddleware === 'object') {\r\n // use(definition)\r\n stack.push(pathOrMiddleware);\r\n }\r\n return chain;\r\n }\r\n\r\n async function execute(ctx: MiddlewareContext): Promise<void> {\r\n let index = -1;\r\n\r\n async function dispatch(i: number): Promise<void> {\r\n if (i <= index) {\r\n throw new Error('next() called multiple times');\r\n }\r\n index = i;\r\n\r\n if (i >= stack.length) {\r\n return;\r\n }\r\n\r\n const definition = stack[i];\r\n if (!definition) {\r\n return dispatch(i + 1);\r\n }\r\n\r\n // Check if middleware should run\r\n if (!shouldRun(definition, ctx)) {\r\n return dispatch(i + 1);\r\n }\r\n\r\n await definition.handler(ctx, () => dispatch(i + 1));\r\n }\r\n\r\n await dispatch(0);\r\n }\r\n\r\n const chain: MiddlewareChain = {\r\n use: use as MiddlewareChain['use'],\r\n execute,\r\n middlewares: () => [...stack],\r\n };\r\n\r\n return chain;\r\n}\r\n\r\n/**\r\n * Check if a middleware should run for the given context\r\n */\r\nfunction shouldRun(def: MiddlewareDefinition, ctx: MiddlewareContext): boolean {\r\n // Check method\r\n if (def.methods && !def.methods.includes(ctx.method.toUpperCase())) {\r\n return false;\r\n }\r\n\r\n // Check path\r\n if (def.path) {\r\n const path = ctx.url.pathname;\r\n\r\n if (typeof def.path === 'string') {\r\n // Simple path matching (supports wildcards)\r\n if (def.path.endsWith('*')) {\r\n const prefix = def.path.slice(0, -1);\r\n if (!path.startsWith(prefix)) {\r\n return false;\r\n }\r\n } else if (def.path !== path) {\r\n return false;\r\n }\r\n } else if (def.path instanceof RegExp) {\r\n if (!def.path.test(path)) {\r\n return false;\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n}\r\n\r\n// ============================================================================\r\n// Built-in Middleware Factories\r\n// ============================================================================\r\n\r\n/**\r\n * CORS middleware factory\r\n */\r\nexport function cors(options: {\r\n origin?: string | string[] | ((origin: string) => boolean);\r\n methods?: string[];\r\n headers?: string[];\r\n credentials?: boolean;\r\n maxAge?: number;\r\n} = {}): Middleware {\r\n const {\r\n origin = '*',\r\n methods = ['GET', 'POST', 'PUT', 'DELETE', 'PATCH', 'OPTIONS'],\r\n headers = ['Content-Type', 'Authorization'],\r\n credentials = false,\r\n maxAge = 86400,\r\n } = options;\r\n\r\n return async (ctx, next) => {\r\n const requestOrigin = ctx.headers.get('Origin') || '';\r\n\r\n // Determine allowed origin\r\n let allowedOrigin: string | null = null;\r\n if (origin === '*') {\r\n allowedOrigin = '*';\r\n } else if (typeof origin === 'string') {\r\n allowedOrigin = origin;\r\n } else if (Array.isArray(origin)) {\r\n if (origin.includes(requestOrigin)) {\r\n allowedOrigin = requestOrigin;\r\n }\r\n } else if (typeof origin === 'function') {\r\n if (origin(requestOrigin)) {\r\n allowedOrigin = requestOrigin;\r\n }\r\n }\r\n\r\n if (allowedOrigin) {\r\n ctx.responseHeaders.set('Access-Control-Allow-Origin', allowedOrigin);\r\n ctx.responseHeaders.set('Access-Control-Allow-Methods', methods.join(', '));\r\n ctx.responseHeaders.set('Access-Control-Allow-Headers', headers.join(', '));\r\n\r\n if (credentials) {\r\n ctx.responseHeaders.set('Access-Control-Allow-Credentials', 'true');\r\n }\r\n\r\n if (maxAge) {\r\n ctx.responseHeaders.set('Access-Control-Max-Age', String(maxAge));\r\n }\r\n }\r\n\r\n // Handle preflight\r\n if (ctx.method === 'OPTIONS') {\r\n ctx.status = 204;\r\n ctx.responseBody = null;\r\n return;\r\n }\r\n\r\n await next();\r\n };\r\n}\r\n// ============================================================================\r\n// Logger Types and Constants\r\n// ============================================================================\r\n\r\n/** Log levels in order of verbosity */\r\nexport type LogLevel = 'debug' | 'info' | 'warn' | 'error' | 'silent';\r\n\r\n/** Log format types */\r\nexport type LogFormat = 'pretty' | 'json' | 'combined' | 'common' | 'short' | 'tiny';\r\n\r\n/** Log entry structure for JSON format */\r\nexport interface LogEntry {\r\n timestamp: string;\r\n level: LogLevel;\r\n method: string;\r\n path: string;\r\n status: number;\r\n duration: number;\r\n userAgent?: string;\r\n ip?: string;\r\n [key: string]: unknown;\r\n}\r\n\r\n/** Custom log writer function */\r\nexport type LogWriter = (entry: LogEntry, formatted: string) => void;\r\n\r\n/** Logger options */\r\nexport interface LoggerOptions {\r\n /** Minimum log level (default: 'info') */\r\n level?: LogLevel;\r\n /** Output format (default: 'pretty') */\r\n format?: LogFormat;\r\n /** Skip logging for certain requests */\r\n skip?: (ctx: MiddlewareContext) => boolean;\r\n /** Custom log writer (replaces console output) */\r\n writer?: LogWriter;\r\n /** Include request headers in logs */\r\n includeHeaders?: boolean;\r\n /** Colorize output (only for 'pretty' format, default: true) */\r\n colorize?: boolean;\r\n}\r\n\r\nconst LOG_LEVEL_PRIORITY: Record<LogLevel, number> = {\r\n debug: 0,\r\n info: 1,\r\n warn: 2,\r\n error: 3,\r\n silent: 4,\r\n};\r\n\r\nconst LOG_LEVEL_COLORS: Record<LogLevel, string> = {\r\n debug: '\\x1b[36m', // Cyan\r\n info: '\\x1b[32m', // Green\r\n warn: '\\x1b[33m', // Yellow\r\n error: '\\x1b[31m', // Red\r\n silent: '',\r\n};\r\n\r\nconst STATUS_COLORS: Record<string, string> = {\r\n '2': '\\x1b[32m', // Green for 2xx\r\n '3': '\\x1b[36m', // Cyan for 3xx\r\n '4': '\\x1b[33m', // Yellow for 4xx\r\n '5': '\\x1b[31m', // Red for 5xx\r\n};\r\n\r\nconst RESET = '\\x1b[0m';\r\nconst DIM = '\\x1b[2m';\r\n\r\n/**\r\n * Professional logger middleware with configurable levels\r\n * \r\n * @example\r\n * ```typescript\r\n * // Basic usage\r\n * server.use(logger());\r\n * \r\n * // With level control\r\n * server.use(logger({ level: 'warn' })); // Only warn and error\r\n * \r\n * // JSON format for production/observability\r\n * server.use(logger({ format: 'json', level: 'info' }));\r\n * \r\n * // Custom writer (e.g., for external logging service)\r\n * server.use(logger({\r\n * writer: (entry) => myLoggingService.log(entry)\r\n * }));\r\n * \r\n * // Silent in production\r\n * server.use(logger({ level: process.env.NODE_ENV === 'production' ? 'error' : 'debug' }));\r\n * ```\r\n */\r\nexport function logger(options: LoggerOptions = {}): Middleware {\r\n const {\r\n level = 'info',\r\n format = 'pretty',\r\n skip,\r\n writer,\r\n includeHeaders = false,\r\n colorize = true,\r\n } = options;\r\n\r\n return async (ctx, next) => {\r\n // Skip if configured\r\n if (skip?.(ctx)) {\r\n return next();\r\n }\r\n\r\n const start = Date.now();\r\n const startTime = new Date().toISOString();\r\n\r\n await next();\r\n\r\n const duration = Date.now() - start;\r\n const status = ctx.status;\r\n\r\n // Determine log level based on status\r\n let logLevel: LogLevel = 'info';\r\n if (status >= 500) {\r\n logLevel = 'error';\r\n } else if (status >= 400) {\r\n logLevel = 'warn';\r\n } else if (status >= 300) {\r\n logLevel = 'info';\r\n } else {\r\n logLevel = 'debug';\r\n }\r\n\r\n // Check if we should log based on level\r\n if (LOG_LEVEL_PRIORITY[logLevel] < LOG_LEVEL_PRIORITY[level]) {\r\n return;\r\n }\r\n\r\n // Build log entry\r\n const entry: LogEntry = {\r\n timestamp: startTime,\r\n level: logLevel,\r\n method: ctx.method,\r\n path: ctx.url.pathname,\r\n status,\r\n duration,\r\n };\r\n\r\n // Add optional fields\r\n const userAgent = ctx.headers.get('user-agent');\r\n if (userAgent) entry.userAgent = userAgent;\r\n\r\n const forwardedFor = ctx.headers.get('x-forwarded-for');\r\n const ip = forwardedFor || 'unknown';\r\n entry.ip = ip;\r\n\r\n if (includeHeaders) {\r\n const headers: Record<string, string> = {};\r\n ctx.headers.forEach((value, key) => {\r\n if (!key.toLowerCase().includes('authorization')) {\r\n headers[key] = value;\r\n }\r\n });\r\n entry.headers = headers;\r\n }\r\n\r\n // Format the log line\r\n let formatted: string;\r\n\r\n switch (format) {\r\n case 'json':\r\n formatted = JSON.stringify(entry);\r\n break;\r\n\r\n case 'combined':\r\n formatted = `${ip} - - [${startTime}] \"${ctx.method} ${ctx.url.pathname}\" ${status} - \"${userAgent || '-'}\"`;\r\n break;\r\n\r\n case 'common':\r\n formatted = `${ip} - - [${startTime}] \"${ctx.method} ${ctx.url.pathname}\" ${status}`;\r\n break;\r\n\r\n case 'short':\r\n formatted = `${ctx.method} ${ctx.url.pathname} ${status} ${duration}ms`;\r\n break;\r\n\r\n case 'tiny':\r\n formatted = `${ctx.method} ${ctx.url.pathname} ${status}`;\r\n break;\r\n\r\n case 'pretty':\r\n default:\r\n if (colorize) {\r\n const statusFirstChar = String(status).charAt(0) || '2';\r\n const statusColor = STATUS_COLORS[statusFirstChar] ?? '';\r\n const levelColor = LOG_LEVEL_COLORS[logLevel];\r\n formatted = `${DIM}${startTime}${RESET} ${levelColor}${logLevel.toUpperCase().padEnd(5)}${RESET} ${ctx.method.padEnd(7)} ${ctx.url.pathname} ${statusColor}${status}${RESET} ${DIM}${duration}ms${RESET}`;\r\n } else {\r\n formatted = `${startTime} ${logLevel.toUpperCase().padEnd(5)} ${ctx.method.padEnd(7)} ${ctx.url.pathname} ${status} ${duration}ms`;\r\n }\r\n break;\r\n }\r\n\r\n // Output\r\n if (writer) {\r\n writer(entry, formatted);\r\n } else {\r\n switch (logLevel) {\r\n case 'error':\r\n console.error(formatted);\r\n break;\r\n case 'warn':\r\n console.warn(formatted);\r\n break;\r\n case 'debug':\r\n console.debug(formatted);\r\n break;\r\n default:\r\n console.log(formatted);\r\n }\r\n }\r\n };\r\n}\r\n\r\n/**\r\n * Security headers middleware\r\n */\r\nexport function securityHeaders(options: {\r\n contentSecurityPolicy?: string | false;\r\n xFrameOptions?: 'DENY' | 'SAMEORIGIN' | false;\r\n xContentTypeOptions?: boolean;\r\n referrerPolicy?: string;\r\n permissionsPolicy?: string;\r\n} = {}): Middleware {\r\n const {\r\n contentSecurityPolicy = \"default-src 'self'\",\r\n xFrameOptions = 'DENY',\r\n xContentTypeOptions = true,\r\n referrerPolicy = 'strict-origin-when-cross-origin',\r\n permissionsPolicy,\r\n } = options;\r\n\r\n return async (ctx, next) => {\r\n if (contentSecurityPolicy) {\r\n ctx.responseHeaders.set('Content-Security-Policy', contentSecurityPolicy);\r\n }\r\n if (xFrameOptions) {\r\n ctx.responseHeaders.set('X-Frame-Options', xFrameOptions);\r\n }\r\n if (xContentTypeOptions) {\r\n ctx.responseHeaders.set('X-Content-Type-Options', 'nosniff');\r\n }\r\n if (referrerPolicy) {\r\n ctx.responseHeaders.set('Referrer-Policy', referrerPolicy);\r\n }\r\n if (permissionsPolicy) {\r\n ctx.responseHeaders.set('Permissions-Policy', permissionsPolicy);\r\n }\r\n\r\n await next();\r\n };\r\n}\r\n\r\n/**\r\n * Compression middleware (requires implementation in adapter)\r\n */\r\nexport function compress(): Middleware {\r\n return async (ctx, next) => {\r\n // Mark that compression is desired\r\n ctx.locals['flight:compress'] = true;\r\n await next();\r\n };\r\n}\r\n\r\n// ============================================================================\r\n// Utility Functions\r\n// ============================================================================\r\n\r\n/**\r\n * Create a middleware context from a Web Request\r\n */\r\nexport function createContextFromRequest(\r\n request: Request,\r\n params: Record<string, string | string[]> = {}\r\n): MiddlewareContext {\r\n const url = new URL(request.url);\r\n\r\n return {\r\n url,\r\n method: request.method,\r\n headers: request.headers,\r\n params,\r\n query: url.searchParams,\r\n locals: {},\r\n request,\r\n status: 200,\r\n responseHeaders: new Headers(),\r\n };\r\n}\r\n\r\n/**\r\n * Create a Web Response from middleware context\r\n */\r\nexport function createResponseFromContext(ctx: MiddlewareContext): Response {\r\n return new Response(ctx.responseBody, {\r\n status: ctx.status,\r\n headers: ctx.responseHeaders,\r\n });\r\n}\r\n\r\n/**\r\n * Compose multiple middleware into one\r\n */\r\nexport function compose(...middlewares: Middleware[]): Middleware {\r\n return async (ctx, next) => {\r\n let index = -1;\r\n\r\n async function dispatch(i: number): Promise<void> {\r\n if (i <= index) {\r\n throw new Error('next() called multiple times');\r\n }\r\n index = i;\r\n\r\n const fn = i === middlewares.length ? next : middlewares[i];\r\n\r\n if (fn) {\r\n await fn(ctx, () => dispatch(i + 1));\r\n }\r\n }\r\n\r\n await dispatch(0);\r\n };\r\n}\r\n"]}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
// src/adapters/index.ts
|
|
2
|
+
function createAdapter(options) {
|
|
3
|
+
return {
|
|
4
|
+
name: options.name,
|
|
5
|
+
adapt: options.adapt,
|
|
6
|
+
emulate: options.emulate,
|
|
7
|
+
supports: options.supports
|
|
8
|
+
};
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export { createAdapter };
|
|
12
|
+
//# sourceMappingURL=chunk-Q4C5CCHK.js.map
|
|
13
|
+
//# sourceMappingURL=chunk-Q4C5CCHK.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/adapters/index.ts"],"names":[],"mappings":";AA2IO,SAAS,cAAc,OAAA,EAAwC;AAClE,EAAA,OAAO;AAAA,IACH,MAAM,OAAA,CAAQ,IAAA;AAAA,IACd,OAAO,OAAA,CAAQ,KAAA;AAAA,IACf,SAAS,OAAA,CAAQ,OAAA;AAAA,IACjB,UAAU,OAAA,CAAQ;AAAA,GACtB;AACJ","file":"chunk-Q4C5CCHK.js","sourcesContent":["/**\r\n * Flight Adapters - Universal deployment adapters\r\n * \r\n * Adapters transform Flight's output for different deployment targets.\r\n * The user chooses where to deploy - Flight provides the adapters.\r\n */\r\n\r\n// ============================================================================\r\n// Types\r\n// ============================================================================\r\n\r\n/** Build manifest containing all generated assets */\r\nexport interface BuildManifest {\r\n /** Entry points by name */\r\n entries: Record<string, string>;\r\n /** All generated files */\r\n files: string[];\r\n /** Route information */\r\n routes: RouteManifestEntry[];\r\n /** Server entry point (if applicable) */\r\n serverEntry?: string;\r\n /** Client entry point */\r\n clientEntry?: string;\r\n}\r\n\r\n/** Route information in manifest */\r\nexport interface RouteManifestEntry {\r\n /** Route path pattern */\r\n path: string;\r\n /** Render mode for this route */\r\n mode: 'ssr' | 'ssg' | 'csr' | 'isr';\r\n /** Component/handler file path */\r\n component: string;\r\n /** Pre-rendered paths (for SSG/ISR) */\r\n prerendered?: string[];\r\n}\r\n\r\n/** Builder utilities passed to adapters */\r\nexport interface AdapterBuilder {\r\n /** Build manifest */\r\n manifest: BuildManifest;\r\n /** Project root directory */\r\n root: string;\r\n /** Build output directory */\r\n outDir: string;\r\n\r\n /** Read a file from the build output */\r\n readFile(path: string): Promise<string>;\r\n /** Write a file to the adapter output */\r\n writeFile(path: string, content: string): Promise<void>;\r\n /** Copy files from build to adapter output */\r\n copy(from: string, to: string): Promise<void>;\r\n /** Get all files matching a pattern */\r\n glob(pattern: string): Promise<string[]>;\r\n /** Compress files for production */\r\n compress?(files: string[]): Promise<void>;\r\n\r\n /** Log info message */\r\n log: {\r\n info(message: string): void;\r\n warn(message: string): void;\r\n error(message: string): void;\r\n };\r\n}\r\n\r\n/** Flight adapter interface */\r\nexport interface FlightAdapter {\r\n /** Adapter name */\r\n name: string;\r\n\r\n /** \r\n * Transform the build output for the target platform\r\n * This is called after Flight's build process completes\r\n */\r\n adapt(builder: AdapterBuilder): Promise<void>;\r\n\r\n /**\r\n * Optional: Start listening for requests (for Node.js/Bun adapters)\r\n */\r\n listen?(server: unknown, port: number): Promise<void>;\r\n\r\n /**\r\n * Optional: Emulate the platform during development\r\n * Useful for testing platform-specific behavior locally\r\n */\r\n emulate?(): {\r\n /** Platform-specific env vars */\r\n env?: Record<string, string>;\r\n /** Custom middleware for dev server */\r\n middleware?: unknown[];\r\n };\r\n\r\n /**\r\n * Optional: Declare what this adapter supports\r\n * Flight uses this to warn about incompatible features\r\n */\r\n supports?: {\r\n /** Can read files at runtime (for dynamic content) */\r\n read?: () => boolean;\r\n /** Supports streaming responses */\r\n streaming?: () => boolean;\r\n /** Supports WebSockets */\r\n websockets?: () => boolean;\r\n /** Supports edge runtime */\r\n edge?: () => boolean;\r\n /** Supports Node.js runtime */\r\n node?: () => boolean;\r\n };\r\n}\r\n\r\n\r\n// ============================================================================\r\n// Adapter Factory Helper\r\n// ============================================================================\r\n\r\nexport interface AdapterOptions {\r\n name: string;\r\n adapt: (builder: AdapterBuilder) => Promise<void>;\r\n emulate?: FlightAdapter['emulate'];\r\n supports?: FlightAdapter['supports'];\r\n}\r\n\r\n/**\r\n * Create a Flight adapter\r\n * \r\n * @example\r\n * ```typescript\r\n * import { createAdapter } from '@flight/core/adapters';\r\n * \r\n * export default function myAdapter(options = {}) {\r\n * return createAdapter({\r\n * name: 'my-adapter',\r\n * async adapt(builder) {\r\n * // Transform build output for your platform\r\n * },\r\n * });\r\n * }\r\n * ```\r\n */\r\nexport function createAdapter(options: AdapterOptions): FlightAdapter {\r\n return {\r\n name: options.name,\r\n adapt: options.adapt,\r\n emulate: options.emulate,\r\n supports: options.supports,\r\n };\r\n}\r\n\r\n// ============================================================================\r\n// Agnostic Service Adapters (Interfaces)\r\n// ============================================================================\r\n\r\n/**\r\n * Storage Adapter Interface\r\n * \r\n * Implement this to use any storage provider:\r\n * S3, Supabase Storage, Cloudflare R2, local filesystem, etc.\r\n */\r\nexport interface StorageAdapter {\r\n name: string;\r\n\r\n /** Upload a file */\r\n upload(path: string, data: Buffer | Uint8Array | string, options?: {\r\n contentType?: string;\r\n metadata?: Record<string, string>;\r\n }): Promise<{ url: string; path: string }>;\r\n\r\n /** Download a file */\r\n download(path: string): Promise<Buffer>;\r\n\r\n /** Delete a file */\r\n delete(path: string): Promise<void>;\r\n\r\n /** List files with optional prefix */\r\n list(prefix?: string): Promise<string[]>;\r\n\r\n /** Get a signed URL for direct access */\r\n getSignedUrl?(path: string, options?: {\r\n expiresIn?: number;\r\n operation?: 'read' | 'write';\r\n }): Promise<string>;\r\n}\r\n\r\n/**\r\n * Auth Adapter Interface\r\n * \r\n * Implement this to use any auth provider:\r\n * Better Auth, Supabase Auth, Lucia, Auth.js, etc.\r\n */\r\nexport interface AuthAdapter {\r\n name: string;\r\n\r\n /** Get the current user from a request */\r\n getUser(request: Request): Promise<AuthUser | null>;\r\n\r\n /** Verify a session token */\r\n verifySession(token: string): Promise<AuthSession | null>;\r\n\r\n /** Optional middleware for handling auth-specific routes (e.g., /api/auth/*) */\r\n middleware?: (req: Request) => Promise<Response | null>;\r\n}\r\n\r\nexport interface AuthUser {\r\n id: string;\r\n email?: string;\r\n name?: string;\r\n avatar?: string;\r\n metadata?: Record<string, unknown>;\r\n}\r\n\r\nexport interface AuthSession {\r\n user: AuthUser;\r\n expiresAt: Date;\r\n token: string;\r\n}\r\n\r\n/**\r\n * Email Adapter Interface\r\n * \r\n * Implement this to use any email provider:\r\n * Resend, SendGrid, Postmark, SMTP, etc.\r\n */\r\nexport interface EmailAdapter {\r\n name: string;\r\n\r\n /** Send an email */\r\n send(options: {\r\n to: string | string[];\r\n subject: string;\r\n html?: string;\r\n text?: string;\r\n from?: string;\r\n replyTo?: string;\r\n attachments?: Array<{\r\n filename: string;\r\n content: Buffer | string;\r\n contentType?: string;\r\n }>;\r\n }): Promise<{ id: string; success: boolean }>;\r\n}\r\n\r\n/**\r\n * Queue/Jobs Adapter Interface\r\n * \r\n * Implement this to use any job queue:\r\n * BullMQ, Quirrel, Inngest, custom, etc.\r\n */\r\nexport interface JobsAdapter {\r\n name: string;\r\n\r\n /** Add a job to the queue */\r\n enqueue<T>(jobName: string, data: T, options?: {\r\n delay?: number;\r\n priority?: number;\r\n retries?: number;\r\n }): Promise<{ id: string }>;\r\n\r\n /** Schedule a recurring job */\r\n schedule?(jobName: string, cron: string, data?: unknown): Promise<{ id: string }>;\r\n\r\n /** Cancel a job */\r\n cancel?(jobId: string): Promise<void>;\r\n}\r\n\r\n/**\r\n * Database Adapter Interface\r\n * \r\n * Note: This is intentionally minimal. \r\n * For databases, use your preferred ORM/query builder directly.\r\n * This interface is for Flight's internal use (sessions, cache, etc.)\r\n */\r\nexport interface DatabaseAdapter {\r\n name: string;\r\n\r\n /** Raw query execution */\r\n query<T = unknown>(sql: string, params?: unknown[]): Promise<T[]>;\r\n\r\n /** Execute a mutation */\r\n execute(sql: string, params?: unknown[]): Promise<{ rowsAffected: number }>;\r\n\r\n /** Transaction support */\r\n transaction?<T>(fn: (tx: DatabaseAdapter) => Promise<T>): Promise<T>;\r\n}\r\n"]}
|