@gravito/core 2.0.4 → 3.0.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.
- package/dist/PlanetCore.d.ts +6 -0
- package/dist/compat/async-local-storage.browser.js +71 -0
- package/dist/compat/async-local-storage.browser.js.map +10 -0
- package/dist/compat/async-local-storage.js +94 -0
- package/dist/compat/async-local-storage.js.map +10 -0
- package/dist/compat/crypto.browser.js +91 -0
- package/dist/compat/crypto.browser.js.map +10 -0
- package/dist/compat/crypto.js +111 -0
- package/dist/compat/crypto.js.map +10 -0
- package/dist/events/CircuitBreaker.d.ts +12 -0
- package/dist/exceptions/AuthException.d.ts +10 -0
- package/dist/exceptions/AuthenticationException.d.ts +2 -2
- package/dist/exceptions/AuthorizationException.d.ts +2 -2
- package/dist/exceptions/CacheException.d.ts +9 -0
- package/dist/exceptions/CircularDependencyException.d.ts +2 -1
- package/dist/exceptions/ConfigurationException.d.ts +9 -0
- package/dist/exceptions/DatabaseException.d.ts +9 -0
- package/dist/exceptions/DomainException.d.ts +9 -0
- package/dist/exceptions/InfrastructureException.d.ts +17 -0
- package/dist/exceptions/QueueException.d.ts +9 -0
- package/dist/exceptions/StorageException.d.ts +9 -0
- package/dist/exceptions/StreamException.d.ts +9 -0
- package/dist/exceptions/SystemException.d.ts +9 -0
- package/dist/exceptions/ValidationException.d.ts +2 -2
- package/dist/exceptions/index.d.ts +10 -0
- package/dist/ffi/NativeAccelerator.js +398 -0
- package/dist/ffi/NativeAccelerator.js.map +12 -0
- package/dist/ffi/NativeHasher.js +125 -0
- package/dist/ffi/{index.js.map → NativeHasher.js.map} +2 -2
- package/dist/ffi/cbor-fallback.js +344 -0
- package/dist/ffi/cbor-fallback.js.map +11 -0
- package/dist/ffi/hash-fallback.js +63 -0
- package/dist/ffi/hash-fallback.js.map +10 -0
- package/dist/ffi/index.js +5 -131
- package/dist/ffi/types.js +82 -0
- package/dist/ffi/types.js.map +10 -0
- package/dist/index.browser.js +9055 -0
- package/dist/index.browser.js.map +90 -0
- package/dist/index.d.ts +479 -1
- package/dist/index.js +401 -158
- package/dist/index.js.map +27 -16
- package/dist/testing/HttpTester.d.ts +1 -0
- package/package.json +7 -7
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../src/compat/async-local-storage.ts", "../src/compat/crypto.ts", "../src/adapters/bun/types.ts", "../src/adapters/bun/RadixNode.ts", "../src/adapters/bun/RadixRouter.ts", "../src/engine/AOTRouter.ts", "../src/transpiler-utils.ts", "../src/engine/analyzer.ts", "../src/engine/constants.ts", "../src/Container/RequestScopeMetrics.ts", "../src/Container/RequestScopeManager.ts", "../src/engine/FastContext.ts", "../src/engine/MinimalContext.ts", "../src/engine/path.ts", "../src/engine/pool.ts", "../src/engine/Gravito.ts", "../src/adapters/GravitoEngineAdapter.ts", "../src/adapters/types.ts", "../src/runtime/adapter-bun.ts", "../src/runtime/detection.ts", "../src/runtime/adapter-deno.ts", "../src/runtime/adapter-node.ts", "../src/runtime/adapter-unknown.ts", "../src/runtime/archive.ts", "../src/runtime/compression.ts", "../src/runtime/markdown.ts", "../src/runtime/deep-equals.ts", "../src/runtime/escape.ts", "../src/runtime/index.ts", "../src/ConfigManager.ts", "../src/exceptions/GravitoException.ts", "../src/exceptions/SystemException.ts", "../src/exceptions/CircularDependencyException.ts", "../src/Container.ts", "../src/cli/queue-commands.ts", "../src/error-handling/RequestScopeErrorContext.ts", "../src/exceptions/HttpException.ts", "../src/exceptions/DomainException.ts", "../src/exceptions/ValidationException.ts", "../src/helpers/response.ts", "../src/ErrorHandler.ts", "../src/types/events.ts", "../src/EventManager.ts", "../src/events/EventOptions.ts", "../src/events/BackpressureManager.ts", "../src/events/CircuitBreaker.ts", "../src/events/DeadLetterQueue.ts", "../src/events/PriorityEscalationManager.ts", "../src/events/queue-core.ts", "../src/events/task-executor.ts", "../src/events/EventPriorityQueue.ts", "../src/events/FlowControlStrategy.ts", "../src/events/IdempotencyCache.ts", "../src/events/MessageQueueBridge.ts", "../src/events/RetryScheduler.ts", "../src/events/WorkerPoolMetrics.ts", "../src/events/WorkerPool.ts", "../src/events/aggregation/types.ts", "../src/events/aggregation/AggregationWindow.ts", "../src/events/aggregation/DeduplicationManager.ts", "../src/events/aggregation/EventBatcher.ts", "../src/events/aggregation/EventAggregationManager.ts", "../src/hooks/ActionManager.ts", "../src/hooks/AsyncDetector.ts", "../src/hooks/dlq-operations.ts", "../src/hooks/FilterManager.ts", "../src/hooks/MigrationWarner.ts", "../src/reliability/RetryPolicy.ts", "../src/reliability/DeadLetterQueueManager.ts", "../src/HookManager.ts", "../src/ServiceProvider.ts", "../src/health/HealthProvider.ts", "../src/helpers/data.ts", "../src/helpers/Arr.ts", "../src/helpers/errors.ts", "../src/helpers/Str.ts", "../src/helpers.ts", "../src/http/CookieJar.ts", "../src/http/cookie.ts", "../src/runtime/index.browser.ts", "../src/index.browser.ts"],
|
|
4
|
+
"sourcesContent": [
|
|
5
|
+
"/**\n * Universal AsyncLocalStorage wrapper.\n * Automatically switches between node:async_hooks and a browser mock.\n */\n\nlet AsyncLocalStorageClass: any\n\n// Try to load Node.js AsyncLocalStorage\nconst tryGetNodeAsyncHooks = () => {\n try {\n if (\n typeof window === 'undefined' &&\n typeof process !== 'undefined' &&\n !(process as any).browser\n ) {\n // Try direct import for Bun/Node.js ESM compatibility\n try {\n const module = require('node:async_hooks')\n return module.AsyncLocalStorage\n } catch (_e1) {\n // Fallback to eval for CommonJS require hiding\n try {\n // biome-ignore lint/security/noGlobalEval: specialized case for hiding node built-ins\n return eval('require')('node:async_hooks').AsyncLocalStorage\n } catch (_e2) {\n return null\n }\n }\n }\n } catch (_e) {\n return null\n }\n}\n\nAsyncLocalStorageClass = tryGetNodeAsyncHooks()\n\nif (!AsyncLocalStorageClass) {\n /**\n * Browser-safe AsyncLocalStorage mock.\n * Note: This mock only works with synchronous functions.\n * For proper async support, ensure Node.js AsyncLocalStorage is available.\n */\n AsyncLocalStorageClass = class AsyncLocalStorage<T> {\n private store: T | undefined\n\n run<R>(store: T, fn: () => R): R {\n const prev = this.store\n this.store = store\n try {\n return fn()\n } finally {\n this.store = prev\n }\n }\n\n getStore(): T | undefined {\n return this.store\n }\n\n disable(): void {\n this.store = undefined\n }\n }\n}\n\n// biome-ignore lint/suspicious/noExplicitAny: generic class mock\nexport const AsyncLocalStorage: { new <_T>(): any } = AsyncLocalStorageClass\n",
|
|
6
|
+
"/**\n * Universal Crypto wrapper.\n * Automatically switches between node:crypto and globalThis.crypto.\n */\n\nlet randomUUIDFn: () => string\nlet randomBytesFn: (size: number) => any\n\nconst tryGetNodeCrypto = () => {\n try {\n if (\n typeof window === 'undefined' &&\n typeof process !== 'undefined' &&\n !(process as any).browser\n ) {\n // biome-ignore lint/security/noGlobalEval: specialized case for hiding node built-ins\n return eval('require')('node:crypto')\n }\n } catch (_e) {\n return null\n }\n}\n\nconst nodeCrypto = tryGetNodeCrypto()\n\nif (nodeCrypto) {\n randomUUIDFn = nodeCrypto.randomUUID\n randomBytesFn = nodeCrypto.randomBytes\n} else {\n // Browser implementations\n randomUUIDFn = () => {\n if (typeof globalThis.crypto?.randomUUID === 'function') {\n return globalThis.crypto.randomUUID()\n }\n return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {\n const r = (Math.random() * 16) | 0\n const v = c === 'x' ? r : (r & 0x3) | 0x8\n return v.toString(16)\n })\n }\n\n randomBytesFn = (size: number) => {\n const bytes = new Uint8Array(size)\n if (typeof globalThis.crypto?.getRandomValues === 'function') {\n globalThis.crypto.getRandomValues(bytes)\n } else {\n for (let i = 0; i < size; i++) {\n bytes[i] = Math.floor(Math.random() * 256)\n }\n }\n // Return a Buffer-like object\n return {\n ...Array.from(bytes),\n length: size,\n [Symbol.iterator]: () => bytes[Symbol.iterator](),\n toString: (encoding: string) => {\n if (encoding === 'base64') {\n let binary = ''\n for (let i = 0; i < bytes.byteLength; i++) {\n binary += String.fromCharCode(bytes[i])\n }\n return btoa(binary)\n }\n if (encoding === 'hex') {\n return Array.from(bytes)\n .map((b) => b.toString(16).padStart(2, '0'))\n .join('')\n }\n return ''\n },\n }\n }\n}\n\nexport const randomUUID = randomUUIDFn\nexport const randomBytes = randomBytesFn\n",
|
|
7
|
+
"/**\n * Route Handler type (simplified for internal use)\n * In the full framework this will align with GravitoHandler\n */\nexport type RouteHandler = Function\n\n/**\n * Route Match Result\n */\nexport interface RouteMatch {\n handlers: RouteHandler[]\n params: Record<string, string>\n}\n\n/**\n * Radix Node Type\n */\nexport enum NodeType {\n STATIC = 0,\n PARAM = 1,\n WILDCARD = 2,\n}\n",
|
|
8
|
+
"import type { HttpMethod } from '../../http/types'\nimport { NodeType, type RouteHandler } from './types'\n\n/**\n * Node in the Radix Router tree.\n * @internal\n */\nexport class RadixNode {\n // Path segment for this node (e.g., \"users\", \":id\")\n public segment: string\n\n // Node type (Static, Param, Wildcard)\n public type: NodeType\n\n // Children nodes (mapped by segment for fast lookup)\n public children: Map<string, RadixNode> = new Map()\n\n // Specialized child for parameter node (only one per level allowed usually to avoid ambiguity, though some routers support multiple)\n public paramChild: RadixNode | null = null\n\n // Specialized child for wildcard node\n public wildcardChild: RadixNode | null = null\n\n // Handlers registered at this node (keyed by HTTP method)\n public handlers: Map<HttpMethod, RouteHandler[]> = new Map()\n\n // Parameter name if this is a PARAM node (e.g., \"id\" for \":id\")\n public paramName: string | null = null\n\n // Parameter constraints (regex) - only applicable if this is a PARAM node\n // If we support per-route constraints, they might need to be stored differently,\n // but for now assume constraints are defined at node level (uncommon) or checked at match time.\n // Laravel allows global pattern constraints or per-route.\n // Ideally, constraints should be stored with the handler or part of matching logic.\n // For a Radix tree, if we have constraints, we might need to backtrack if constraint fails?\n // Or simply store constraint with the param node.\n public regex: RegExp | null = null\n\n constructor(segment = '', type: NodeType = NodeType.STATIC) {\n this.segment = segment\n this.type = type\n }\n\n toJSON(): any {\n return {\n segment: this.segment,\n type: this.type,\n children: Array.from(this.children.entries()).map(([k, v]) => [k, v.toJSON()]),\n paramChild: this.paramChild?.toJSON() || null,\n wildcardChild: this.wildcardChild?.toJSON() || null,\n paramName: this.paramName,\n regex: this.regex ? this.regex.source : null,\n }\n }\n\n static fromJSON(json: any): RadixNode {\n const node = new RadixNode(json.segment, json.type)\n node.paramName = json.paramName\n if (json.regex) {\n node.regex = new RegExp(json.regex)\n }\n if (json.children) {\n for (const [key, childJson] of json.children) {\n node.children.set(key, RadixNode.fromJSON(childJson))\n }\n }\n if (json.paramChild) {\n node.paramChild = RadixNode.fromJSON(json.paramChild)\n }\n if (json.wildcardChild) {\n node.wildcardChild = RadixNode.fromJSON(json.wildcardChild)\n }\n return node\n }\n}\n",
|
|
9
|
+
"import type { HttpMethod } from '../../http/types'\nimport { RadixNode } from './RadixNode'\nimport { NodeType, type RouteHandler, type RouteMatch } from './types'\n\n/**\n * Simple LRU Cache for route matching\n * @internal\n */\nclass RouteCache {\n private cache = new Map<string, RouteMatch | null>()\n private readonly maxSize = 10000\n\n get(key: string): RouteMatch | null | undefined {\n return this.cache.get(key)\n }\n\n set(key: string, value: RouteMatch | null): void {\n if (this.cache.size >= this.maxSize) {\n // Remove first (oldest) entry\n const firstKey = this.cache.keys().next().value\n if (firstKey) {\n this.cache.delete(firstKey)\n }\n }\n this.cache.set(key, value)\n }\n\n clear(): void {\n this.cache.clear()\n }\n\n has(key: string): boolean {\n return this.cache.has(key)\n }\n}\n\n/**\n * High-performance Radix Tree Router for Bun\n */\nexport class RadixRouter {\n private root: RadixNode = new RadixNode()\n\n // Global parameter constraints (e.g., id => /^\\d+$/)\n private globalConstraints: Map<string, RegExp> = new Map()\n\n // Route cache (P2 optimization)\n private routeCache = new RouteCache()\n\n /**\n * Add a generic parameter constraint\n */\n where(param: string, regex: RegExp): void {\n this.globalConstraints.set(param, regex)\n }\n\n /**\n * Register a route\n */\n add(method: HttpMethod, path: string, handlers: RouteHandler[]): void {\n let node = this.root\n const segments = this.splitPath(path)\n\n for (let i = 0; i < segments.length; i++) {\n const segment = segments[i]!\n\n // Determine node type\n if (segment === '*') {\n // Wildcard\n if (!node.wildcardChild) {\n node.wildcardChild = new RadixNode('*', NodeType.WILDCARD)\n }\n node = node.wildcardChild\n break\n } else if (segment.startsWith(':')) {\n // Parameter\n const paramName = segment.slice(1)\n if (!node.paramChild) {\n const child = new RadixNode(segment, NodeType.PARAM)\n child.paramName = paramName\n // Apply global constraint if exists\n const constraint = this.globalConstraints.get(paramName)\n if (constraint) {\n child.regex = constraint\n }\n node.paramChild = child\n }\n node = node.paramChild!\n } else {\n // Static\n if (!node.children.has(segment)) {\n node.children.set(segment, new RadixNode(segment, NodeType.STATIC))\n }\n node = node.children.get(segment)!\n }\n }\n\n // Ensure we store handlers for lowercase method\n node.handlers.set(method.toLowerCase() as HttpMethod, handlers)\n\n // P2 optimization: Clear cache when new route added\n this.routeCache.clear()\n }\n\n /**\n * Match a request\n */\n match(method: string, path: string): RouteMatch | null {\n const normalizedMethod = method.toLowerCase() as HttpMethod\n\n // Fast path for root\n if (path === '/' || path === '') {\n const handlers = this.root.handlers.get(normalizedMethod)\n if (handlers) {\n return { handlers, params: {} }\n }\n return null\n }\n\n // P2 optimization: Check route cache first\n const cacheKey = `${normalizedMethod}:${path}`\n if (this.routeCache.has(cacheKey)) {\n return this.routeCache.get(cacheKey) ?? null\n }\n\n // Removing leading slash for faster segmenting\n const searchPath = path.startsWith('/') ? path.slice(1) : path\n const segments = searchPath.split('/')\n\n const result = this.matchRecursive(this.root, segments, 0, {}, normalizedMethod)\n\n // P2 optimization: Cache the result\n this.routeCache.set(cacheKey, result)\n\n return result\n }\n\n private matchRecursive(\n node: RadixNode,\n segments: string[],\n depth: number,\n params: Record<string, string>,\n method: HttpMethod\n ): RouteMatch | null {\n // 1. Check if we reached end of path\n if (depth >= segments.length) {\n let handlers = node.handlers.get(method)\n if (!handlers) {\n handlers = node.handlers.get('all' as HttpMethod)\n }\n if (handlers) {\n return { handlers, params }\n }\n return null\n }\n\n const segment = segments[depth]!\n\n // 2. Try Static Match\n const staticChild = node.children.get(segment)\n if (staticChild) {\n const match = this.matchRecursive(staticChild, segments, depth + 1, params, method)\n if (match) {\n return match\n }\n }\n\n // 3. Try Param Match\n const paramChild = node.paramChild\n if (paramChild) {\n // Check constraints\n if (paramChild.regex && !paramChild.regex.test(segment)) {\n // Constraint failed, continue to wildcard\n } else {\n // Capture param\n if (paramChild.paramName) {\n params[paramChild.paramName] = decodeURIComponent(segment)\n const match = this.matchRecursive(paramChild, segments, depth + 1, params, method)\n if (match) {\n return match\n }\n // Backtrack\n delete params[paramChild.paramName]\n }\n }\n }\n\n // 4. Try Wildcard Match\n if (node.wildcardChild) {\n let handlers = node.wildcardChild.handlers.get(method)\n if (!handlers) {\n handlers = node.wildcardChild.handlers.get('all' as HttpMethod)\n }\n if (handlers) {\n return { handlers, params }\n }\n }\n\n return null\n }\n\n private splitPath(path: string): string[] {\n if (path === '/' || path === '') {\n return []\n }\n let p = path\n if (p.startsWith('/')) {\n p = p.slice(1)\n }\n if (p.endsWith('/')) {\n p = p.slice(0, -1)\n }\n return p.split('/')\n }\n\n /**\n * Serialize the router to a JSON string\n */\n serialize(): string {\n return JSON.stringify({\n root: this.root.toJSON(),\n globalConstraints: Array.from(this.globalConstraints.entries()).map(([k, v]) => [\n k,\n v.source,\n ]),\n })\n }\n\n /**\n * Restore a router from a serialized JSON string\n */\n static fromSerialized(json: string): RadixRouter {\n const data = JSON.parse(json)\n const router = new RadixRouter()\n\n router.root = RadixNode.fromJSON(data.root)\n\n if (data.globalConstraints) {\n for (const [key, source] of data.globalConstraints) {\n router.globalConstraints.set(key, new RegExp(source))\n }\n }\n\n return router\n }\n}\n",
|
|
10
|
+
"/**\n * @fileoverview AOT (Ahead-of-Time) Router\n *\n * Hybrid routing strategy:\n * - Static routes: O(1) Map lookup\n * - Dynamic routes: Optimized Radix Tree\n *\n * The key optimization is separating static from dynamic routes at registration time,\n * not at match time. This eliminates unnecessary tree traversal for static paths.\n *\n * @module @gravito/core/engine\n */\n\nimport { RadixRouter } from '../adapters/bun/RadixRouter'\nimport type { HttpMethod } from '../http/types'\nimport type { Handler, Middleware, RouteMatch, RouteMetadata } from './types'\n\n/**\n * Route definition for re-playing routes (mounting)\n */\ninterface RouteDefinition {\n method: HttpMethod\n path: string\n handler: Handler\n middleware: Middleware[]\n}\n\n/**\n * AOT Router - Optimized for Bun\n */\nexport class AOTRouter {\n // Static route cache: \"METHOD:PATH\" -> RouteMetadata\n /** @internal */\n public readonly staticRoutes = new Map<string, RouteMetadata>()\n\n // Dynamic route handler (Radix Tree)\n private dynamicRouter = new RadixRouter()\n\n // Store all route definitions to support mounting/merging\n /** @internal */\n public readonly routeDefinitions: RouteDefinition[] = []\n\n // Global middleware (applies to all routes)\n /** @internal */\n public readonly globalMiddleware: Middleware[] = []\n\n // Path-based middleware: pattern -> middleware[]\n /** @internal */\n public readonly pathMiddleware = new Map<string, Middleware[]>()\n\n // Dynamic route patterns: handler function -> route pattern\n // 用於追蹤動態路由的模式,防止高基數問題\n private dynamicRoutePatterns = new Map<Function, string>()\n\n private middlewareCache = new Map<string, { data: Middleware[]; version: number }>()\n private cacheMaxSize = 1000\n private _version = 0\n\n /**\n * Get the current version for cache invalidation\n * Incremented whenever middleware or routes are modified\n */\n public get version(): number {\n return this._version\n }\n\n /**\n * Register a route\n *\n * Automatically determines if route is static or dynamic.\n * Static routes are stored in a Map for O(1) lookup.\n * Dynamic routes use the Radix Tree.\n *\n * @param method - HTTP method\n * @param path - Route path\n * @param handler - Route handler\n * @param middleware - Route-specific middleware\n */\n add(method: HttpMethod, path: string, handler: Handler, middleware: Middleware[] = []): void {\n // Store definition for mounting support\n this.routeDefinitions.push({ method, path, handler, middleware })\n\n const normalizedMethod = method.toLowerCase() as HttpMethod\n\n if (this.isStaticPath(path)) {\n // Static route - use Map\n const key = `${normalizedMethod}:${path}`\n this.staticRoutes.set(key, { handler, middleware })\n } else {\n // Dynamic route - use Radix Tree\n // Wrap handler to match our Handler type\n const wrappedHandler = handler as unknown as Function\n this.dynamicRouter.add(normalizedMethod, path, [wrappedHandler])\n\n // Store route pattern for this handler\n this.dynamicRoutePatterns.set(wrappedHandler, path)\n\n // Store middleware separately\n if (middleware.length > 0) {\n this.pathMiddleware.set(`${normalizedMethod}:${path}`, middleware)\n }\n }\n }\n\n /**\n * Mount another router at a prefix\n */\n mount(prefix: string, other: AOTRouter): void {\n // 1. Convert other's global middleware to pattern middleware\n if (other.globalMiddleware.length > 0) {\n // Apply to both /prefix and /prefix/*\n this.usePattern(prefix, ...other.globalMiddleware)\n\n const wildcard = prefix === '/' ? '/*' : `${prefix}/*`\n this.usePattern(wildcard, ...other.globalMiddleware)\n }\n\n // 2. Transfer pattern-based middleware (only user-defined patterns, not route-specific)\n for (const [pattern, mws] of other.pathMiddleware) {\n // Skip route-specific middleware entries (they have method prefix like \"get:/path/:id\")\n // These are stored by add() with format \"method:path\"\n const hasMethodPrefix = /^(get|post|put|delete|patch|options|head):/.test(pattern)\n if (hasMethodPrefix) {\n continue\n }\n\n // Normalize pattern\n let newPattern: string\n if (pattern === '*') {\n newPattern = prefix === '/' ? '/*' : `${prefix}/*`\n } else if (pattern.startsWith('/')) {\n newPattern = prefix === '/' ? pattern : `${prefix}${pattern}`\n } else {\n newPattern = prefix === '/' ? `/${pattern}` : `${prefix}/${pattern}`\n }\n\n this.usePattern(newPattern, ...mws)\n }\n\n // 3. Transfer all routes\n for (const def of other.routeDefinitions) {\n // Calculate new path\n let newPath: string\n if (prefix === '/') {\n newPath = def.path\n } else if (def.path === '/') {\n newPath = prefix\n } else {\n newPath = `${prefix}${def.path}`\n }\n\n this.add(def.method, newPath, def.handler, def.middleware)\n }\n }\n\n /**\n * Add global middleware\n *\n * These run for every request, before route-specific middleware.\n *\n * @param middleware - Middleware functions\n */\n use(...middleware: Middleware[]): void {\n this.globalMiddleware.push(...middleware)\n this._version++\n }\n\n /**\n * Add path-based middleware\n *\n * Supports wildcard patterns like '/api/*'\n *\n * @param pattern - Path pattern\n * @param middleware - Middleware functions\n */\n usePattern(pattern: string, ...middleware: Middleware[]): void {\n // Special case: '*' pattern should be treated as global middleware\n if (pattern === '*') {\n this.globalMiddleware.push(...middleware)\n } else {\n const existing = this.pathMiddleware.get(pattern) ?? []\n this.pathMiddleware.set(pattern, [...existing, ...middleware])\n }\n this._version++\n }\n\n /**\n * Match a request to a route\n *\n * Returns the handler, params, and all applicable middleware.\n *\n * @param method - HTTP method\n * @param path - Request path\n * @returns Route match or null if not found\n */\n match(method: string, path: string): RouteMatch {\n const normalizedMethod = method.toLowerCase() as HttpMethod\n\n // Try static route first (O(1))\n const staticKey = `${normalizedMethod}:${path}`\n const staticRoute = this.staticRoutes.get(staticKey)\n\n if (staticRoute) {\n return {\n handler: staticRoute.handler,\n params: {},\n middleware: this.collectMiddleware(path, staticRoute.middleware),\n routePattern: path,\n }\n }\n\n // Try dynamic route (Radix Tree)\n const match = this.dynamicRouter.match(normalizedMethod, path)\n\n if (match && match.handlers.length > 0) {\n const handler = match.handlers[0] as unknown as Handler\n const wrappedHandler = match.handlers[0]\n\n // 從 Map 中取得路由模式\n const routePattern = this.dynamicRoutePatterns.get(wrappedHandler)\n const routeKey = routePattern ? `${normalizedMethod}:${routePattern}` : null\n const routeMiddleware = routeKey ? (this.pathMiddleware.get(routeKey) ?? []) : []\n\n return {\n handler,\n params: match.params,\n middleware: this.collectMiddleware(path, routeMiddleware),\n routePattern,\n }\n }\n\n // No match\n return {\n handler: null,\n params: {},\n middleware: [],\n }\n }\n\n /**\n * Public wrapper for collectMiddleware (used by Gravito for optimization)\n */\n collectMiddlewarePublic(path: string, routeMiddleware: Middleware[]): Middleware[] {\n return this.collectMiddleware(path, routeMiddleware)\n }\n\n /**\n * Collect all applicable middleware for a path\n *\n * Order: global -> pattern-based -> route-specific\n *\n * @param path - Request path\n * @param routeMiddleware - Route-specific middleware\n * @returns Combined middleware array\n */\n private collectMiddleware(path: string, routeMiddleware: Middleware[]): Middleware[] {\n // Fast path: no middleware at all\n if (\n this.globalMiddleware.length === 0 &&\n this.pathMiddleware.size === 0 &&\n routeMiddleware.length === 0\n ) {\n return []\n }\n\n // Optimized cache key: use path directly for O(1) lookups\n // Cache format: path -> { middleware array, version }\n const cacheKey = path\n const cached = this.middlewareCache.get(cacheKey)\n\n // Cache hit: return immediately (same path gets same middleware)\n if (cached !== undefined && cached.version === this._version) {\n return cached.data\n }\n\n const middleware: Middleware[] = []\n\n // 1. Global middleware (most common case)\n if (this.globalMiddleware.length > 0) {\n middleware.push(...this.globalMiddleware)\n }\n\n // 2. Pattern-based middleware (check only if pathMiddleware exists)\n if (this.pathMiddleware.size > 0) {\n for (const [pattern, mw] of this.pathMiddleware) {\n // Skip route-specific entries (they have method prefix)\n if (pattern.includes(':')) {\n continue\n }\n\n if (this.matchPattern(pattern, path)) {\n middleware.push(...mw)\n }\n }\n }\n\n // 3. Route-specific middleware\n if (routeMiddleware.length > 0) {\n middleware.push(...routeMiddleware)\n }\n\n // LRU cache: only cache if under max size\n if (this.middlewareCache.size < this.cacheMaxSize) {\n this.middlewareCache.set(cacheKey, { data: middleware, version: this._version })\n } else if (this.middlewareCache.has(cacheKey)) {\n // Update existing cache entry\n this.middlewareCache.set(cacheKey, { data: middleware, version: this._version })\n }\n\n return middleware\n }\n\n /**\n * Get all static routes optimized for Bun's native router.\n *\n * Unlike basic offloading, this version supports routes with middleware\n * by pre-compiling the middleware chain into a single native handler.\n *\n * @param onMatch - Factory to wrap handler and middleware into a Bun-compatible function\n * @returns Record of path -> Handler (Bun-compatible)\n */\n getNativeRoutes(\n onMatch: (\n handler: Handler,\n middleware: Middleware[],\n path: string\n ) => (req: Request) => Response | Promise<Response>\n ): Record<string, any> {\n const routes: Record<string, any> = {}\n\n for (const [key, metadata] of this.staticRoutes) {\n const [method, path] = key.split(':')\n if (method !== 'get') {\n continue // Bun's native routes primarily focus on GET for Response objects\n }\n\n // Collect all applicable middleware (Global + Pattern + Route-specific)\n const allMiddleware = this.collectMiddleware(path!, metadata.middleware)\n\n // Map to Bun's native router format\n // The middleware chain is compiled and wrapped by the caller (Gravito)\n routes[path!] = onMatch(metadata.handler, allMiddleware, path!)\n }\n\n return routes\n }\n\n /**\n * Check if a path is static (no parameters or wildcards)\n */\n private isStaticPath(path: string): boolean {\n return !path.includes(':') && !path.includes('*')\n }\n\n /**\n * Match a pattern against a path\n *\n * Supports:\n * - Exact match: '/api/users'\n * - Wildcard suffix: '/api/*'\n *\n * @param pattern - Pattern to match\n * @param path - Path to test\n * @returns True if pattern matches\n */\n private matchPattern(pattern: string, path: string): boolean {\n if (pattern === '*') {\n return true\n }\n if (pattern === path) {\n return true\n }\n\n if (pattern.endsWith('/*')) {\n const prefix = pattern.slice(0, -2)\n return path.startsWith(prefix)\n }\n\n return false\n }\n\n /**\n * Get all registered routes (for debugging)\n */\n getRoutes(): Array<{ method: string; path: string; type: 'static' | 'dynamic' }> {\n const routes: Array<{ method: string; path: string; type: 'static' | 'dynamic' }> = []\n\n // Static routes\n for (const key of this.staticRoutes.keys()) {\n const [method, path] = key.split(':')\n routes.push({ method: method!, path: path!, type: 'static' })\n }\n\n // Dynamic routes (harder to enumerate from Radix Tree)\n // For now, we'll skip this - it's mainly for debugging anyway\n\n return routes\n }\n}\n",
|
|
11
|
+
"/**\n * @fileoverview Transpiler 工具庫 - AST 層級代碼分析\n *\n * 使用 Bun.Transpiler API 進行精確的 handler 函式分析,\n * 相比傳統字串匹配,精確度從 ~85% 提升至 ~99%。\n *\n * 核心策略:\n * 1. 使用 transformSync() 標準化代碼格式(統一縮排、引號等)\n * 2. 對轉換後的代碼使用精確的正規表達式匹配 member expression\n * 3. 區分 API 呼叫(.req.header())與變數名稱(const header = ...)\n * 4. 支援解構賦值模式(const { header } = ctx.req)\n * 5. 快取 Transpiler 實例(性能提升 5.9x)+ LRU 快取結果(額外 128x)\n *\n * @module @gravito/core/transpiler-utils\n * @since 3.1.0\n */\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Transpiler 快取類別\n// ─────────────────────────────────────────────────────────────────────────────\n\n/**\n * Transpiler 分析結果的快取鍵對應值\n * 儲存轉換後的代碼字串\n */\ninterface TranspilerCacheEntry {\n /** 轉換後的標準化代碼 */\n transformed: string\n /** 快取建立時間(ms,用於 TTL 過期) */\n createdAt: number\n}\n\n/**\n * TranspilerCache - 管理 Bun.Transpiler 實例與結果快取\n *\n * 避免重複建立 Transpiler(每次建立約需 40µs),\n * 並快取 transformSync 結果(重用快取比每次 transform 快 128x)。\n *\n * @example\n * ```typescript\n * const cache = TranspilerCache.getInstance()\n * const transformed = cache.transform(handlerSource)\n * ```\n */\nexport class TranspilerCache {\n private static instance: TranspilerCache | null = null\n\n /** 共享的 Bun.Transpiler 實例(避免重複建立) */\n private readonly transpiler: Bun.Transpiler\n\n /** LRU 快取:原始代碼 → 轉換結果 */\n private readonly cache: Map<string, TranspilerCacheEntry>\n\n /** 快取大小上限 */\n private readonly maxSize: number\n\n /** 快取 TTL(毫秒),預設 5 分鐘 */\n private readonly ttlMs: number\n\n private constructor(maxSize = 512, ttlMs = 5 * 60 * 1000) {\n // 使用 'ts' loader 以支援 TypeScript 語法\n this.transpiler = new Bun.Transpiler({ loader: 'ts' })\n this.cache = new Map()\n this.maxSize = maxSize\n this.ttlMs = ttlMs\n }\n\n /**\n * 取得單例實例\n * 確保全程只建立一個 Transpiler 實例\n */\n static getInstance(): TranspilerCache {\n TranspilerCache.instance ??= new TranspilerCache()\n return TranspilerCache.instance\n }\n\n /**\n * 重置單例(主要用於測試)\n */\n static resetInstance(): void {\n TranspilerCache.instance = null\n }\n\n /**\n * 轉換代碼並快取結果\n *\n * 先嘗試從快取取得,若未命中則呼叫 transformSync 並儲存結果。\n * 快取已滿時淘汰最舊的條目(近似 LRU)。\n *\n * 處理兩個 Bun.Transpiler 邊緣案例:\n * 1. 箭頭函式表達式:`async (ctx) => ...` → transformSync 返回空字串\n * 解法:包裝成 `const __fn = <source>` 後再轉換\n * 2. 匿名函式表達式:`function(ctx) {...}` → transformSync 拋出 Parse error\n * 解法:同樣包裝後轉換\n *\n * @param source - 原始 handler 函式字串\n * @returns 轉換後的標準化代碼,若完全失敗則回傳 null\n */\n transform(source: string): string | null {\n // 1. 快取命中\n const cached = this.cache.get(source)\n if (cached !== undefined) {\n // 檢查 TTL\n if (Date.now() - cached.createdAt < this.ttlMs) {\n return cached.transformed\n }\n // 過期,移除\n this.cache.delete(source)\n }\n\n // 2. 執行 transform(含邊緣案例處理)\n const transformed = this.doTransform(source)\n if (transformed === null) {\n // 完全失敗,不快取,讓呼叫方 fallback\n return null\n }\n\n // 3. 淘汰舊條目(若已達上限)\n if (this.cache.size >= this.maxSize) {\n // 刪除最舊的條目(Map 迭代順序即插入順序)\n const firstKey = this.cache.keys().next().value\n if (firstKey !== undefined) {\n this.cache.delete(firstKey)\n }\n }\n\n // 4. 存入快取\n this.cache.set(source, { transformed, createdAt: Date.now() })\n return transformed\n }\n\n /**\n * 實際執行 transformSync,處理箭頭函式和匿名函式的邊緣案例\n *\n * @param source - 原始代碼字串\n * @returns 轉換後的代碼,或失敗時回傳 null\n */\n private doTransform(source: string): string | null {\n // 嘗試直接 transform\n try {\n const out = this.transpiler.transformSync(source)\n // 若輸出非空,直接使用(named function / async function 正常情況)\n if (out.trim().length > 0) {\n return out\n }\n // 空字串輸出 = 箭頭函式表達式(如 `async (ctx) => ctx.json({})`)\n // 嘗試包裝後再 transform\n return this.transformWrapped(source)\n } catch {\n // Parse error = 匿名函式表達式(如 `function(ctx) {...}`)\n // 嘗試包裝後再 transform\n return this.transformWrapped(source)\n }\n }\n\n /**\n * 將代碼包裝成賦值語句後再 transform\n *\n * 用於處理無法直接 transform 的函式表達式。\n * 包裝格式:`const __fn = <source>`\n *\n * @param source - 原始函式字串\n * @returns 包裝後的轉換結果,或失敗時回傳 null\n */\n private transformWrapped(source: string): string | null {\n try {\n const wrapped = `const __fn = ${source}`\n const out = this.transpiler.transformSync(wrapped)\n return out.trim().length > 0 ? out : null\n } catch {\n return null\n }\n }\n\n /**\n * 取得目前快取大小\n */\n get size(): number {\n return this.cache.size\n }\n\n /**\n * 清除所有快取條目\n */\n clear(): void {\n this.cache.clear()\n }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// 精確模式偵測正規表達式\n// ─────────────────────────────────────────────────────────────────────────────\n\n/**\n * 所有偵測模式集中定義,方便維護與測試\n *\n * 設計原則:\n * - Member access: 匹配 `.req.<方法名>(` 形式(函式呼叫)\n * - Destructure: 匹配 `{ <屬性名> } = <任意>.<req前綴>` 形式(解構賦值)\n * - 排除字串賦值:轉換後的字串賦值格式為 `= \"...\"` 或 `= '...'`,\n * 正規表達式只匹配 `.req.` 開頭的 member access,因此自然排除\n */\nconst PATTERNS = {\n /**\n * 偵測 headers/header 的成員存取\n *\n * 匹配:\n * - `.req.header(` 或 `.req.headers(`\n * - `{ header } = *.req` 或 `{ headers } = *.req`\n */\n HEADERS_CALL: /\\.req\\.headers?\\s*\\(/,\n HEADERS_DESTR: /\\{[^}]*\\bheaders?\\b[^}]*\\}\\s*=\\s*\\w+\\.req/,\n\n /**\n * 偵測 query/queries 的成員存取\n *\n * 匹配:\n * - `.req.query(` 或 `.req.queries(`\n * - `{ query } = *.req` 或 `{ queries } = *.req`\n */\n QUERY_CALL: /\\.req\\.quer(?:y|ies)\\s*\\(/,\n QUERY_DESTR: /\\{[^}]*\\bquer(?:y|ies)\\b[^}]*\\}\\s*=\\s*\\w+\\.req/,\n\n /**\n * 偵測 body 相關的成員存取\n *\n * 匹配:\n * - `.req.json(` `.req.text(` `.req.formData(` `.req.blob(` `.req.arrayBuffer(`\n * - `{ body } = *.req`\n * - `.req.body`(直接屬性存取)\n */\n BODY_CALL: /\\.req\\.(?:json|text|formData|blob|arrayBuffer)\\s*\\(/,\n BODY_DESTR: /\\{[^}]*\\bbody\\b[^}]*\\}\\s*=\\s*\\w+\\.req/,\n BODY_PROP: /\\.req\\.body\\b/,\n\n /**\n * 偵測 param/params 的成員存取\n *\n * 匹配:\n * - `.req.param(` 或 `.req.params(`\n * - `{ param } = *.req` 或 `{ params } = *.req`\n */\n PARAMS_CALL: /\\.req\\.params?\\s*\\(/,\n PARAMS_DESTR: /\\{[^}]*\\bparams?\\b[^}]*\\}\\s*=\\s*\\w+\\.req/,\n\n /**\n * 偵測非同步函式(用於原始碼)\n *\n * 直接匹配原始碼中的 async 關鍵字。\n * 注意:isAsync 不用 transformSync 結果,因為:\n * 1. 箭頭函式 `async (ctx) => ...` 的 transformSync 返回空字串\n * 2. `async` 關鍵字本身不存在假陽性問題(沒有 API 叫做 async)\n *\n * 匹配:\n * - `async function` - async 具名/匿名函式\n * - `async (` - async 箭頭函式(括號形式)\n * - `async ctx` - async 箭頭函式(無括號單參數)\n */\n IS_ASYNC: /\\basync\\b/,\n} as const\n\n// ─────────────────────────────────────────────────────────────────────────────\n// 核心分析函式\n// ─────────────────────────────────────────────────────────────────────────────\n\n/**\n * Transpiler 分析的返回結果\n * 與 HandlerAnalysis 介面對應\n */\nexport interface TranspilerAnalysisResult {\n usesHeaders: boolean\n usesQuery: boolean\n usesBody: boolean\n usesParams: boolean\n isAsync: boolean\n}\n\n/**\n * 使用 Bun.Transpiler 進行精確的 handler 靜態分析\n *\n * 相比字串匹配,此函式能正確區分:\n * - API 呼叫(`ctx.req.header(name)`)vs 變數名稱(`const header = '...'`)\n * - 解構賦值(`const { header } = ctx.req`)\n * - Minified 代碼(transformSync 先標準化)\n * - 箭頭函式與匿名函式(包裝策略處理 Bun.Transpiler 邊緣案例)\n *\n * 若 Transpiler 轉換失敗,會自動 fallback 到字串匹配模式。\n *\n * ## isAsync 特殊處理\n *\n * `isAsync` 直接從原始碼偵測 `async` 關鍵字,而不是從 transformSync 結果:\n * - 箭頭函式 `async (ctx) => ...` 的 transformSync 返回空字串\n * - `async` 關鍵字本身不存在假陽性問題\n *\n * @param source - handler 函式的字串表示(通常來自 handler.toString())\n * @returns 分析結果,或在 fallback 模式下的近似結果\n *\n * @example\n * ```typescript\n * const handler = async (ctx) => {\n * const name = ctx.req.query('name')\n * return ctx.json({ name })\n * }\n * const result = analyzeHandlerWithTranspiler(handler.toString())\n * // result.usesQuery === true\n * // result.usesHeaders === false(即使有 'header' 字串,也不會誤判)\n * // result.isAsync === true(即使是箭頭函式也能正確偵測)\n * ```\n */\nexport function analyzeHandlerWithTranspiler(source: string): TranspilerAnalysisResult {\n const cache = TranspilerCache.getInstance()\n const transformed = cache.transform(source)\n\n // isAsync 直接從原始碼偵測(不依賴 transformSync 結果)\n // 因為箭頭函式的 transformSync 返回空字串,且 async 關鍵字無假陽性問題\n const isAsync = PATTERNS.IS_ASYNC.test(source)\n\n // 若 Transpiler 成功,使用精確的 AST 級別模式分析其他屬性\n if (transformed !== null) {\n return { ...analyzeTransformedCode(transformed), isAsync }\n }\n\n // Fallback:Transpiler 失敗時使用字串匹配(保守估計)\n return { ...fallbackStringAnalysis(source), isAsync }\n}\n\n/**\n * 對 Transpiler 轉換後的代碼進行精確模式分析(不含 isAsync)\n *\n * @param code - transformSync() 返回的標準化代碼\n * @returns 分析結果(isAsync 由呼叫方根據原始碼設定)\n */\nfunction analyzeTransformedCode(\n code: string\n): Omit<TranspilerAnalysisResult, 'isAsync'> & { isAsync: boolean } {\n return {\n usesHeaders: PATTERNS.HEADERS_CALL.test(code) || PATTERNS.HEADERS_DESTR.test(code),\n usesQuery: PATTERNS.QUERY_CALL.test(code) || PATTERNS.QUERY_DESTR.test(code),\n usesBody:\n PATTERNS.BODY_CALL.test(code) ||\n PATTERNS.BODY_DESTR.test(code) ||\n PATTERNS.BODY_PROP.test(code),\n usesParams: PATTERNS.PARAMS_CALL.test(code) || PATTERNS.PARAMS_DESTR.test(code),\n // 這個值會被呼叫方覆蓋,這裡保留是為了型別完整性\n isAsync: PATTERNS.IS_ASYNC.test(code),\n }\n}\n\n/**\n * 字串匹配 Fallback 分析\n *\n * 當 Bun.Transpiler 不可用或轉換失敗時使用,\n * 比原始實現略為保守(寬鬆匹配以減少假陰性)。\n *\n * @param source - 原始 handler 函式字串\n * @returns 分析結果(精確度較低,~85%)\n */\nfunction fallbackStringAnalysis(source: string): TranspilerAnalysisResult {\n return {\n usesHeaders:\n source.includes('.header(') ||\n source.includes('.header)') ||\n source.includes('.headers(') ||\n source.includes('.headers)'),\n usesQuery:\n source.includes('.query(') ||\n source.includes('.query)') ||\n source.includes('.queries(') ||\n source.includes('.queries)'),\n usesBody:\n source.includes('.json()') ||\n source.includes('.text()') ||\n source.includes('.formData()') ||\n source.includes('.body'),\n usesParams:\n source.includes('.param(') ||\n source.includes('.param)') ||\n source.includes('.params(') ||\n source.includes('.params)'),\n isAsync: source.includes('async') || source.includes('await'),\n }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// 輔助工具函式\n// ─────────────────────────────────────────────────────────────────────────────\n\n/**\n * 測試 handler 源代碼是否存取特定的 req 成員屬性\n *\n * 工具函式,方便在 Gravito.ts 等地方進行特定屬性的快速檢測。\n *\n * @param source - handler 函式的字串表示\n * @param property - 要測試的屬性名稱(如 'header', 'query', 'body')\n * @returns 若該屬性被存取則回傳 true\n *\n * @example\n * ```typescript\n * const src = handler.toString()\n * if (hasReqMemberAccess(src, 'header')) {\n * // handler 存取了 header\n * }\n * ```\n */\nexport function hasReqMemberAccess(source: string, property: string): boolean {\n const cache = TranspilerCache.getInstance()\n const transformed = cache.transform(source)\n const code = transformed ?? source\n\n // 成員函式呼叫:.req.<property>(\n const callPattern = new RegExp(`\\\\.req\\\\.${property}\\\\s*\\\\(`)\n // 解構賦值:{ <property> } = *.req\n const destrPattern = new RegExp(`\\\\{[^}]*\\\\b${property}\\\\b[^}]*\\\\}\\\\s*=\\\\s*\\\\w+\\\\.req`)\n // 直接屬性存取:.req.<property>(用於 .body 等)\n const propPattern = new RegExp(`\\\\.req\\\\.${property}\\\\b`)\n\n return callPattern.test(code) || destrPattern.test(code) || propPattern.test(code)\n}\n\n/**\n * 判斷 handler 是否為非同步函式\n *\n * 直接從原始碼偵測 `async` 關鍵字,不依賴 transformSync 結果,\n * 因為箭頭函式的 transformSync 返回空字串。\n * `async` 關鍵字本身不存在假陽性問題。\n *\n * @param source - handler 函式的字串表示\n * @returns 若為 async 函式則回傳 true\n */\nexport function isAsyncHandler(source: string): boolean {\n return PATTERNS.IS_ASYNC.test(source)\n}\n\n/**\n * 預熱 Transpiler 快取\n *\n * 在應用啟動時呼叫,觸發 Transpiler 實例建立,\n * 避免第一個請求時的冷啟動延遲。\n */\nexport function warmupTranspilerCache(): void {\n const cache = TranspilerCache.getInstance()\n // 轉換一個簡單的 handler 觸發初始化\n cache.transform('function warmup(ctx) { return ctx.json({}) }')\n}\n",
|
|
12
|
+
"/**\n * @fileoverview Handler 靜態分析器(Elysia-inspired,升級版)\n *\n * 分析 handler 函式,偵測其存取了哪些請求屬性,\n * 以選擇最優化的 Context 類型(minimal/fast/full)。\n *\n * ## 版本歷史\n *\n * ### v2(目前版本)- 使用 Bun.Transpiler\n * - 精確度:~99%(AST 層級分析)\n * - 正確處理假陽性:`const header = '...'` 不再誤判\n * - 支援解構賦值:`const { header } = ctx.req`\n * - 支援 Minified 代碼(transformSync 先標準化)\n * - Fallback:若 Transpiler 失敗,退回字串匹配\n *\n * ### v1(原版本)- 字串匹配\n * - 精確度:~85%\n * - 假陽性:變數名稱包含目標字串會誤判\n * - 假陰性:解構賦值無法偵測\n * - Minified 代碼可能失效\n *\n * @module @gravito/core/engine/analyzer\n * @since 3.0.0\n */\n\nimport { analyzeHandlerWithTranspiler } from '../transpiler-utils'\n\n// ─────────────────────────────────────────────────────────────────────────────\n// 公開介面(向後相容,未改變)\n// ─────────────────────────────────────────────────────────────────────────────\n\n/**\n * Handler 靜態分析結果\n *\n * 記錄 handler 函式使用了哪些請求屬性,\n * 用於選擇最優化的 Context 類型。\n *\n * @public\n * @since 3.0.0\n */\nexport interface HandlerAnalysis {\n /** 是否存取了 request headers */\n usesHeaders: boolean\n /** 是否存取了 query string 參數 */\n usesQuery: boolean\n /** 是否存取了 request body */\n usesBody: boolean\n /** 是否存取了 route 路徑參數 */\n usesParams: boolean\n /** 是否為非同步函式(含 async/await) */\n isAsync: boolean\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// 主要分析函式(升級版)\n// ─────────────────────────────────────────────────────────────────────────────\n\n/**\n * 分析 handler 函式,偵測其使用了哪些請求屬性\n *\n * 使用 Bun.Transpiler 進行 AST 層級的精確分析(精確度 ~99%)。\n * 若 Transpiler 不可用,自動 fallback 到字串匹配(精確度 ~85%)。\n *\n * ## 精確度提升說明\n *\n * **假陽性修復**(原本誤判,現在正確):\n * ```typescript\n * // 這個 handler 原本會誤判 usesHeaders = true\n * // 因為字串 'header' 出現在變數名稱中\n * function handler(ctx) {\n * const header = 'Content-Type' // ← 變數名稱,不是 API 呼叫\n * return ctx.json({ header })\n * }\n * // 現在:usesHeaders = false ✅\n * ```\n *\n * **假陰性修復**(原本漏偵測,現在正確):\n * ```typescript\n * // 這個 handler 原本會漏偵測解構賦值\n * function handler(ctx) {\n * const { header, query } = ctx.req // ← 解構賦值\n * return ctx.json({ header })\n * }\n * // 現在:usesHeaders = true, usesQuery = true ✅\n * ```\n *\n * @param handler - 要分析的 handler 函式\n * @returns HandlerAnalysis 分析結果\n *\n * @example\n * ```typescript\n * const handler = async (ctx) => {\n * const name = ctx.req.query('name')\n * return ctx.json({ name })\n * }\n *\n * const analysis = analyzeHandler(handler)\n * // analysis.usesQuery === true\n * // analysis.usesHeaders === false\n * // analysis.isAsync === true\n *\n * const type = getOptimalContextType(analysis)\n * // type === 'fast'\n * ```\n */\n// biome-ignore lint/complexity/noBannedTypes: 保持 public API 向後相容,handler 可以是任意 callable\nexport function analyzeHandler(handler: Function): HandlerAnalysis {\n const source = handler.toString()\n // 委派給 Transpiler 版本(含快取與 fallback)\n return analyzeHandlerWithTranspiler(source)\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Context 類型選擇(不變,保持向後相容)\n// ─────────────────────────────────────────────────────────────────────────────\n\n/**\n * 根據分析結果決定最優化的 Context 類型\n *\n * Context 類型由輕到重:\n * - `minimal`:僅支援路徑參數與靜態回應,零 overhead\n * - `fast`:支援 headers 與 query,使用物件池\n * - `full`:支援完整功能含 body 解析(async)\n *\n * 選擇邏輯(優先順序):\n * 1. 若存取 headers → `fast`(header 設置需要完整支援)\n * 2. 若不存取任何屬性 → `minimal`\n * 3. 若僅存取 params → `minimal`(params 在 minimal 中也可用)\n * 4. 若存取 body → `full`(async body 解析需要完整 context)\n * 5. 其他 → `fast`\n *\n * @param analysis - HandlerAnalysis 分析結果\n * @returns 最優化的 context 類型\n */\nexport function getOptimalContextType(analysis: HandlerAnalysis): 'minimal' | 'fast' | 'full' {\n // 存取 headers → 需要 fast context(header 設定功能)\n if (analysis.usesHeaders) {\n return 'fast'\n }\n\n // 不存取任何請求屬性 → 可使用 minimal context\n if (!analysis.usesQuery && !analysis.usesBody && !analysis.usesParams) {\n return 'minimal'\n }\n\n // 僅存取 params → minimal context 已有路徑參數支援\n if (!analysis.usesQuery && !analysis.usesBody && analysis.usesParams) {\n return 'minimal'\n }\n\n // 存取 body → 需要 full context(async body 解析)\n if (analysis.usesBody) {\n return 'full'\n }\n\n return 'fast'\n}\n",
|
|
13
|
+
"/**\n * @fileoverview Engine Constants & Cached Buffers\n *\n * Pre-allocated resources to minimize runtime allocation overhead.\n * Specifically targets Bun's zero-copy capabilities.\n */\n\nconst encoder = new TextEncoder()\n\n// Pre-encoded JSON responses for common scenarios\nexport const CACHED_RESPONSES = {\n NOT_FOUND: encoder.encode('{\"error\":\"Not Found\"}'),\n INTERNAL_ERROR: encoder.encode('{\"error\":\"Internal Server Error\"}'),\n OK: encoder.encode('{\"ok\":true}'),\n EMPTY: new Uint8Array(0),\n} as const\n\n// Content-Type constants (avoid string allocation)\nexport const HEADERS = {\n JSON: { 'Content-Type': 'application/json; charset=utf-8' },\n TEXT: { 'Content-Type': 'text/plain; charset=utf-8' },\n HTML: { 'Content-Type': 'text/html; charset=utf-8' },\n} as const\n",
|
|
14
|
+
"/**\n * RequestScopeMetrics - Observability for RequestScope lifecycle\n *\n * Tracks cleanup execution time, scope size, and service counts\n * for performance monitoring and diagnostics.\n *\n * @example\n * ```typescript\n * const metrics = new RequestScopeMetrics()\n * metrics.recordCleanupStart()\n * await scope.cleanup()\n * metrics.recordCleanupEnd()\n *\n * console.log(metrics.toJSON())\n * // { cleanupDuration: 2.5, scopeSize: 3, servicesCleaned: 3 }\n * ```\n */\nexport class RequestScopeMetrics {\n private cleanupStartTime: number | null = null\n private cleanupDuration: number | null = null\n private scopeSize = 0\n private servicesCleaned = 0\n private errorsOccurred = 0\n\n /**\n * Record start of cleanup operation\n */\n recordCleanupStart(): void {\n this.cleanupStartTime = performance.now()\n }\n\n /**\n * Record end of cleanup operation\n *\n * @param scopeSize - Number of services in the scope\n * @param servicesCleaned - Number of services that had cleanup called\n * @param errorsOccurred - Number of cleanup errors\n */\n recordCleanupEnd(scopeSize: number, servicesCleaned: number, errorsOccurred = 0): void {\n if (this.cleanupStartTime !== null) {\n this.cleanupDuration = performance.now() - this.cleanupStartTime\n this.cleanupStartTime = null\n }\n\n this.scopeSize = scopeSize\n this.servicesCleaned = servicesCleaned\n this.errorsOccurred = errorsOccurred\n }\n\n /**\n * Get cleanup duration in milliseconds\n *\n * @returns Duration in ms, or null if cleanup not completed\n */\n getCleanupDuration(): number | null {\n return this.cleanupDuration\n }\n\n /**\n * Check if cleanup took longer than threshold (default 2ms)\n * Useful for detecting slow cleanups\n *\n * @param thresholdMs - Threshold in milliseconds\n * @returns True if cleanup exceeded threshold\n */\n isSlowCleanup(thresholdMs = 2): boolean {\n if (this.cleanupDuration === null) {\n return false\n }\n return this.cleanupDuration > thresholdMs\n }\n\n /**\n * Export metrics as JSON for logging/monitoring\n */\n toJSON() {\n return {\n cleanupDuration: this.cleanupDuration,\n scopeSize: this.scopeSize,\n servicesCleaned: this.servicesCleaned,\n errorsOccurred: this.errorsOccurred,\n hasErrors: this.errorsOccurred > 0,\n isSlowCleanup: this.isSlowCleanup(),\n }\n }\n\n /**\n * Export metrics as compact string for logging\n */\n toString(): string {\n const duration = this.cleanupDuration ?? 'pending'\n return (\n `cleanup: ${duration}ms, ` +\n `scope: ${this.scopeSize}, ` +\n `cleaned: ${this.servicesCleaned}, ` +\n `errors: ${this.errorsOccurred}`\n )\n }\n}\n\n/**\n * RequestScopeObserver - Hook for monitoring RequestScope lifecycle\n *\n * Implement this interface to receive callbacks during scope operations\n */\nexport interface RequestScopeObserver {\n /**\n * Called when a service is resolved in the scope\n */\n onServiceResolved?(key: string | symbol, isFromCache: boolean): void\n\n /**\n * Called when cleanup starts\n */\n onCleanupStart?(): void\n\n /**\n * Called when cleanup completes\n */\n onCleanupEnd?(metrics: RequestScopeMetrics): void\n\n /**\n * Called when cleanup encounters an error\n */\n onCleanupError?(error: Error): void\n}\n\n/**\n * RequestScopeMetricsCollector - Aggregates metrics across multiple scopes\n *\n * Used for application-level monitoring and performance tracking\n *\n * @example\n * ```typescript\n * const collector = new RequestScopeMetricsCollector()\n *\n * // Record metrics from multiple requests\n * collector.record(metrics1)\n * collector.record(metrics2)\n * collector.record(metrics3)\n *\n * // Get aggregated stats\n * const stats = collector.getStats()\n * console.log(stats.averageCleanupTime) // 3.5ms\n * ```\n */\nexport class RequestScopeMetricsCollector {\n private metrics: RequestScopeMetrics[] = []\n\n /**\n * Record metrics from a request scope\n */\n record(metrics: RequestScopeMetrics): void {\n this.metrics.push(metrics)\n }\n\n /**\n * Get aggregated statistics\n */\n getStats() {\n if (this.metrics.length === 0) {\n return {\n count: 0,\n averageCleanupTime: null,\n maxCleanupTime: null,\n minCleanupTime: null,\n totalErrorCount: 0,\n errorRate: 0,\n }\n }\n\n const durations = this.metrics\n .map((m) => m.getCleanupDuration())\n .filter((d): d is number => d !== null)\n\n const errorCounts = this.metrics.map((m) => m.toJSON().errorsOccurred)\n const totalErrors = errorCounts.reduce((a, b) => a + b, 0)\n\n return {\n count: this.metrics.length,\n averageCleanupTime:\n durations.length > 0 ? durations.reduce((a, b) => a + b) / durations.length : null,\n maxCleanupTime: durations.length > 0 ? Math.max(...durations) : null,\n minCleanupTime: durations.length > 0 ? Math.min(...durations) : null,\n totalErrorCount: totalErrors,\n errorRate: totalErrors / this.metrics.length,\n }\n }\n\n /**\n * Clear collected metrics\n */\n clear(): void {\n this.metrics = []\n }\n\n /**\n * Get number of recorded metrics\n */\n size(): number {\n return this.metrics.length\n }\n\n /**\n * Export metrics as JSON array\n */\n toJSON() {\n return this.metrics.map((m) => m.toJSON())\n }\n}\n",
|
|
15
|
+
"import type { ServiceKey } from '../Container'\nimport { RequestScopeMetrics, type RequestScopeObserver } from './RequestScopeMetrics'\n\n/**\n * Manages request-scoped service instances within a single HTTP request.\n *\n * Each request gets its own RequestScopeManager instance with isolated state.\n * Services are cached within the request and automatically cleaned up when\n * the request ends.\n *\n * @example\n * ```typescript\n * const scope = new RequestScopeManager()\n * const cache = scope.resolve('productCache', () => new ProductCache())\n * // ... use cache ...\n * await scope.cleanup() // Called automatically by Gravito engine\n * ```\n */\nexport class RequestScopeManager {\n private scoped = new Map<string, unknown>()\n private metadata = new Map<string, Record<string, unknown>>()\n private metrics = new RequestScopeMetrics()\n private observer: RequestScopeObserver | null = null\n\n constructor(observer?: RequestScopeObserver) {\n this.observer = observer || null\n }\n\n /**\n * Set observer for monitoring scope lifecycle\n */\n setObserver(observer: RequestScopeObserver): void {\n this.observer = observer\n }\n\n /**\n * Get metrics for this scope\n */\n getMetrics(): RequestScopeMetrics {\n return this.metrics\n }\n\n /**\n * Resolve or retrieve a request-scoped service instance.\n *\n * If the service already exists in this scope, returns the cached instance.\n * Otherwise, calls the factory function to create a new instance and caches it.\n *\n * Automatically detects and records services with cleanup methods.\n *\n * @template T - The type of the service.\n * @param key - The service key (for caching).\n * @param factory - Factory function to create the instance if not cached.\n * @returns The cached or newly created instance.\n */\n resolve<T>(key: ServiceKey, factory: () => T): T {\n const keyStr = String(key)\n const isFromCache = this.scoped.has(keyStr)\n\n if (!isFromCache) {\n const instance = factory()\n this.scoped.set(keyStr, instance)\n\n // Record metadata for cleanup detection\n if (instance && typeof instance === 'object' && 'cleanup' in instance) {\n this.metadata.set(keyStr, { hasCleanup: true })\n }\n }\n\n // Notify observer\n this.observer?.onServiceResolved?.(key, isFromCache)\n\n return this.scoped.get(keyStr) as T\n }\n\n /**\n * Clean up all request-scoped instances.\n *\n * Calls the cleanup() method on each service that has one.\n * Silently ignores cleanup errors to prevent cascading failures.\n * Called automatically by the Gravito engine in the request finally block.\n *\n * @returns Promise that resolves when all cleanup is complete.\n */\n async cleanup(): Promise<void> {\n this.metrics.recordCleanupStart()\n this.observer?.onCleanupStart?.()\n\n const errors: unknown[] = []\n let servicesCleaned = 0\n\n for (const [, instance] of this.scoped) {\n if (instance && typeof instance === 'object' && 'cleanup' in instance) {\n const fn = (instance as any).cleanup\n if (typeof fn === 'function') {\n try {\n await fn.call(instance)\n servicesCleaned++\n } catch (error) {\n // Collect errors but continue cleanup\n errors.push(error)\n this.observer?.onCleanupError?.(\n error instanceof Error ? error : new Error(String(error))\n )\n }\n }\n }\n }\n\n const scopeSize = this.scoped.size\n this.scoped.clear()\n this.metadata.clear()\n\n // Record metrics\n this.metrics.recordCleanupEnd(scopeSize, servicesCleaned, errors.length)\n this.observer?.onCleanupEnd?.(this.metrics)\n\n // Log errors if any occurred\n if (errors.length > 0) {\n console.error('RequestScope cleanup errors:', errors)\n }\n }\n\n /**\n * Get the number of services in this scope (for monitoring).\n *\n * @returns The count of cached services.\n */\n size(): number {\n return this.scoped.size\n }\n}\n",
|
|
16
|
+
"/**\n * @fileoverview FastContext - Pooled Request Context\n *\n * Minimal, high-performance context implementation designed for object pooling.\n * Lazy parsing strategy: only parse what's actually accessed.\n *\n * @module @gravito/core/engine\n */\n\nimport { RequestScopeManager } from '../Container/RequestScopeManager'\nimport type { FastRequest, FastContext as IFastContext } from './types'\n\n// Bun runtime optimization: cache frequently used functions\nlet bunEscapeHTML: (html: string) => string = (html: string) => {\n // Default fallback for Node.js environments\n return html\n .replace(/&/g, '&')\n .replace(/</g, '<')\n .replace(/>/g, '>')\n .replace(/\"/g, '"')\n .replace(/'/g, ''')\n}\n\ntry {\n bunEscapeHTML = (require('bun') as any).escapeHTML\n} catch {\n // Already have default fallback\n}\n\n/**\n * Lazy-parsed request wrapper\n *\n * Delays parsing of query params, headers, and body until accessed.\n * This is a key optimization for requests that don't need all data.\n */\nclass FastRequestImpl implements FastRequest {\n private _request!: Request\n private _params!: Record<string, string>\n private _path!: string\n private _routePattern?: string\n private _url: URL | null = null\n private _query: URLSearchParams | null = null\n private _headers: Record<string, string> | null = null\n private _cachedJson: unknown = undefined\n private _jsonParsed = false\n private _cachedText: string | undefined = undefined\n private _textParsed = false\n private _cachedFormData: FormData | undefined = undefined\n private _formDataParsed = false\n private _cachedQueries: Record<string, string | string[]> | null = null\n private _cachedCookies: Record<string, string> | null = null\n // Back-reference for release check optimization\n private _ctx: FastContext\n\n constructor(ctx: FastContext) {\n this._ctx = ctx\n }\n\n /**\n * Initialize for new request\n */\n init(\n request: Request,\n params: Record<string, string> = {},\n path = '',\n routePattern?: string\n ): this {\n this._request = request\n this._params = params\n this._path = path\n this._routePattern = routePattern\n this._url = null\n this._query = null\n this._headers = null\n this._cachedJson = undefined\n this._jsonParsed = false\n this._cachedText = undefined\n this._textParsed = false\n this._cachedFormData = undefined\n this._formDataParsed = false\n this._cachedQueries = null\n this._cachedCookies = null\n return this\n }\n\n /**\n * Reset for pooling\n */\n reset(): void {\n // Release references to allow GC\n this._request = undefined as any\n this._params = undefined as any\n this._url = null\n this._query = null\n this._headers = null\n this._cachedJson = undefined\n this._jsonParsed = false\n this._cachedText = undefined\n this._textParsed = false\n this._cachedFormData = undefined\n this._formDataParsed = false\n this._cachedQueries = null\n this._cachedCookies = null\n }\n\n private checkReleased(): void {\n if (this._ctx._isReleased) {\n throw new Error(\n 'FastContext usage after release detected! (Object Pool Strict Lifecycle Guard)'\n )\n }\n }\n\n get url(): string {\n this.checkReleased()\n return this._request.url\n }\n\n get method(): string {\n this.checkReleased()\n return this._request.method\n }\n\n get path(): string {\n this.checkReleased()\n return this._path\n }\n\n get routePattern(): string | undefined {\n this.checkReleased()\n return this._routePattern\n }\n\n param(name: string): string | undefined {\n this.checkReleased()\n return this._params[name]\n }\n\n params(): Record<string, string> {\n this.checkReleased()\n return { ...this._params }\n }\n\n private getUrl(): URL {\n if (!this._url) {\n this._url = new URL(this._request.url)\n }\n return this._url\n }\n\n query(name: string): string | undefined {\n this.checkReleased()\n if (!this._query) {\n this._query = this.getUrl().searchParams\n }\n return this._query.get(name) ?? undefined\n }\n\n queries(): Record<string, string | string[]> {\n this.checkReleased()\n // Return cached queries object if available\n if (this._cachedQueries !== null) {\n return this._cachedQueries\n }\n\n if (!this._query) {\n this._query = this.getUrl().searchParams\n }\n\n const result: Record<string, string | string[]> = {}\n for (const [key, value] of this._query.entries()) {\n const existing = result[key]\n if (existing === undefined) {\n result[key] = value\n } else if (Array.isArray(existing)) {\n existing.push(value)\n } else {\n result[key] = [existing, value]\n }\n }\n this._cachedQueries = result\n return result\n }\n\n header(name: string): string | undefined {\n this.checkReleased()\n return this._request.headers.get(name) ?? undefined\n }\n\n headers(): Record<string, string> {\n this.checkReleased()\n if (!this._headers) {\n this._headers = {}\n for (const [key, value] of this._request.headers.entries()) {\n this._headers[key] = value\n }\n }\n return { ...this._headers }\n }\n\n get cookies(): Record<string, string> {\n this.checkReleased()\n // Return cached cookies if available\n if (this._cachedCookies !== null) {\n return this._cachedCookies\n }\n\n // 1. Try Bun Native CookieMap (Injected in Bun.serve routes API)\n const nativeCookies = (this._request as any).cookies\n if (nativeCookies) {\n this._cachedCookies = nativeCookies\n return nativeCookies\n }\n\n // 2. Fallback: Parse from Header manually (Lazy Parsed)\n const cookieHeader = this._request.headers.get('cookie')\n if (!cookieHeader) {\n this._cachedCookies = {}\n return {}\n }\n\n // Manual parse implementation (optimized for speed)\n // Handles URL-encoded values and flexible separators ('; ' or ';')\n const cookies: Record<string, string> = {}\n const pairs = cookieHeader.split(/;\\s*/)\n for (let i = 0; i < pairs.length; i++) {\n const pair = pairs[i]!\n const idx = pair.indexOf('=')\n if (idx > 0) {\n const name = pair.substring(0, idx)\n const value = pair.substring(idx + 1)\n // Decode value if URL-encoded, fallback to raw value on decode error\n try {\n cookies[name] = decodeURIComponent(value)\n } catch {\n cookies[name] = value\n }\n }\n }\n this._cachedCookies = cookies\n return cookies\n }\n\n async json<T = unknown>(): Promise<T> {\n this.checkReleased()\n if (!this._jsonParsed) {\n this._cachedJson = await this._request.json()\n this._jsonParsed = true\n }\n return this._cachedJson as T\n }\n\n async text(): Promise<string> {\n this.checkReleased()\n if (!this._textParsed) {\n this._cachedText = await this._request.text()\n this._textParsed = true\n }\n return this._cachedText!\n }\n\n async formData(): Promise<FormData> {\n this.checkReleased()\n if (!this._formDataParsed) {\n this._cachedFormData = await this._request.formData()\n this._formDataParsed = true\n }\n return this._cachedFormData!\n }\n\n get raw(): Request {\n this.checkReleased()\n return this._request\n }\n}\n\n/**\n * FastContext - Pooled request context\n *\n * Designed for minimal memory allocation and maximum reuse.\n * All response helpers create Response objects directly without intermediate wrappers.\n */\nexport class FastContext implements IFastContext {\n public readonly req: FastRequestImpl = new FastRequestImpl(this)\n // private _statusCode = 200\n private _headers = new Headers() // Reuse this object\n\n public _isReleased = false // Made public for internal check access\n private _requestScope: RequestScopeManager | null = null // Request-scoped services\n\n /**\n * Initialize context for a new request\n *\n * This is called when acquiring from the pool.\n */\n init(\n request: Request,\n params: Record<string, string> = {},\n path = '',\n routePattern?: string\n ): this {\n this._isReleased = false\n this.req.init(request, params, path, routePattern)\n // Optimization: Creating new Headers is faster than iterating to delete in Bun\n // But for strict object pooling, we might want to reconsider.\n // For now, new Headers() is safe and fast enough.\n this._headers = new Headers()\n this._requestScope = new RequestScopeManager() // Initialize request scope\n return this\n }\n\n /**\n * Reset context for pooling (Cleanup)\n *\n * This is called when releasing back to the pool.\n * Implements \"Deep-Reset Protocol\" and \"Release Guard\".\n */\n reset(): void {\n this._isReleased = true\n this.req.reset()\n // We don't clear _headers here because init() will create a new one.\n // If we wanted to reuse, we would clear it here.\n this._store.clear()\n this._requestScope = null // Release reference for GC\n }\n\n /**\n * Check if context is released\n */\n private checkReleased(): void {\n if (this._isReleased) {\n throw new Error(\n 'FastContext usage after release detected! (Object Pool Strict Lifecycle Guard)'\n )\n }\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // Response Helpers\n // ─────────────────────────────────────────────────────────────────────────\n\n json<T>(data: T, status = 200): Response {\n this.checkReleased()\n // Merge custom headers with Content-Type\n const headers = new Headers(this._headers)\n headers.set('Content-Type', 'application/json; charset=utf-8')\n return Response.json(data, { status, headers })\n }\n\n text(text: string, status = 200): Response {\n this.checkReleased()\n // Merge custom headers with Content-Type\n const headers = new Headers(this._headers)\n headers.set('Content-Type', 'text/plain; charset=utf-8')\n return new Response(text, { status, headers })\n }\n\n html(html: string, status = 200): Response {\n this.checkReleased()\n // Merge custom headers with Content-Type\n const headers = new Headers(this._headers)\n headers.set('Content-Type', 'text/html; charset=utf-8')\n return new Response(html, { status, headers })\n }\n\n /**\n * Escape HTML using Bun's SIMD-accelerated native implementation\n */\n escape(html: string): string {\n return bunEscapeHTML(html)\n }\n\n redirect(url: string, status: 301 | 302 | 303 | 307 | 308 = 302): Response {\n this.checkReleased()\n return new Response(null, {\n status,\n headers: { Location: url },\n })\n }\n\n body(data: BodyInit | null, status = 200): Response {\n this.checkReleased()\n return new Response(data, {\n status,\n headers: this._headers,\n })\n }\n\n /**\n * Send high-performance binary response (e.g. CBOR, Protobuf)\n * Utilizing Bun's native ArrayBufferSink for zero-allocation construction.\n */\n binary(data: Uint8Array | ArrayBuffer, status = 200): Response {\n this.checkReleased()\n const body = data instanceof ArrayBuffer ? new Uint8Array(data) : data\n return new Response(body as any, {\n status,\n headers: { 'Content-Type': 'application/octet-stream' },\n })\n }\n\n stream(stream: any, status = 200): Response {\n this.checkReleased()\n // Direct streaming for zero-copy socket transfers\n return new Response(stream, {\n status,\n headers: { 'Content-Type': 'application/octet-stream' },\n })\n }\n\n notFound(message = 'Not Found'): Response {\n return this.text(message, 404)\n }\n\n forbidden(message = 'Forbidden'): Response {\n return this.text(message, 403)\n }\n\n unauthorized(message = 'Unauthorized'): Response {\n return this.text(message, 401)\n }\n\n badRequest(message = 'Bad Request'): Response {\n return this.text(message, 400)\n }\n\n async forward(target: string, _options: any = {}): Promise<Response> {\n this.checkReleased()\n // Minimal implementation of forwarding\n const url = new URL(this.req.url)\n const targetUrl = new URL(\n target.startsWith('http') ? target : `${url.protocol}//${target}${this.req.path}`\n )\n\n // Copy query params\n const searchParams = new URLSearchParams(url.search)\n searchParams.forEach((v, k) => {\n targetUrl.searchParams.set(k, v)\n })\n\n return fetch(targetUrl.toString(), {\n method: this.req.method,\n headers: this.req.raw.headers,\n body: this.req.method !== 'GET' && this.req.method !== 'HEAD' ? this.req.raw.body : null,\n // @ts-expect-error - Bun/Fetch specific\n duplex: 'half',\n })\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // Header Management\n // ─────────────────────────────────────────────────────────────────────────\n\n header(name: string): string | undefined\n header(name: string, value: string): void\n header(name: string, value?: string): string | undefined | undefined {\n this.checkReleased()\n if (value !== undefined) {\n this._headers.set(name, value)\n return\n }\n return this.req.header(name)\n }\n\n /**\n * Status code setter (no-op)\n *\n * Note: Since all response helpers accept a `status` parameter,\n * this method is not actively used. Status should be set directly\n * in the response helper call (e.g., `ctx.json({}, 201)`).\n */\n status(_code: number): void {\n this.checkReleased()\n // Status is set per-response, not stored on context\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // Context Variables\n // ─────────────────────────────────────────────────────────────────────────\n\n private _store = new Map<string, any>()\n\n get<T>(key: string): T {\n return this._store.get(key)\n }\n\n set(key: string, value: any): void {\n this._store.set(key, value)\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // Request Scope Management\n // ─────────────────────────────────────────────────────────────────────────\n\n /**\n * Get the request-scoped service manager for this request.\n *\n * @returns The RequestScopeManager for this request.\n * @throws Error if called before init() or after reset().\n */\n requestScope(): RequestScopeManager {\n if (!this._requestScope) {\n throw new Error('RequestScope not initialized. Call init() first.')\n }\n return this._requestScope\n }\n\n /**\n * Resolve a request-scoped service (convenience method).\n *\n * @template T - The service type.\n * @param key - The service key for caching.\n * @param factory - Factory function to create the service.\n * @returns The cached or newly created service instance.\n */\n scoped<T>(key: string | symbol, factory: () => T): T {\n return this.requestScope().resolve(key, factory)\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // Lifecycle helpers\n // ─────────────────────────────────────────────────────────────────────────\n\n public route: (name: string, params?: any, query?: any) => string = () => ''\n\n get native(): this {\n return this\n }\n}\n",
|
|
17
|
+
"/**\n * @fileoverview MinimalContext - Ultra-lightweight Request Context\n *\n * Designed for zero-middleware static routes where pool overhead\n * exceeds the cost of creating a new object.\n *\n * Key difference from FastContext:\n * - No object pooling (direct instantiation is faster for simple cases)\n * - No Headers object reuse (creates inline)\n * - Minimal memory footprint\n *\n * @module @gravito/core/engine\n */\n\nimport { RequestScopeManager } from '../Container/RequestScopeManager'\nimport type { FastRequest, FastContext as IFastContext } from './types'\n\n// Bun runtime optimization: cache frequently used functions\nlet bunEscapeHTML: (html: string) => string = (html: string) => {\n // Default fallback for Node.js environments\n return html\n .replace(/&/g, '&')\n .replace(/</g, '<')\n .replace(/>/g, '>')\n .replace(/\"/g, '"')\n .replace(/'/g, ''')\n}\n\ntry {\n bunEscapeHTML = (require('bun') as any).escapeHTML\n} catch {\n // Already have default fallback\n}\n\n/**\n * Minimal request wrapper\n */\nclass MinimalRequest implements FastRequest {\n private _searchParams: URLSearchParams | null = null\n private _cachedQueries: Record<string, string | string[]> | null = null\n private _cachedJsonPromise: Promise<unknown> | null = null\n private _cachedTextPromise: Promise<string> | null = null\n private _cachedFormDataPromise: Promise<FormData> | null = null\n\n constructor(\n private readonly _request: Request,\n private readonly _params: Record<string, string>,\n private readonly _path: string,\n private readonly _routePattern?: string\n ) {}\n\n get url(): string {\n return this._request.url\n }\n\n get method(): string {\n return this._request.method\n }\n\n get path(): string {\n return this._path\n }\n\n get routePattern(): string | undefined {\n return this._routePattern\n }\n\n param(name: string): string | undefined {\n return this._params[name]\n }\n\n params(): Record<string, string> {\n return { ...this._params }\n }\n\n /**\n * Lazy-initialize searchParams, only parse once\n */\n private getSearchParams(): URLSearchParams {\n if (this._searchParams === null) {\n const url = this._request.url\n const queryStart = url.indexOf('?')\n if (queryStart === -1) {\n this._searchParams = new URLSearchParams()\n } else {\n const hashStart = url.indexOf('#', queryStart)\n const queryString =\n hashStart === -1 ? url.slice(queryStart + 1) : url.slice(queryStart + 1, hashStart)\n this._searchParams = new URLSearchParams(queryString)\n }\n }\n return this._searchParams\n }\n\n query(name: string): string | undefined {\n return this.getSearchParams().get(name) ?? undefined\n }\n\n queries(): Record<string, string | string[]> {\n // Cache the parsed queries object to avoid repeated parsing\n if (this._cachedQueries !== null) {\n return this._cachedQueries\n }\n\n const params = this.getSearchParams()\n const result: Record<string, string | string[]> = {}\n for (const [key, value] of params.entries()) {\n const existing = result[key]\n if (existing === undefined) {\n result[key] = value\n } else if (Array.isArray(existing)) {\n existing.push(value)\n } else {\n result[key] = [existing, value]\n }\n }\n this._cachedQueries = result\n return result\n }\n\n header(name: string): string | undefined {\n return this._request.headers.get(name) ?? undefined\n }\n\n headers(): Record<string, string> {\n const result: Record<string, string> = {}\n for (const [key, value] of this._request.headers.entries()) {\n result[key] = value\n }\n return result\n }\n\n async json<T = unknown>(): Promise<T> {\n // Cache the json promise to prevent \"Body has already been consumed\" errors\n // when multiple handlers or middleware try to read the body\n if (this._cachedJsonPromise === null) {\n this._cachedJsonPromise = this._request.json()\n }\n return this._cachedJsonPromise as Promise<T>\n }\n\n async text(): Promise<string> {\n // Cache the text promise to prevent \"Body has already been consumed\" errors\n if (this._cachedTextPromise === null) {\n this._cachedTextPromise = this._request.text()\n }\n return this._cachedTextPromise\n }\n\n async formData(): Promise<FormData> {\n // Cache the formData promise to prevent \"Body has already been consumed\" errors\n if (this._cachedFormDataPromise === null) {\n this._cachedFormDataPromise = this._request.formData()\n }\n return this._cachedFormDataPromise\n }\n\n get cookies(): Record<string, string> {\n // 1. Try Bun Native CookieMap (Injected in Bun.serve routes API)\n const nativeCookies = (this._request as any).cookies\n if (nativeCookies) {\n return nativeCookies\n }\n\n // 2. Fallback: Parse from Header manually\n const cookieHeader = this._request.headers.get('cookie')\n if (!cookieHeader) {\n return {}\n }\n\n // Manual parse implementation (optimized for speed)\n // Handles URL-encoded values and flexible separators ('; ' or ';')\n const cookies: Record<string, string> = {}\n const pairs = cookieHeader.split(/;\\s*/)\n for (let i = 0; i < pairs.length; i++) {\n const pair = pairs[i]!\n const idx = pair.indexOf('=')\n if (idx > 0) {\n const name = pair.substring(0, idx)\n const value = pair.substring(idx + 1)\n // Decode value if URL-encoded, fallback to raw value on decode error\n try {\n cookies[name] = decodeURIComponent(value)\n } catch {\n cookies[name] = value\n }\n }\n }\n return cookies\n }\n\n get raw(): Request {\n return this._request\n }\n}\n\n/**\n * MinimalContext - Optimized for simple, fast responses\n *\n * Use when:\n * - No middleware\n * - Static routes\n * - Simple JSON/text responses\n * - No custom headers needed\n */\nexport class MinimalContext implements IFastContext {\n public readonly req: MinimalRequest\n private _resHeaders: Record<string, string> = {}\n private _requestScope: RequestScopeManager\n\n constructor(\n request: Request,\n params: Record<string, string>,\n path: string,\n routePattern?: string\n ) {\n this.req = new MinimalRequest(request, params, path, routePattern)\n this._requestScope = new RequestScopeManager()\n }\n\n // get req(): FastRequest {\n // return this._req\n // }\n\n // Response helpers - merge custom headers with defaults\n // Optimized: use Object.assign instead of spread to avoid shallow copy overhead\n private getHeaders(contentType: string): Record<string, string> {\n const headers = Object.assign({ 'Content-Type': contentType }, this._resHeaders)\n return headers\n }\n\n json<T>(data: T, status = 200): Response {\n return new Response(JSON.stringify(data), {\n status,\n headers: this.getHeaders('application/json; charset=utf-8'),\n })\n }\n\n text(text: string, status = 200): Response {\n return new Response(text, {\n status,\n headers: this.getHeaders('text/plain; charset=utf-8'),\n })\n }\n\n html(html: string, status = 200): Response {\n return new Response(html, {\n status,\n headers: this.getHeaders('text/html; charset=utf-8'),\n })\n }\n\n redirect(url: string, status: 301 | 302 | 303 | 307 | 308 = 302): Response {\n return new Response(null, {\n status,\n headers: { ...this._resHeaders, Location: url },\n })\n }\n\n body(data: BodyInit | null, status = 200): Response {\n return new Response(data, {\n status,\n headers: this._resHeaders,\n })\n }\n\n header(name: string): string | undefined\n header(name: string, value: string): void\n header(name: string, value?: string): string | undefined | undefined {\n if (value !== undefined) {\n this._resHeaders[name] = value\n return\n }\n return this.req.header(name)\n }\n\n status(_code: number): void {\n // Status is set per response helper call, not stored on context\n }\n\n stream(stream: ReadableStream, status = 200): Response {\n return new Response(stream, {\n status,\n headers: this.getHeaders('application/octet-stream'),\n })\n }\n\n notFound(message = 'Not Found'): Response {\n return this.text(message, 404)\n }\n\n forbidden(message = 'Forbidden'): Response {\n return this.text(message, 403)\n }\n\n unauthorized(message = 'Unauthorized'): Response {\n return this.text(message, 401)\n }\n\n badRequest(message = 'Bad Request'): Response {\n return this.text(message, 400)\n }\n\n async forward(target: string, _options: any = {}): Promise<Response> {\n const url = new URL(this.req.url)\n const targetUrl = new URL(\n target.startsWith('http') ? target : `${url.protocol}//${target}${this.req.path}`\n )\n return fetch(targetUrl.toString(), {\n method: this.req.method,\n headers: this.req.raw.headers,\n })\n }\n\n escape(html: string): string {\n return bunEscapeHTML(html)\n }\n\n get<T>(_key: string): T {\n return undefined as any\n }\n\n set(_key: string, _value: any): void {}\n\n /**\n * Get the request-scoped service manager for this request.\n *\n * @returns The RequestScopeManager for this request.\n */\n requestScope(): RequestScopeManager {\n return this._requestScope\n }\n\n /**\n * Resolve a request-scoped service (convenience method).\n *\n * @template T - The service type.\n * @param key - The service key for caching.\n * @param factory - Factory function to create the service.\n * @returns The cached or newly created service instance.\n */\n scoped<T>(key: string | symbol, factory: () => T): T {\n return this._requestScope.resolve(key, factory)\n }\n\n public route: (name: string, params?: any, query?: any) => string = () => ''\n\n get native(): this {\n return this\n }\n\n // Required for interface compatibility\n init(_request: Request, _params?: Record<string, string>, _path?: string): this {\n throw new Error('MinimalContext does not support init. Create a new instance instead.')\n }\n\n // Required for interface compatibility\n reset(): void {\n // MinimalContext is not pooled, so no-op\n }\n}\n",
|
|
18
|
+
"/**\n * @fileoverview Lightweight Path Utilities\n *\n * High-performance path extraction without creating URL objects.\n * Performance critical - every optimization matters.\n *\n * @module @gravito/core/engine\n */\n\n/**\n * Extract pathname from URL string without creating URL object\n *\n * @param url - Full URL string (e.g., \"http://localhost:3000/api/users?id=1\")\n * @returns pathname (e.g., \"/api/users\")\n *\n * @example\n * ```typescript\n * extractPath(\"http://localhost:3000/api/users?id=1\") // \"/api/users\"\n * extractPath(\"https://example.com/\") // \"/\"\n * ```\n */\nexport function extractPath(url: string): string {\n // Optimization: use indexOf which is typically faster than JS loops\n // Find \"://\" to skip protocol\n const protocolEnd = url.indexOf('://')\n\n // Start searching after \"://\" (add 3)\n // If no \"://\", start from 0 (relative implementation or malformed)\n const searchStart = protocolEnd === -1 ? 0 : protocolEnd + 3\n\n const pathStart = url.indexOf('/', searchStart)\n\n // If no slash found after protocol, it's root \"/\" (e.g. http://localhost)\n if (pathStart === -1) {\n return '/'\n }\n\n const queryStart = url.indexOf('?', pathStart)\n\n if (queryStart === -1) {\n return url.slice(pathStart)\n }\n\n return url.slice(pathStart, queryStart)\n}\n\n/**\n * Extract pathname using simpler logic (alternative implementation)\n * Use this if the above doesn't cover edge cases\n */\nexport function extractPathSimple(url: string): string {\n // Find \"://\" then find next \"/\"\n const protocolEnd = url.indexOf('://')\n if (protocolEnd === -1) {\n // Relative URL or malformed\n const queryStart = url.indexOf('?')\n return queryStart === -1 ? url : url.slice(0, queryStart)\n }\n\n const pathStart = url.indexOf('/', protocolEnd + 3)\n if (pathStart === -1) {\n return '/'\n }\n\n const queryStart = url.indexOf('?', pathStart)\n return queryStart === -1 ? url.slice(pathStart) : url.slice(pathStart, queryStart)\n}\n",
|
|
19
|
+
"/**\n * @fileoverview Generic Object Pool Implementation\n *\n * High-performance object pooling to reduce GC pressure.\n * Implements \"fixed pool + overflow fallback\" strategy.\n *\n * @module @gravito/core/engine\n */\n\n/**\n * Generic object pool with fixed size and overflow handling\n *\n * @typeParam T - Type of objects to pool\n *\n * @example\n * ```typescript\n * const pool = new ObjectPool(\n * () => new MyObject(),\n * (obj) => obj.reset(),\n * 256\n * )\n *\n * const obj = pool.acquire()\n * try {\n * // Use object\n * } finally {\n * pool.release(obj)\n * }\n * ```\n */\nexport class ObjectPool<T> {\n private pool: T[] = []\n private readonly factory: () => T\n private readonly reset: (obj: T) => void\n private readonly maxSize: number\n\n /**\n * Create a new object pool\n *\n * @param factory - Function to create new objects\n * @param reset - Function to reset objects before reuse\n * @param maxSize - Maximum pool size (default: 256)\n */\n constructor(factory: () => T, reset: (obj: T) => void, maxSize = 256) {\n this.factory = factory\n this.reset = reset\n this.maxSize = maxSize\n }\n\n /**\n * Acquire an object from the pool\n *\n * If the pool is empty, creates a new object (overflow strategy).\n * This ensures the pool never blocks under high load.\n *\n * @returns Object from pool or newly created\n */\n acquire(): T {\n const obj = this.pool.pop()\n if (obj !== undefined) {\n return obj\n }\n\n // Pool exhausted - create new object (overflow)\n return this.factory()\n }\n\n /**\n * Release an object back to the pool\n *\n * If the pool is full, the object is discarded (will be GC'd).\n * This prevents unbounded memory growth.\n *\n * @param obj - Object to release\n */\n release(obj: T): void {\n if (this.pool.length < this.maxSize) {\n this.reset(obj)\n this.pool.push(obj)\n }\n // else: pool is full, let object be GC'd\n }\n\n /**\n * Clear all objects from the pool\n *\n * Useful for testing or when you need to force a clean slate.\n */\n clear(): void {\n this.pool = []\n }\n\n /**\n * Get current pool size\n */\n get size(): number {\n return this.pool.length\n }\n\n /**\n * Get maximum pool size\n */\n get capacity(): number {\n return this.maxSize\n }\n\n /**\n * Pre-warm the pool by creating objects in advance\n *\n * This can reduce latency for the first N requests.\n *\n * @param count - Number of objects to pre-create\n */\n prewarm(count: number): void {\n const targetSize = Math.min(count, this.maxSize)\n while (this.pool.length < targetSize) {\n const obj = this.factory()\n this.reset(obj)\n this.pool.push(obj)\n }\n }\n}\n",
|
|
20
|
+
"/**\n * @fileoverview Gravito - High-Performance Web Engine for Bun\n *\n * The standalone engine optimized exclusively for Bun runtime.\n * 99% API-compatible with Hono, but faster through Bun-specific optimizations.\n *\n * Key optimizations:\n * 1. Object pooling for zero-allocation request handling\n * 2. AOT router with O(1) static route lookup\n * 3. Lazy parsing - only parse what's accessed\n * 4. Direct Bun.serve integration without wrapper layers\n *\n * @module @gravito/core/engine\n */\n\nimport type { HttpMethod } from '../http/types'\nimport { AOTRouter } from './AOTRouter'\nimport { analyzeHandler, getOptimalContextType } from './analyzer'\nimport { CACHED_RESPONSES, HEADERS } from './constants'\nimport { FastContext as FastContextImpl } from './FastContext'\nimport { MinimalContext } from './MinimalContext'\nimport { extractPath } from './path'\nimport { ObjectPool } from './pool'\nimport type {\n CompiledHandler,\n EngineOptions,\n ErrorHandler,\n Handler,\n Middleware,\n NotFoundHandler,\n RouteMetadata,\n} from './types'\n\n// Bun runtime optimization: cache frequently used functions\nlet bunPeek = (promise: any) => promise // Default fallback\nlet bunFile = (path: string) => path // Default fallback\n\ntry {\n const bunModule = require('bun') as any\n bunPeek = bunModule.peek\n bunFile = bunModule.file\n} catch {\n // Already have default fallback\n}\n\n/**\n * Precompile middleware chain into a single function\n */\nfunction compileMiddlewareChain(middleware: Middleware[], handler: Handler): CompiledHandler {\n // Fast path: no middleware\n if (middleware.length === 0) {\n return handler as CompiledHandler\n }\n\n // Single middleware optimization - avoid wrapping overhead\n if (middleware.length === 1) {\n const mw = middleware[0]!\n return async (ctx) => {\n let nextCalled = false\n const result = mw(ctx, async () => {\n nextCalled = true\n return undefined\n })\n\n // Optimization: Only use peek/await if result is truly a Promise\n let finalResult: any\n if (result instanceof Promise) {\n const peeked = bunPeek(result)\n finalResult = peeked === result ? await result : peeked\n } else {\n finalResult = result\n }\n\n if (finalResult instanceof Response) {\n return finalResult\n }\n\n if (nextCalled) {\n const hResult = handler(ctx)\n if (hResult instanceof Promise) {\n const p = bunPeek(hResult)\n return p === hResult ? await hResult : p\n }\n return hResult\n }\n return ctx.json({ error: 'Middleware did not call next or return response' }, 500)\n }\n }\n\n // Multiple middleware: compile right-to-left into a chain\n let compiled: CompiledHandler = handler as CompiledHandler\n\n for (let i = middleware.length - 1; i >= 0; i--) {\n const mw = middleware[i]!\n const nextHandler = compiled\n compiled = async (ctx) => {\n let nextCalled = false\n const result = mw(ctx, async () => {\n nextCalled = true\n return undefined\n })\n\n // Optimization: Only use peek/await if result is truly a Promise\n let finalResult: any\n if (result instanceof Promise) {\n const peeked = bunPeek(result)\n finalResult = peeked === result ? await result : peeked\n } else {\n finalResult = result\n }\n\n if (finalResult instanceof Response) {\n return finalResult\n }\n\n if (nextCalled) {\n const nextResult = nextHandler(ctx)\n if (nextResult instanceof Promise) {\n const p = bunPeek(nextResult)\n return p === nextResult ? await nextResult : p\n }\n return nextResult\n }\n\n return ctx.json({ error: 'Middleware did not call next or return response' }, 500)\n }\n }\n\n return compiled\n}\n\n/**\n * Gravito - The High-Performance Web Engine\n */\nexport class Gravito {\n private router = new AOTRouter()\n private contextPool: ObjectPool<FastContextImpl>\n private errorHandler?: ErrorHandler\n private notFoundHandler?: NotFoundHandler\n\n // Direct reference to static routes Map (O(1) access)\n /** @internal */\n public staticRoutes!: Map<string, RouteMetadata>\n // Flag: pure static app (no middleware at all) allows ultra-fast path\n private isPureStaticApp = true\n\n // Cache for precompiled dynamic routes\n private compiledDynamicRoutes = new Map<string, { compiled: CompiledHandler; version: number }>()\n\n /**\n * Create a new Gravito instance\n *\n * @param options - Engine configuration options\n */\n constructor(options: EngineOptions = {}) {\n const poolSize = options.poolSize ?? 256\n\n // Initialize context pool\n this.contextPool = new ObjectPool(\n () => new FastContextImpl(),\n (ctx) => ctx.reset(),\n poolSize\n )\n\n // Pre-warm pool for lower first-request latency\n this.contextPool.prewarm(Math.min(32, poolSize))\n\n // Set custom handlers if provided\n if (options.onError) {\n this.errorHandler = options.onError\n }\n if (options.onNotFound) {\n this.notFoundHandler = options.onNotFound\n }\n\n // Initialize route compilation\n this.compileRoutes()\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // HTTP Method Registration\n // ─────────────────────────────────────────────────────────────────────────\n\n /**\n * Register a GET route\n */\n get(path: string, ...handlers: Handler[]): this {\n return this.addRoute('get', path, handlers)\n }\n\n /**\n * Register a POST route\n */\n post(path: string, ...handlers: Handler[]): this {\n return this.addRoute('post', path, handlers)\n }\n\n /**\n * Register a PUT route\n */\n put(path: string, ...handlers: Handler[]): this {\n return this.addRoute('put', path, handlers)\n }\n\n /**\n * Register a DELETE route\n */\n delete(path: string, ...handlers: Handler[]): this {\n return this.addRoute('delete', path, handlers)\n }\n\n /**\n * Register a PATCH route\n */\n patch(path: string, ...handlers: Handler[]): this {\n return this.addRoute('patch', path, handlers)\n }\n\n /**\n * Register an OPTIONS route\n */\n options(path: string, ...handlers: Handler[]): this {\n return this.addRoute('options', path, handlers)\n }\n\n /**\n * Register a HEAD route\n */\n head(path: string, ...handlers: Handler[]): this {\n return this.addRoute('head', path, handlers)\n }\n\n /**\n * Register a route for all HTTP methods\n */\n all(path: string, ...handlers: Handler[]): this {\n const methods: HttpMethod[] = ['get', 'post', 'put', 'delete', 'patch', 'options', 'head']\n for (const method of methods) {\n this.addRoute(method, path, handlers)\n }\n return this\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // Middleware Registration\n // ─────────────────────────────────────────────────────────────────────────\n\n /**\n * Register global or path-based middleware\n */\n use(path: string, ...middleware: Middleware[]): this\n use(...middleware: Middleware[]): this\n use(pathOrMiddleware: string | Middleware, ...middleware: Middleware[]): this {\n // Mark as not pure static since we have middleware\n this.isPureStaticApp = false\n\n if (typeof pathOrMiddleware === 'string') {\n // Path-based middleware\n this.router.usePattern(pathOrMiddleware, ...middleware)\n } else {\n // Global middleware\n this.router.use(pathOrMiddleware, ...middleware)\n }\n\n this.compileRoutes()\n return this\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // Route Grouping\n // ─────────────────────────────────────────────────────────────────────────\n\n /**\n * Mount a sub-application at a path prefix\n */\n route(path: string, app: Gravito): this {\n // Mount the sub-application's router using the AOTRouter optimization\n this.router.mount(path, app.router)\n\n // Re-compile routes to update the static route map and optimizations\n this.compileRoutes()\n\n return this\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // Error Handling\n // ─────────────────────────────────────────────────────────────────────────\n\n /**\n * Set custom error handler\n */\n onError(handler: ErrorHandler): this {\n this.errorHandler = handler\n return this\n }\n\n /**\n * Set custom 404 handler\n */\n notFound(handler: NotFoundHandler): this {\n this.notFoundHandler = handler\n return this\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // Request Handling (Bun.serve integration)\n // ─────────────────────────────────────────────────────────────────────────\n\n /**\n * Predictive Route Warming (JIT Optimization)\n *\n * Simulates requests to specified routes to trigger JIT compilation (FTL)\n * before real traffic arrives.\n *\n * @param paths List of paths to warm up (e.g. ['/api/users', '/health'])\n */\n async warmup(paths: string[]): Promise<void> {\n const dummyReqOpts = { headers: { 'User-Agent': 'Gravito-Warmup/1.0' } }\n\n for (const path of paths) {\n const req = new Request(`http://localhost${path}`, dummyReqOpts)\n await this.fetch(req)\n }\n }\n\n /**\n * Generate Native Bun.serve Configuration\n *\n * Offloads static routes to Bun's SIMD-accelerated native router.\n * Supports pre-compiled middleware chains for zero runtime lookup.\n */\n serveConfig(baseConfig: Record<string, unknown> = {}): Record<string, unknown> {\n // 1. Extract and pre-compile routes for Bun's native router\n const nativeRoutes = this.router.getNativeRoutes((handler, middleware, path) => {\n // Pre-compile the middleware chain for this specific static route\n const compiled = compileMiddlewareChain(middleware, handler)\n\n // Create a native request handler that uses our pooling\n return async (req: Request) => {\n const ctx = this.contextPool.acquire()\n ctx.init(req, {}, path, path)\n\n try {\n const result = compiled(ctx)\n\n // Pro-level optimization: skip await if synchronous\n let response: Response\n if (result instanceof Promise) {\n const peeked = bunPeek(result)\n response = peeked === result ? await result : peeked\n } else {\n response = result\n }\n\n // Cleanup and release\n const cleanup = ctx.requestScope().cleanup()\n if (cleanup instanceof Promise) {\n await cleanup\n }\n this.contextPool.release(ctx)\n return response\n } catch (error) {\n // Cleanup even on error\n const cleanup = ctx.requestScope().cleanup()\n if (cleanup instanceof Promise) {\n await cleanup\n }\n this.contextPool.release(ctx)\n return this.handleErrorSync(error as Error, req, path)\n }\n }\n })\n\n return {\n ...baseConfig,\n routes: nativeRoutes,\n fetch: this.fetch, // Fallback for dynamic routes\n // Optimize TLS if provided in baseConfig\n tls: baseConfig.tls ? this.optimizeTLS(baseConfig.tls) : undefined,\n error: (error: Error) => {\n console.error('Native route error:', error)\n return new Response(CACHED_RESPONSES.INTERNAL_ERROR, {\n status: 500,\n headers: HEADERS.JSON,\n })\n },\n }\n }\n\n /**\n * Optimize TLS Configuration for Bun 1.39+\n */\n private optimizeTLS(tls: any): any {\n const isProd = process.env.NODE_ENV === 'production'\n\n const optimizeEntry = (entry: any) => {\n const optimized = { ...entry }\n if (typeof optimized.key === 'string' && !optimized.key.startsWith('-----BEGIN')) {\n optimized.key = bunFile(optimized.key)\n }\n if (typeof optimized.cert === 'string' && !optimized.cert.startsWith('-----BEGIN')) {\n optimized.cert = bunFile(optimized.cert)\n }\n if (isProd && optimized.lowMemoryMode === undefined) {\n optimized.lowMemoryMode = true\n }\n return optimized\n }\n\n return Array.isArray(tls) ? tls.map(optimizeEntry) : optimizeEntry(tls)\n }\n\n /**\n * Handle an incoming request\n */\n fetch = async (request: Request): Promise<Response> => {\n // Fast path: extract pathname without creating URL object\n const path = extractPath(request.url)\n const method = request.method.toLowerCase()\n\n // Try static route first (O(1) lookup, inlined for performance)\n const staticKey = `${method}:${path}`\n const staticRoute = this.staticRoutes.get(staticKey)\n\n if (staticRoute) {\n const ctx = this.contextPool.acquire()\n ctx.init(request, {}, path, path)\n\n try {\n const compiled =\n staticRoute.compiled ||\n compileMiddlewareChain(staticRoute.middleware, staticRoute.handler)\n const result = compiled(ctx)\n\n let response: Response\n if (result instanceof Promise) {\n const peeked = bunPeek(result)\n response = peeked === result ? await result : peeked\n } else {\n response = result\n }\n\n const cleanup = ctx.requestScope().cleanup()\n if (cleanup instanceof Promise) {\n await cleanup\n }\n this.contextPool.release(ctx)\n return response\n } catch (error) {\n const cleanup = ctx.requestScope().cleanup()\n if (cleanup instanceof Promise) {\n await cleanup\n }\n this.contextPool.release(ctx)\n return this.handleErrorSync(error as Error, request, path)\n }\n }\n\n // Dynamic route: use Radix Tree\n return await this.handleDynamicRoute(request, method, path)\n }\n\n /**\n * Handle dynamic routes with Radix Tree\n */\n private async handleDynamicRoute(\n request: Request,\n method: string,\n path: string\n ): Promise<Response> {\n const match = this.router.match(method, path)\n\n if (match.handler) {\n const ctx = this.contextPool.acquire()\n ctx.init(request, match.params, path, match.routePattern)\n\n try {\n // Use cached compilation if available\n const routeKey = `${method}:${match.routePattern}`\n let compiledObj = this.compiledDynamicRoutes.get(routeKey)\n\n if (!compiledObj || compiledObj.version !== this.router.version) {\n const compiled = compileMiddlewareChain(match.middleware, match.handler)\n compiledObj = { compiled, version: this.router.version }\n this.compiledDynamicRoutes.set(routeKey, compiledObj)\n }\n\n const result = compiledObj.compiled(ctx)\n\n let response: Response\n if (result instanceof Promise) {\n const peeked = bunPeek(result)\n response = peeked === result ? await result : peeked\n } else {\n response = result\n }\n\n const cleanup = ctx.requestScope().cleanup()\n if (cleanup instanceof Promise) {\n await cleanup\n }\n this.contextPool.release(ctx)\n return response\n } catch (error) {\n const cleanup = ctx.requestScope().cleanup()\n if (cleanup instanceof Promise) {\n await cleanup\n }\n this.contextPool.release(ctx)\n return this.handleErrorSync(error as Error, request, path)\n }\n }\n\n // 404\n return this.handleNotFoundSync(request, path)\n }\n\n /**\n * Sync error handler (for ultra-fast path)\n */\n private handleErrorSync(\n error: Error,\n request: Request,\n path: string\n ): Response | Promise<Response> {\n if (this.errorHandler) {\n const ctx = new MinimalContext(request, {}, path)\n const result = this.errorHandler(error, ctx)\n if (result instanceof Response) {\n return result\n }\n return result as Promise<Response>\n }\n\n console.error('Unhandled error:', error)\n return new Response(CACHED_RESPONSES.INTERNAL_ERROR, {\n status: 500,\n headers: HEADERS.JSON,\n })\n }\n\n /**\n * Sync 404 handler (for ultra-fast path)\n */\n private handleNotFoundSync(request: Request, path: string): Response | Promise<Response> {\n if (this.notFoundHandler) {\n const ctx = new MinimalContext(request, {}, path)\n const result = this.notFoundHandler(ctx)\n if (result instanceof Response) {\n return result\n }\n return result as Promise<Response>\n }\n\n return new Response(CACHED_RESPONSES.NOT_FOUND, {\n status: 404,\n headers: HEADERS.JSON,\n })\n }\n\n /**\n * Collect middleware for a specific path\n */\n private collectMiddlewareForPath(path: string, routeMiddleware: Middleware[]): Middleware[] {\n if (this.router.globalMiddleware.length === 0 && this.router.pathMiddleware.size === 0) {\n return routeMiddleware\n }\n\n return this.router.collectMiddlewarePublic(path, routeMiddleware)\n }\n\n /**\n * Compile routes for optimization\n */\n private compileRoutes(): void {\n this.staticRoutes = this.router.staticRoutes\n\n // Check if pure static app\n const hasGlobalMiddleware = this.router.globalMiddleware.length > 0\n const hasPathMiddleware = this.router.pathMiddleware.size > 0\n\n this.isPureStaticApp = !hasGlobalMiddleware && !hasPathMiddleware\n\n // Pre-mark routes\n for (const [key, route] of this.staticRoutes) {\n // Skip if already compiled for this version\n if (route.compiledVersion === this.router.version) {\n continue\n }\n\n const analysis = analyzeHandler(route.handler)\n const optimalType = getOptimalContextType(analysis)\n\n route.useMinimal =\n this.isPureStaticApp && route.middleware.length === 0 && optimalType === 'minimal'\n\n // Precompile middleware chain\n if (!route.useMinimal) {\n const allMiddleware = this.collectMiddlewareForPath(key.split(':')[1]!, route.middleware)\n route.compiled = compileMiddlewareChain(allMiddleware, route.handler)\n }\n\n route.compiledVersion = this.router.version\n }\n }\n\n /**\n * Add a route to the router\n */\n private addRoute(method: string, path: string, handlers: Handler[]): this {\n if (handlers.length === 0) {\n throw new Error(`No handler provided for ${method.toUpperCase()} ${path}`)\n }\n\n const handler = handlers[handlers.length - 1]!\n const middleware = handlers.slice(0, -1) as unknown as Middleware[]\n\n this.router.add(method as HttpMethod, path, handler, middleware)\n\n // Re-compile routes when new ones are added\n this.compileRoutes()\n\n return this\n }\n}\n",
|
|
21
|
+
"import { Gravito } from '../engine/Gravito'\nimport type {\n GravitoContext,\n GravitoErrorHandler,\n GravitoHandler,\n GravitoMiddleware,\n GravitoNotFoundHandler,\n GravitoVariables,\n HttpMethod,\n} from '../http/types'\nimport type { AdapterConfig, HttpAdapter, RouteDefinition } from './types'\n\n/**\n * GravitoEngineAdapter - Optimized adapter using the Standalone Gravito Engine\n *\n * This adapter is exclusively for Bun and provides the best performance\n * by using the specialized Gravito engine with object pooling and AOT routing.\n */\nexport class GravitoEngineAdapter<V extends GravitoVariables = GravitoVariables>\n implements HttpAdapter<V>\n{\n readonly name = 'gravito-engine'\n readonly version = '1.0.0'\n\n private engine: Gravito\n\n constructor(config: AdapterConfig = {}) {\n this.engine = new Gravito(config.engineOptions)\n }\n\n get native(): Gravito {\n return this.engine\n }\n\n route(\n method: HttpMethod,\n path: string,\n ...handlers: (GravitoHandler<V> | GravitoMiddleware<V>)[]\n ): void {\n // Gravito engine uses Hono-like API for methods\n const methodFn = (this.engine as any)[method.toLowerCase()]\n if (typeof methodFn !== 'function') {\n throw new Error(`Unsupported HTTP method: ${method}`)\n }\n methodFn.call(this.engine, path, ...handlers)\n }\n\n routes(routes: RouteDefinition[]): void {\n for (const route of routes) {\n this.route(route.method, route.path, ...(route.handlers as any[]))\n }\n }\n\n use(path: string, ...middleware: GravitoMiddleware<V>[]): void {\n this.engine.use(path, ...(middleware as any))\n }\n\n useGlobal(...middleware: GravitoMiddleware<V>[]): void {\n this.engine.use(...(middleware as any))\n }\n\n mount(path: string, subAdapter: HttpAdapter<V>): void {\n // For now, simple relay\n this.use(`${path}/*`, async (ctx) => {\n return await subAdapter.fetch(ctx.req.raw)\n })\n }\n\n onError(handler: GravitoErrorHandler<V>): void {\n this.engine.onError(handler as any)\n }\n\n onNotFound(handler: GravitoNotFoundHandler<V>): void {\n this.engine.notFound(handler as any)\n }\n\n fetch = (request: Request, _server?: unknown): Response | Promise<Response> => {\n return this.engine.fetch(request)\n }\n\n async warmup(paths: string[]): Promise<void> {\n await this.engine.warmup(paths)\n }\n\n createContext(_request: Request): GravitoContext<V> {\n // Gravito engine has its own internal context management\n throw new Error('GravitoEngineAdapter manages context internally through pooling.')\n }\n\n useScoped(scope: string, path: string, ...middleware: GravitoMiddleware<V>[]): void {\n if (path === '*' || path === '*/*') {\n throw new Error(\n `useScoped(): Cannot use wildcard path '*' in Orbit-scoped middleware. ` +\n `Use regular use('*') for global middleware, or specify explicit paths like '${scope}/*'`\n )\n }\n\n const normalizedScope = scope.startsWith('/') ? scope : `/${scope}`\n const fullPath = normalizedScope + (path.startsWith('/') ? '' : '/') + path\n\n this.use(fullPath, ...middleware)\n }\n}\n",
|
|
22
|
+
"/**\n * @fileoverview HTTP Adapter Interface for Gravito Framework\n *\n * This module defines the contract that all HTTP adapters must implement.\n * By programming to this interface, Gravito can swap out the underlying\n * HTTP engine without changing application code.\n *\n * @module @gravito/core/adapters\n * @since 2.0.0\n */\n\nimport type {\n GravitoContext,\n GravitoErrorHandler,\n GravitoHandler,\n GravitoMiddleware,\n GravitoNotFoundHandler,\n GravitoVariables,\n HttpMethod,\n} from '../http/types'\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Adapter Configuration\n// ─────────────────────────────────────────────────────────────────────────────\n\n/**\n * Configuration options for HTTP adapters\n */\nexport interface AdapterConfig {\n /**\n * Base path prefix for all routes\n * @default ''\n */\n basePath?: string\n\n /**\n * Whether to enable strict routing (trailing slashes matter)\n * @default false\n */\n strictRouting?: boolean\n\n /**\n * Custom options passed to the underlying HTTP engine\n */\n engineOptions?: Record<string, unknown>\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Route Definition\n// ─────────────────────────────────────────────────────────────────────────────\n\n/**\n * Route definition structure\n */\nexport interface RouteDefinition {\n method: HttpMethod\n path: string\n handlers: (GravitoHandler | GravitoMiddleware)[]\n name?: string\n middleware?: GravitoMiddleware[]\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// HTTP Adapter Interface\n// ─────────────────────────────────────────────────────────────────────────────\n\n/**\n * HttpAdapter - The core interface for HTTP engine abstraction\n *\n * Any HTTP engine (Photon, Express, Fastify, custom Bun implementation)\n * must implement this interface to be usable with Gravito.\n *\n * @typeParam V - Context variables type\n *\n * @example\n * ```typescript\n * // Using the default Photon adapter\n * import { PhotonAdapter } from '@gravito/core/adapters'\n *\n * const core = new PlanetCore({\n * adapter: new PhotonAdapter()\n * })\n *\n * // Using a custom adapter\n * import { BunNativeAdapter } from '@gravito/adapter-bun'\n *\n * const core = new PlanetCore({\n * adapter: new BunNativeAdapter()\n * })\n * ```\n */\nexport interface HttpAdapter<V extends GravitoVariables = GravitoVariables> {\n // ─────────────────────────────────────────────\n // Identity\n // ─────────────────────────────────────────────\n\n /**\n * Adapter name for identification\n * @example 'photon', 'bun-native', 'express'\n */\n readonly name: string\n\n /**\n * Adapter version\n */\n readonly version: string\n\n // ─────────────────────────────────────────────\n // Native Access\n // ─────────────────────────────────────────────\n\n /**\n * Access the underlying native HTTP engine instance.\n *\n * ⚠️ WARNING: Using this ties your code to a specific adapter.\n *\n * @example\n * ```typescript\n * // For Photon adapter\n * const photonApp = adapter.native as Photon\n *\n * // For custom Bun adapter\n * const bunApp = adapter.native as BunApp\n * ```\n */\n readonly native: unknown\n\n // ─────────────────────────────────────────────\n // Routing\n // ─────────────────────────────────────────────\n\n /**\n * Register a route with the adapter\n *\n * @param method - HTTP method\n * @param path - Route path (may include parameters like ':id')\n * @param handlers - One or more handlers for this route (handlers or middleware)\n */\n route(\n method: HttpMethod,\n path: string,\n ...handlers: (GravitoHandler<V> | GravitoMiddleware<V>)[]\n ): void\n\n /**\n * Register multiple routes at once\n *\n * @param routes - Array of route definitions\n */\n routes(routes: RouteDefinition[]): void\n\n /**\n * Register a middleware for a path\n *\n * @param path - Path pattern to match\n * @param middleware - One or more middleware functions\n */\n use(path: string, ...middleware: GravitoMiddleware<V>[]): void\n\n /**\n * Register a global middleware (applied to all routes)\n *\n * @param middleware - Middleware function\n */\n useGlobal(...middleware: GravitoMiddleware<V>[]): void\n\n /**\n * Register a scoped middleware for Orbit-level isolation\n *\n * Unlike regular `use()`, this method enforces stricter scoping rules:\n * - REJECTS '*' wildcard paths (prevents global middleware in Orbits)\n * - ENFORCES that all middleware paths must include the scope prefix\n * - Throws error if attempting to register global middleware within an Orbit scope\n *\n * This is designed to prevent accidental middleware cross-contamination\n * when multiple Orbits are mounted to a single PlanetCore instance.\n *\n * @param scope - The scope/path prefix (e.g., '/api', '/blog')\n * @param path - Path pattern to match (cannot be '*')\n * @param middleware - One or more middleware functions\n * @throws {Error} If path is '*' when in Orbit scope\n * @since 2.3.0\n *\n * @example\n * ```typescript\n * // Correct: Scoped to specific path\n * adapter.useScoped('/api', '/users/*', authMiddleware)\n *\n * // Error: Cannot use wildcard in Orbit scope\n * adapter.useScoped('/api', '*', loggerMiddleware)\n * ```\n */\n useScoped(scope: string, path: string, ...middleware: GravitoMiddleware<V>[]): void\n\n /**\n * Mount a sub-adapter at a path\n *\n * @param path - Mount path\n * @param subAdapter - The adapter to mount\n */\n mount(path: string, subAdapter: HttpAdapter<V>): void\n\n // ─────────────────────────────────────────────\n // Error Handling\n // ─────────────────────────────────────────────\n\n /**\n * Set the error handler\n *\n * @param handler - Error handler function\n */\n onError(handler: GravitoErrorHandler<V>): void\n\n /**\n * Set the not-found handler\n *\n * @param handler - Not-found handler function\n */\n onNotFound(handler: GravitoNotFoundHandler<V>): void\n\n // ─────────────────────────────────────────────\n // Request Handling\n // ─────────────────────────────────────────────\n\n /**\n * The main fetch handler for serving requests.\n *\n * This is compatible with `Bun.serve()`, Cloudflare Workers,\n * and other fetch-based runtimes.\n *\n * @param request - Incoming HTTP request\n * @param server - Optional server context (Bun.Server, etc.)\n * @returns HTTP response\n *\n * @example\n * ```typescript\n * // With Bun.serve\n * Bun.serve({\n * port: 3000,\n * fetch: adapter.fetch\n * })\n * ```\n */\n fetch(request: Request, server?: unknown): Response | Promise<Response>\n\n /**\n * Predictive Route Warming (JIT Optimization)\n *\n * Simulates requests to specified routes to trigger JIT compilation (FTL)\n * before real traffic arrives.\n *\n * @param paths List of paths to warm up (e.g. ['/api/users', '/health'])\n * @since 2.1.0\n */\n warmup?(paths: string[]): Promise<void>\n\n /**\n * WebSocket Handler for Bun.serve\n *\n * @since 2.2.0\n */\n websocket?: {\n open?(ws: unknown): void | Promise<void>\n message?(ws: unknown, message: string | Buffer | Uint8Array): void | Promise<void>\n close?(ws: unknown, code: number, message: string): void | Promise<void>\n drain?(ws: unknown): void | Promise<void>\n // Allow any other Bun websocket properties\n [key: string]: unknown\n }\n\n // ─────────────────────────────────────────────\n // Lifecycle\n // ─────────────────────────────────────────────\n\n /**\n * Initialize the adapter\n *\n * Called during PlanetCore.boot()\n */\n init?(): void | Promise<void>\n\n /**\n * Cleanup resources\n *\n * Called during graceful shutdown\n */\n shutdown?(): void | Promise<void>\n\n // ─────────────────────────────────────────────\n // Context Factory\n // ─────────────────────────────────────────────\n\n /**\n * Create a GravitoContext from a raw request.\n *\n * This is used internally for testing and advanced scenarios.\n *\n * @param request - Raw HTTP request\n * @returns Gravito context\n */\n createContext(request: Request): GravitoContext<V>\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Adapter Factory\n// ─────────────────────────────────────────────────────────────────────────────\n\n/**\n * Factory function type for creating adapters\n */\nexport type AdapterFactory<V extends GravitoVariables = GravitoVariables> = (\n config?: AdapterConfig\n) => HttpAdapter<V>\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Type Guards\n// ─────────────────────────────────────────────────────────────────────────────\n\n/**\n * Check if a value is an HttpAdapter\n */\nexport function isHttpAdapter(value: unknown): value is HttpAdapter {\n return (\n typeof value === 'object' &&\n value !== null &&\n 'name' in value &&\n 'fetch' in value &&\n typeof (value as HttpAdapter).fetch === 'function' &&\n 'route' in value &&\n typeof (value as HttpAdapter).route === 'function'\n )\n}\n",
|
|
23
|
+
"/**\n * Bun runtime adapter implementation.\n *\n * @module runtime/adapter-bun\n * @since 3.2.0\n */\n\nimport type { RuntimeAdapter, RuntimeSpawnOptions } from './types'\n\nfunction createEphemeralPort(): number {\n return Math.floor(Math.random() * 50000) + 10000\n}\n\n/**\n * Create a RuntimeAdapter for the Bun runtime.\n * @internal\n */\nexport function createBunAdapter(): RuntimeAdapter {\n return {\n kind: 'bun',\n spawn(command, options = {}) {\n const [cmd, ...args] = command\n if (!cmd) {\n throw new Error('[RuntimeAdapter] spawn() requires a command')\n }\n const proc = Bun.spawn([cmd, ...args], {\n cwd: options.cwd,\n env: options.env,\n stdin: options.stdin,\n stdout: options.stdout ?? 'pipe',\n stderr: options.stderr ?? 'pipe',\n })\n\n let timeoutHandle: Timer | undefined\n let abortListener: (() => void) | undefined\n\n const exitedWithTimeout = new Promise<number>((resolve, reject) => {\n if (options.timeout) {\n timeoutHandle = setTimeout(() => {\n proc.kill('SIGTERM')\n reject(new Error('[RuntimeAdapter] Process timeout'))\n }, options.timeout)\n }\n\n if (options.signal) {\n abortListener = () => {\n proc.kill('SIGTERM')\n reject(new DOMException('Aborted', 'AbortError'))\n }\n options.signal.addEventListener('abort', abortListener)\n }\n\n proc.exited\n .then(resolve)\n .catch(reject)\n .finally(() => {\n if (timeoutHandle) {\n clearTimeout(timeoutHandle)\n }\n if (abortListener && options.signal) {\n options.signal.removeEventListener('abort', abortListener)\n }\n })\n })\n\n return {\n exited: exitedWithTimeout,\n stdout: proc.stdout ?? null,\n stderr: proc.stderr ?? null,\n kill: (signal?: string | number) => {\n const bunSignal =\n typeof signal === 'number' ? signal : (signal as NodeJS.Signals | undefined)\n proc.kill(bunSignal)\n },\n unref: () => {\n // Bun.spawn doesn't have unref\n },\n resourceUsage: async () => {\n // Bun doesn't expose resource usage through spawn API\n return undefined\n },\n }\n },\n async spawnAndCollect(command, options = {}) {\n const [cmd, ...args] = command\n if (!cmd) {\n throw new Error('[RuntimeAdapter] spawn() requires a command')\n }\n\n const proc = Bun.spawn([cmd, ...args], {\n cwd: options.cwd,\n env: options.env,\n stdin: options.stdin,\n stdout: 'pipe',\n stderr: 'pipe',\n })\n\n let timedOut = false\n let timeoutHandle: Timer | undefined\n\n const exitPromise = proc.exited\n .then((code: number) => ({ code, timedOut }))\n .catch(() => ({\n code: -1,\n timedOut: true,\n }))\n\n if (options.timeout) {\n const timeoutPromise = new Promise<{\n code: number\n timedOut: boolean\n }>((resolve) => {\n timeoutHandle = setTimeout(() => {\n timedOut = true\n proc.kill('SIGTERM')\n resolve({ code: -1, timedOut: true })\n }, options.timeout)\n })\n\n const result = await Promise.race([exitPromise, timeoutPromise])\n\n if (timeoutHandle) {\n clearTimeout(timeoutHandle)\n }\n\n const [stdoutText, stderrText] = await Promise.all([\n new Response(proc.stdout ?? null).text(),\n new Response(proc.stderr ?? null).text(),\n ])\n\n return {\n exitCode: result.code,\n stdout: stdoutText,\n stderr: stderrText,\n success: result.code === 0 && !result.timedOut,\n timedOut: result.timedOut,\n }\n }\n\n let abortListener: (() => void) | undefined\n if (options.signal) {\n abortListener = () => {\n proc.kill('SIGTERM')\n }\n options.signal.addEventListener('abort', abortListener)\n }\n\n const { code, timedOut: timeout } = await exitPromise\n\n if (abortListener && options.signal) {\n options.signal.removeEventListener('abort', abortListener)\n }\n\n const [stdoutText, stderrText] = await Promise.all([\n new Response(proc.stdout ?? null).text(),\n new Response(proc.stderr ?? null).text(),\n ])\n\n return {\n exitCode: code,\n stdout: stdoutText,\n stderr: stderrText,\n success: code === 0 && !timeout,\n timedOut: timeout,\n }\n },\n spawnSync(command, options: Omit<RuntimeSpawnOptions, 'signal'> = {}) {\n const [cmd, ...args] = command\n if (!cmd) {\n throw new Error('[RuntimeAdapter] spawn() requires a command')\n }\n\n const timedOut = false\n const proc = Bun.spawnSync([cmd, ...args], {\n cwd: options.cwd,\n env: options.env,\n stdin: options.stdin,\n stdout: 'pipe',\n stderr: 'pipe',\n })\n\n const decoder = new TextDecoder('utf-8', {\n fatal: false,\n })\n const stdoutText = decoder.decode(proc.stdout ?? new Uint8Array())\n const stderrText = decoder.decode(proc.stderr ?? new Uint8Array())\n\n return {\n exitCode: proc.exitCode ?? 0,\n stdout: stdoutText,\n stderr: stderrText,\n success: (proc.exitCode ?? 0) === 0 && !timedOut,\n timedOut,\n }\n },\n async writeFile(path, data) {\n await Bun.write(path, data)\n },\n async appendFile(path, data) {\n const file = Bun.file(path)\n const writer = file.writer()\n writer.write(data)\n await writer.end()\n },\n createFileSink(path) {\n const writer = Bun.file(path).writer()\n return {\n write(data) {\n writer.write(data)\n },\n async flush() {\n await writer.flush()\n },\n async end() {\n await writer.end()\n },\n }\n },\n async readFile(path) {\n const file = Bun.file(path)\n const buffer = await file.arrayBuffer()\n return new Uint8Array(buffer)\n },\n async readFileAsBlob(path) {\n return Bun.file(path)\n },\n async exists(path) {\n return await Bun.file(path).exists()\n },\n async stat(path) {\n const stats = await Bun.file(path).stat()\n return { size: stats.size }\n },\n async deleteFile(path) {\n const fs = await import('node:fs/promises')\n try {\n await fs.unlink(path)\n } catch {\n // Ignore if not found\n }\n },\n async mkdir(path, options = {}) {\n const fs = await import('node:fs/promises')\n await fs.mkdir(path, options)\n },\n async readDir(path) {\n const fs = await import('node:fs/promises')\n const entries = await fs.readdir(path, { withFileTypes: true })\n return entries.map((entry) => ({\n name: entry.name,\n isFile: entry.isFile(),\n isDirectory: entry.isDirectory(),\n }))\n },\n async removeRecursive(path) {\n const fs = await import('node:fs/promises')\n await fs.rm(path, { recursive: true, force: true })\n },\n serve(config) {\n if (config.port !== 0) {\n return Bun.serve({\n port: config.port,\n fetch: config.fetch,\n websocket: config.websocket as any,\n })\n }\n\n let lastError: unknown\n for (let attempt = 0; attempt < 20; attempt++) {\n try {\n return Bun.serve({\n port: createEphemeralPort(),\n fetch: config.fetch,\n websocket: config.websocket as any,\n })\n } catch (error) {\n lastError = error\n }\n }\n\n throw lastError ?? new Error('[RuntimeAdapter] Failed to allocate an ephemeral port')\n },\n }\n}\n",
|
|
24
|
+
"/**\n * Runtime detection utilities.\n *\n * @module runtime/detection\n * @since 3.2.0\n */\n\nimport type { RuntimeKind } from './types'\n\n/**\n * Detect the current JavaScript runtime environment.\n * @internal\n */\nexport function getRuntimeKind(): RuntimeKind {\n if (typeof Bun !== 'undefined' && typeof Bun.spawn === 'function') {\n return 'bun'\n }\n const denoRuntime = (globalThis as any).Deno\n if (typeof denoRuntime !== 'undefined' && typeof denoRuntime?.version?.deno === 'string') {\n return 'deno'\n }\n if (typeof process !== 'undefined' && process.versions?.node) {\n return 'node'\n }\n return 'unknown'\n}\n\n/**\n * Get environment variables from the current runtime.\n * @public\n */\nexport function getRuntimeEnv(): Record<string, string | undefined> {\n const kind = getRuntimeKind()\n if (kind === 'bun' && typeof Bun !== 'undefined') {\n return Bun.env\n }\n if (kind === 'deno') {\n const deno = (globalThis as any).Deno\n if (deno?.env?.toObject) {\n return deno.env.toObject()\n }\n }\n if (typeof process !== 'undefined' && process.env) {\n return process.env\n }\n return {}\n}\n\n/**\n * Convert various data types to Uint8Array.\n * @internal\n */\nexport async function toUint8Array(\n data: Blob | Buffer | string | ArrayBuffer | Uint8Array\n): Promise<Uint8Array> {\n if (data instanceof Uint8Array) {\n return data\n }\n if (typeof data === 'string') {\n return new TextEncoder().encode(data)\n }\n if (data instanceof ArrayBuffer) {\n return new Uint8Array(data)\n }\n if (typeof Buffer !== 'undefined' && data instanceof Buffer) {\n return new Uint8Array(data)\n }\n if (data instanceof Blob) {\n return new Uint8Array(await data.arrayBuffer())\n }\n return new Uint8Array()\n}\n",
|
|
25
|
+
"/**\n * Deno runtime adapter implementation.\n *\n * @module runtime/adapter-deno\n * @since 3.2.0\n */\n\nimport { toUint8Array } from './detection'\nimport type { RuntimeAdapter, RuntimeSpawnOptions } from './types'\n\n/**\n * Map stdio option to Deno-compatible value.\n * @internal\n */\nfunction mapDenoStdio(\n value: RuntimeSpawnOptions['stdout'] | undefined,\n defaultValue: 'piped' | 'null'\n): 'inherit' | 'piped' | 'null' {\n if (value === 'inherit') {\n return 'inherit'\n }\n if (value === 'ignore') {\n return 'null'\n }\n return defaultValue\n}\n\n/**\n * Create a RuntimeAdapter for the Deno runtime.\n * @internal\n */\nexport function createDenoAdapter(): RuntimeAdapter {\n return {\n kind: 'deno',\n spawn(command, options = {}) {\n const [cmd, ...args] = command\n if (!cmd) {\n throw new Error('[RuntimeAdapter] spawn() requires a command')\n }\n const deno = (globalThis as any).Deno\n if (!deno?.Command) {\n throw new Error('[RuntimeAdapter] Deno runtime is required for spawn()')\n }\n\n const proc = new deno.Command(cmd, {\n args,\n cwd: options.cwd,\n env: options.env,\n stdin: mapDenoStdio(options.stdin, 'piped'),\n stdout: mapDenoStdio(options.stdout, 'piped'),\n stderr: mapDenoStdio(options.stderr, 'piped'),\n }).spawn()\n\n let timeoutHandle: number | undefined\n let abortListener: (() => void) | undefined\n\n const exitedWithTimeout = new Promise<number>((resolve, reject) => {\n if (options.timeout) {\n timeoutHandle = setTimeout(() => {\n proc.kill('SIGTERM')\n reject(new Error('[RuntimeAdapter] Process timeout'))\n }, options.timeout) as unknown as number\n }\n\n if (options.signal) {\n abortListener = () => {\n proc.kill('SIGTERM')\n reject(new DOMException('Aborted', 'AbortError'))\n }\n options.signal.addEventListener('abort', abortListener)\n }\n\n proc.status\n .then((status: { code: number }) => {\n if (timeoutHandle) {\n clearTimeout(timeoutHandle)\n }\n if (abortListener && options.signal) {\n options.signal.removeEventListener('abort', abortListener)\n }\n resolve(status.code ?? 0)\n })\n .catch(reject)\n })\n\n return {\n exited: exitedWithTimeout,\n stdout: (proc.stdout as unknown as ReadableStream<Uint8Array>) ?? null,\n stderr: (proc.stderr as unknown as ReadableStream<Uint8Array>) ?? null,\n kill: (signal?: string | number) => {\n const killSignal = typeof signal === 'string' ? signal : 'SIGTERM'\n proc.kill(killSignal)\n },\n }\n },\n async spawnAndCollect(command, options = {}) {\n const [cmd, ...args] = command\n if (!cmd) {\n throw new Error('[RuntimeAdapter] spawn() requires a command')\n }\n const deno = (globalThis as any).Deno\n if (!deno?.Command) {\n throw new Error('[RuntimeAdapter] Deno runtime is required for spawn()')\n }\n\n const proc = new deno.Command(cmd, {\n args,\n cwd: options.cwd,\n env: options.env,\n stdout: 'piped',\n stderr: 'piped',\n stdin: mapDenoStdio(options.stdin, 'null'),\n }).spawn()\n\n let timedOut = false\n let timeoutHandle: number | undefined\n let abortListener: (() => void) | undefined\n\n const statusPromise = proc.status.then((status: { code: number; success: boolean }) => ({\n code: status.code ?? 0,\n timedOut,\n }))\n\n if (options.timeout) {\n const timeoutPromise = new Promise<{\n code: number\n timedOut: boolean\n }>((resolve) => {\n timeoutHandle = setTimeout(() => {\n timedOut = true\n proc.kill('SIGTERM')\n resolve({ code: -1, timedOut: true })\n }, options.timeout) as unknown as number\n })\n\n const raceResult = await Promise.race([statusPromise, timeoutPromise])\n\n if (timeoutHandle) {\n clearTimeout(timeoutHandle)\n }\n\n const [stdoutText, stderrText] = await Promise.all([\n new Response(proc.stdout ?? null).text(),\n new Response(proc.stderr ?? null).text(),\n ])\n\n return {\n exitCode: raceResult.code,\n stdout: stdoutText,\n stderr: stderrText,\n success: raceResult.code === 0 && !raceResult.timedOut,\n timedOut: raceResult.timedOut,\n }\n }\n\n if (options.signal) {\n abortListener = () => {\n proc.kill('SIGTERM')\n }\n options.signal.addEventListener('abort', abortListener)\n }\n\n const { code, timedOut: timeout } = await statusPromise\n\n if (abortListener && options.signal) {\n options.signal.removeEventListener('abort', abortListener)\n }\n\n const [stdoutText, stderrText] = await Promise.all([\n new Response(proc.stdout ?? null).text(),\n new Response(proc.stderr ?? null).text(),\n ])\n\n return {\n exitCode: code,\n stdout: stdoutText,\n stderr: stderrText,\n success: code === 0 && !timeout,\n timedOut: timeout,\n }\n },\n spawnSync(command, options = {}) {\n const [cmd, ...args] = command\n if (!cmd) {\n throw new Error('[RuntimeAdapter] spawn() requires a command')\n }\n const deno = (globalThis as any).Deno\n if (!deno?.Command) {\n throw new Error('[RuntimeAdapter] Deno runtime is required for spawnSync()')\n }\n\n const result = new deno.Command(cmd, {\n args,\n cwd: options.cwd,\n env: options.env,\n stdin: mapDenoStdio(options.stdin, 'null'),\n }).outputSync()\n\n const decoder = new TextDecoder('utf-8', {\n fatal: false,\n })\n const stdoutText = decoder.decode(result.stdout ?? new Uint8Array())\n const stderrText = decoder.decode(result.stderr ?? new Uint8Array())\n\n return {\n exitCode: result.code ?? 0,\n stdout: stdoutText,\n stderr: stderrText,\n success: (result.code ?? 0) === 0,\n timedOut: false,\n }\n },\n async writeFile(path, data) {\n const deno = (globalThis as any).Deno\n if (!deno?.writeFile) {\n throw new Error('[RuntimeAdapter] Deno runtime is required for writeFile()')\n }\n const payload = await toUint8Array(data)\n await deno.writeFile(path, payload)\n },\n async readFile(path) {\n const deno = (globalThis as any).Deno\n if (!deno?.readFile) {\n throw new Error('[RuntimeAdapter] Deno runtime is required for readFile()')\n }\n return await deno.readFile(path)\n },\n async readFileAsBlob(path) {\n const buffer = await this.readFile(path)\n return new Blob([buffer as unknown as BlobPart])\n },\n async exists(path) {\n const deno = (globalThis as any).Deno\n if (!deno?.stat) {\n throw new Error('[RuntimeAdapter] Deno runtime is required for exists()')\n }\n try {\n await deno.stat(path)\n return true\n } catch {\n return false\n }\n },\n async stat(path) {\n const deno = (globalThis as any).Deno\n if (!deno?.stat) {\n throw new Error('[RuntimeAdapter] Deno runtime is required for stat()')\n }\n const stats = await deno.stat(path)\n return { size: stats.size }\n },\n async deleteFile(path) {\n const deno = (globalThis as any).Deno\n if (!deno?.remove) {\n throw new Error('[RuntimeAdapter] Deno runtime is required for deleteFile()')\n }\n try {\n await deno.remove(path)\n } catch {\n // Ignore if not found\n }\n },\n serve(_config) {\n throw new Error('[RuntimeAdapter] Bun runtime is required for Bun.serve()')\n },\n }\n}\n",
|
|
26
|
+
"/**\n * Node.js runtime adapter implementation.\n *\n * @module runtime/adapter-node\n * @since 3.2.0\n */\n\nimport { toUint8Array } from './detection'\nimport type { RuntimeAdapter, RuntimeSpawnOptions } from './types'\n\n/**\n * Create a RuntimeAdapter for the Node.js runtime.\n * @internal\n */\nexport function createNodeAdapter(): RuntimeAdapter {\n return {\n kind: 'node',\n spawn(command, options = {}) {\n const [cmd, ...args] = command\n if (!cmd) {\n throw new Error('[RuntimeAdapter] spawn() requires a command')\n }\n\n // biome-ignore lint: node context\n const childProcess = require('node:child_process')\n // biome-ignore lint: node context\n const stream = require('node:stream')\n\n const stdioMap = (value: RuntimeSpawnOptions['stdout']) => {\n if (value === 'inherit') {\n return 'inherit'\n }\n if (value === 'ignore') {\n return 'ignore'\n }\n return 'pipe'\n }\n const stdinMap = (value: RuntimeSpawnOptions['stdin']) => {\n if (value === 'inherit') {\n return 'inherit'\n }\n if (value === 'ignore') {\n return 'ignore'\n }\n return 'pipe'\n }\n\n const child = childProcess.spawn(cmd, args, {\n cwd: options.cwd,\n env: options.env as Record<string, string>,\n stdio: [stdinMap(options.stdin), stdioMap(options.stdout), stdioMap(options.stderr)],\n }) as import('node:child_process').ChildProcess\n\n const toWeb = (streamReadable: NodeJS.ReadableStream | null) => {\n if (!streamReadable) {\n return null\n }\n const maybeWeb = streamReadable as unknown as ReadableStream<Uint8Array>\n if (typeof (maybeWeb as any).getReader === 'function') {\n return maybeWeb\n }\n return stream.Readable.toWeb(streamReadable as any) as unknown as ReadableStream<Uint8Array>\n }\n\n let timeoutHandle: NodeJS.Timeout | undefined\n let abortListener: (() => void) | undefined\n\n const exitedWithTimeout = new Promise<number>((resolve, reject) => {\n if (options.timeout) {\n timeoutHandle = setTimeout(() => {\n child.kill('SIGTERM')\n reject(new Error('[RuntimeAdapter] Process timeout'))\n }, options.timeout)\n }\n\n if (options.signal) {\n abortListener = () => {\n child.kill('SIGTERM')\n reject(new DOMException('Aborted', 'AbortError'))\n }\n options.signal.addEventListener('abort', abortListener)\n }\n\n child.on('error', reject)\n child.on('exit', (code) => {\n if (timeoutHandle) {\n clearTimeout(timeoutHandle)\n }\n if (abortListener && options.signal) {\n options.signal.removeEventListener('abort', abortListener)\n }\n resolve(code ?? 0)\n })\n })\n\n return {\n exited: exitedWithTimeout,\n stdout: toWeb(child.stdout),\n stderr: toWeb(child.stderr),\n kill: (signal?: string | number) => child.kill(signal as NodeJS.Signals | number),\n unref: () => {\n child.unref()\n },\n resourceUsage: async () => {\n try {\n const usage = (child as any).resourceUsage?.()\n if (!usage) {\n return undefined\n }\n return {\n cpuTime:\n usage.user && usage.system ? { user: usage.user, system: usage.system } : undefined,\n maxRSS: usage.maxRss,\n }\n } catch {\n return undefined\n }\n },\n }\n },\n async spawnAndCollect(command, options = {}) {\n const [cmd, ...args] = command\n if (!cmd) {\n throw new Error('[RuntimeAdapter] spawn() requires a command')\n }\n\n // biome-ignore lint: node context\n const childProcess = require('node:child_process')\n\n return new Promise((resolve, reject) => {\n let timedOut = false\n let timeoutHandle: NodeJS.Timeout | undefined\n let abortListener: (() => void) | undefined\n\n const child = childProcess.spawn(cmd, args, {\n cwd: options.cwd,\n env: options.env as Record<string, string>,\n stdio: ['pipe', 'pipe', 'pipe'],\n }) as import('node:child_process').ChildProcess\n\n const stdoutChunks: Buffer[] = []\n const stderrChunks: Buffer[] = []\n\n child.stdout?.on('data', (chunk) => stdoutChunks.push(chunk))\n child.stderr?.on('data', (chunk) => stderrChunks.push(chunk))\n\n if (options.timeout) {\n timeoutHandle = setTimeout(() => {\n timedOut = true\n child.kill('SIGTERM')\n }, options.timeout)\n }\n\n if (options.signal) {\n abortListener = () => {\n child.kill('SIGTERM')\n }\n options.signal.addEventListener('abort', abortListener)\n }\n\n child.on('error', reject)\n child.on('exit', (code) => {\n if (timeoutHandle) {\n clearTimeout(timeoutHandle)\n }\n if (abortListener && options.signal) {\n options.signal.removeEventListener('abort', abortListener)\n }\n\n const stdout = Buffer.concat(stdoutChunks).toString('utf-8')\n const stderr = Buffer.concat(stderrChunks).toString('utf-8')\n\n resolve({\n exitCode: code ?? 0,\n stdout,\n stderr,\n success: (code ?? 0) === 0 && !timedOut,\n timedOut,\n })\n })\n })\n },\n spawnSync(command, options = {}) {\n const [cmd, ...args] = command\n if (!cmd) {\n throw new Error('[RuntimeAdapter] spawn() requires a command')\n }\n\n // biome-ignore lint: node context\n const childProcess = require('node:child_process')\n\n const result = childProcess.spawnSync(cmd, args, {\n cwd: options.cwd,\n env: options.env as Record<string, string>,\n encoding: 'utf-8',\n })\n\n return {\n exitCode: result.status ?? 0,\n stdout: (result.stdout ?? '') as string,\n stderr: (result.stderr ?? '') as string,\n success: (result.status ?? 0) === 0,\n timedOut: (result.error as any)?.code === 'ERR_CHILD_PROCESS_STDIO_MAXBUFFER',\n }\n },\n async writeFile(path, data) {\n const fs = await import('node:fs/promises')\n const payload = await toUint8Array(data)\n await fs.writeFile(path, payload)\n },\n async appendFile(path, data) {\n const fs = await import('node:fs/promises')\n const payload = await toUint8Array(data)\n await fs.appendFile(path, payload)\n },\n createFileSink(path) {\n // biome-ignore lint: node context\n const fs = require('node:fs')\n const stream = fs.createWriteStream(path, { flags: 'a' })\n return {\n write(data) {\n stream.write(data)\n },\n async flush() {\n return new Promise((resolve, reject) => {\n stream.once('error', reject)\n stream.write('', (err: Error | null) => {\n if (err) {\n reject(err)\n } else {\n resolve()\n }\n })\n })\n },\n async end() {\n return new Promise((resolve, reject) => {\n stream.once('error', reject)\n stream.end(() => resolve())\n })\n },\n }\n },\n async readFile(path) {\n const fs = await import('node:fs/promises')\n const buffer = await fs.readFile(path)\n return new Uint8Array(buffer)\n },\n async readFileAsBlob(path) {\n const buffer = await this.readFile(path)\n return new Blob([buffer as unknown as BlobPart])\n },\n async exists(path) {\n const fs = await import('node:fs/promises')\n try {\n await fs.access(path)\n return true\n } catch {\n return false\n }\n },\n async stat(path) {\n const fs = await import('node:fs/promises')\n const stats = await fs.stat(path)\n return { size: stats.size }\n },\n async deleteFile(path) {\n const fs = await import('node:fs/promises')\n try {\n await fs.unlink(path)\n } catch {\n // Ignore if not found\n }\n },\n async mkdir(path, options = {}) {\n const fs = await import('node:fs/promises')\n await fs.mkdir(path, options)\n },\n async readDir(path) {\n const fs = await import('node:fs/promises')\n const entries = await fs.readdir(path, { withFileTypes: true })\n return entries.map((entry) => ({\n name: entry.name,\n isFile: entry.isFile(),\n isDirectory: entry.isDirectory(),\n }))\n },\n async removeRecursive(path) {\n const fs = await import('node:fs/promises')\n await fs.rm(path, { recursive: true, force: true })\n },\n serve(_config) {\n throw new Error('[RuntimeAdapter] Bun runtime is required for Bun.serve()')\n },\n }\n}\n",
|
|
27
|
+
"/**\n * Unknown runtime adapter implementation (fallback).\n *\n * @module runtime/adapter-unknown\n * @since 3.2.0\n */\n\nimport type { RuntimeAdapter } from './types'\n\n/**\n * Create a RuntimeAdapter for unsupported runtimes.\n * All methods throw with descriptive error messages.\n * @internal\n */\nexport function createUnknownAdapter(): RuntimeAdapter {\n return {\n kind: 'unknown',\n spawn() {\n throw new Error('[RuntimeAdapter] Unsupported runtime for spawn()')\n },\n async spawnAndCollect() {\n throw new Error('[RuntimeAdapter] Unsupported runtime for spawnAndCollect()')\n },\n spawnSync() {\n throw new Error('[RuntimeAdapter] Unsupported runtime for spawnSync()')\n },\n async writeFile() {\n throw new Error('[RuntimeAdapter] Unsupported runtime for writeFile()')\n },\n async readFile() {\n throw new Error('[RuntimeAdapter] Unsupported runtime for readFile()')\n },\n async readFileAsBlob() {\n throw new Error('[RuntimeAdapter] Unsupported runtime for readFileAsBlob()')\n },\n async exists() {\n throw new Error('[RuntimeAdapter] Unsupported runtime for exists()')\n },\n async stat() {\n throw new Error('[RuntimeAdapter] Unsupported runtime for stat()')\n },\n async deleteFile() {\n throw new Error('[RuntimeAdapter] Unsupported runtime for deleteFile()')\n },\n serve() {\n throw new Error('[RuntimeAdapter] Unsupported runtime for serve()')\n },\n }\n}\n",
|
|
28
|
+
"/**\n * Runtime archive adapter implementations.\n *\n * @module runtime/archive\n * @since 3.2.0\n */\n\nimport { getRuntimeKind } from './detection'\nimport type {\n ArchiveCreateOptions,\n ArchiveFromDirectoryOptions,\n RuntimeArchiveAdapter,\n} from './types'\n\n// ============ Bun Archive Adapter ============\n\n/**\n * 建立 Bun 原生封裝 adapter (使用 Bun.Tar)\n * @internal\n */\nfunction createBunArchiveAdapter(): RuntimeArchiveAdapter {\n return {\n async create(entries, _options = {}) {\n // Use globalThis to avoid Vite import errors\n const B = (globalThis as any).Bun\n if (!B) {\n throw new Error('[RuntimeArchiveAdapter] Bun global not found')\n }\n if (!B.Tar) {\n throw new Error('[RuntimeArchiveAdapter] Bun.Tarball is not available')\n }\n\n const tar = new B.Tar()\n for (const [path, data] of Object.entries(entries)) {\n // Handle different data types\n let content: Uint8Array\n if (data instanceof Uint8Array) {\n content = data\n } else if (typeof data === 'string') {\n content = new TextEncoder().encode(data)\n } else if (data instanceof ArrayBuffer) {\n content = new Uint8Array(data)\n } else if (data instanceof Blob) {\n content = new Uint8Array(await data.arrayBuffer())\n } else {\n throw new Error(`[RuntimeArchiveAdapter] Unsupported data type for path: ${path}`)\n }\n tar.append(path, content)\n }\n return new Uint8Array(await new Response(tar.out).arrayBuffer())\n },\n\n async extract(_data, _targetDir, _options = {}) {\n const B = (globalThis as any).Bun\n if (!B?.Tar) {\n throw new Error('[RuntimeArchiveAdapter] Bun.Tarball is not available')\n }\n throw new Error('[RuntimeArchiveAdapter] Tar extraction not yet implemented via Bun.Tar')\n },\n\n async list(_data, _glob?) {\n const B = (globalThis as any).Bun\n if (!B?.Tar) {\n throw new Error('[RuntimeArchiveAdapter] Bun.Tarball is not available')\n }\n throw new Error('[RuntimeArchiveAdapter] Tar listing not yet implemented via Bun.Tar')\n },\n\n async readFile(_data, _filePath) {\n const B = (globalThis as any).Bun\n if (!B?.Tar) {\n throw new Error('[RuntimeArchiveAdapter] Bun.Tarball is not available')\n }\n throw new Error('[RuntimeArchiveAdapter] Tar readFile not yet implemented via Bun.Tar')\n },\n }\n}\n\n// ============ Node.js Archive Adapter ============\n\n/**\n * 建立 Node.js 封裝 adapter\n * @internal\n */\nfunction createNodeArchiveAdapter(): RuntimeArchiveAdapter {\n return {\n async create(_entries, _options = {}) {\n throw new Error(\n '[RuntimeArchiveAdapter] Node.js archive creation requires external dependencies'\n )\n },\n async extract(_data, _targetDir, _options = {}) {\n throw new Error(\n '[RuntimeArchiveAdapter] Node.js archive extraction requires external dependencies'\n )\n },\n async list(_data, _glob?) {\n throw new Error(\n '[RuntimeArchiveAdapter] Node.js archive listing requires external dependencies'\n )\n },\n async readFile(_data, _filePath) {\n throw new Error(\n '[RuntimeArchiveAdapter] Node.js archive readFile requires external dependencies'\n )\n },\n }\n}\n\n// ============ Unsupported Archive Adapter ============\n\n/**\n * 建立不支援的封裝 adapter\n * @internal\n */\nfunction createUnsupportedArchiveAdapter(message: string): RuntimeArchiveAdapter {\n return {\n async create() {\n throw new Error(message)\n },\n async extract() {\n throw new Error(message)\n },\n async list() {\n throw new Error(message)\n },\n async readFile() {\n throw new Error(message)\n },\n }\n}\n\n// ============ Singleton ============\n\nlet archiveAdapter: RuntimeArchiveAdapter | null = null\n\n/**\n * 取得封裝操作 adapter\n * @public\n */\nexport function getArchiveAdapter(): RuntimeArchiveAdapter {\n if (archiveAdapter) {\n return archiveAdapter\n }\n const kind = getRuntimeKind()\n switch (kind) {\n case 'bun':\n archiveAdapter = createBunArchiveAdapter()\n break\n case 'node':\n archiveAdapter = createNodeArchiveAdapter()\n break\n default:\n archiveAdapter = createUnsupportedArchiveAdapter(\n `[RuntimeArchiveAdapter] Archive operations unavailable in unknown runtime: ${kind}`\n )\n }\n return archiveAdapter\n}\n\n/**\n * 使用 Node.js readdir 遞迴掃描目錄(Fallback 實作)\n * @internal\n */\nasync function scanDirectoryNode(dir: string): Promise<Record<string, Uint8Array>> {\n // biome-ignore lint/security/noGlobalEval: hide from Vite\n const fs = await eval('import(\"node:fs/promises\")')\n // biome-ignore lint/security/noGlobalEval: hide from Vite\n const path = await eval('import(\"node:path\")')\n\n const { readdir, readFile, stat } = fs\n const { join } = path\n const entries: Record<string, Uint8Array> = {}\n\n async function walk(currentDir: string, base: string) {\n const files = await readdir(currentDir)\n for (const file of files) {\n const fullPath = join(currentDir, file)\n const relativePath = join(base, file)\n const s = await stat(fullPath)\n\n if (s.isDirectory()) {\n await walk(fullPath, relativePath)\n } else {\n entries[relativePath] = new Uint8Array(await readFile(fullPath))\n }\n }\n }\n\n await walk(dir, '')\n return entries\n}\n\n/**\n * 將目錄封裝為歸檔檔案\n * @public\n */\nexport async function archiveFromDirectory(\n dirPath: string,\n archivePath: string,\n options: ArchiveFromDirectoryOptions = {}\n): Promise<Uint8Array> {\n // Validate archivePath\n if (!archivePath) {\n throw new Error('archivePath cannot be empty')\n }\n if (!archivePath.endsWith('.tar.gz') && !archivePath.endsWith('.tgz')) {\n throw new Error('archivePath must end with .tar.gz or .tgz')\n }\n\n const kind = getRuntimeKind()\n const createOptions: ArchiveCreateOptions = {\n compress: options.compress === 'gzip' ? 'gzip' : undefined,\n level: options.level,\n }\n\n let entries: Record<string, Uint8Array> = {}\n\n if (kind === 'bun') {\n const B = (globalThis as any).Bun\n const glob = new B.Glob(options.glob ?? '**/*')\n for await (const file of glob.scan(dirPath)) {\n // biome-ignore lint/security/noGlobalEval: hide from Vite\n const pathMod = await eval('import(\"node:path\")')\n const filePath = pathMod.join(dirPath, file)\n entries[file] = new Uint8Array(await B.file(filePath).arrayBuffer())\n }\n } else {\n entries = await scanDirectoryNode(dirPath)\n }\n\n const adapter = getArchiveAdapter()\n const archiveData = await adapter.create(entries, createOptions)\n\n // biome-ignore lint/security/noGlobalEval: hide from Vite\n const fs = await eval('import(\"node:fs/promises\")')\n await fs.writeFile(archivePath, archiveData)\n\n return archiveData\n}\n",
|
|
29
|
+
"/**\n * Runtime compression adapter implementations.\n *\n * Provides unified gzip/deflate compression across Bun and Node.js runtimes.\n * Bun uses native C++ implementations (Bun.gzipSync etc.) for 2-5x better performance.\n * Node.js falls back to the standard node:zlib module.\n *\n * @module runtime/compression\n * @since 3.2.0\n */\n\nimport { getRuntimeKind } from './detection'\nimport type { RuntimeCompressionAdapter } from './types'\n\n// ============ Bun Compression Adapter ============\n\n/**\n * 建立 Bun 原生壓縮 adapter\n * 使用 Bun.gzipSync / Bun.gunzipSync / Bun.deflateSync / Bun.inflateSync\n * @internal\n */\nfunction createBunCompressionAdapter(): RuntimeCompressionAdapter {\n // Bun 壓縮等級型別\n type BunLevel = 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | -1\n\n // Bun API 要求 Uint8Array<ArrayBuffer>,確保型別相容\n function toBunBuffer(data: Uint8Array): Uint8Array<ArrayBuffer> {\n const buf = new ArrayBuffer(data.byteLength)\n const view = new Uint8Array(buf)\n view.set(data)\n return view\n }\n\n function toLevel(level: number): BunLevel {\n return Math.max(-1, Math.min(12, level)) as BunLevel\n }\n\n return {\n gzipSync(data, options = {}) {\n return Bun.gzipSync(toBunBuffer(data), {\n level: toLevel(options.level ?? 6),\n })\n },\n\n gunzipSync(data) {\n return Bun.gunzipSync(toBunBuffer(data))\n },\n\n deflateSync(data, options = {}) {\n return Bun.deflateSync(toBunBuffer(data), {\n level: toLevel(options.level ?? 6),\n })\n },\n\n inflateSync(data) {\n return Bun.inflateSync(toBunBuffer(data))\n },\n\n async gzip(data, options = {}) {\n return Bun.gzipSync(toBunBuffer(data), {\n level: toLevel(options.level ?? 6),\n })\n },\n\n async gunzip(data) {\n return Bun.gunzipSync(toBunBuffer(data))\n },\n }\n}\n\n// ============ Node.js Compression Adapter ============\n\n/**\n * 建立 Node.js 壓縮 adapter(使用 node:zlib)\n * @internal\n */\nfunction createNodeCompressionAdapter(): RuntimeCompressionAdapter {\n return {\n gzipSync(data, options = {}) {\n // biome-ignore lint: node context\n const zlib = require('node:zlib')\n const result = zlib.gzipSync(Buffer.from(data), {\n level: options.level ?? 6,\n })\n return new Uint8Array(result)\n },\n\n gunzipSync(data) {\n // biome-ignore lint: node context\n const zlib = require('node:zlib')\n const result = zlib.gunzipSync(Buffer.from(data))\n return new Uint8Array(result)\n },\n\n deflateSync(data, options = {}) {\n // biome-ignore lint: node context\n const zlib = require('node:zlib')\n const result = zlib.deflateSync(Buffer.from(data), {\n level: options.level ?? 6,\n })\n return new Uint8Array(result)\n },\n\n inflateSync(data) {\n // biome-ignore lint: node context\n const zlib = require('node:zlib')\n const result = zlib.inflateSync(Buffer.from(data))\n return new Uint8Array(result)\n },\n\n async gzip(data, options = {}) {\n const zlib = await import('node:zlib')\n const { promisify } = await import('node:util')\n const gzipAsync = promisify(zlib.gzip)\n const result = await gzipAsync(Buffer.from(data), {\n level: options.level ?? 6,\n })\n return new Uint8Array(result)\n },\n\n async gunzip(data) {\n const zlib = await import('node:zlib')\n const { promisify } = await import('node:util')\n const gunzipAsync = promisify(zlib.gunzip)\n const result = await gunzipAsync(Buffer.from(data))\n return new Uint8Array(result)\n },\n }\n}\n\n// ============ Unsupported Compression Adapter ============\n\n/**\n * 建立不支援的壓縮 adapter(Deno / Unknown runtime 共用)\n * @internal\n */\nfunction createUnsupportedCompressionAdapter(message: string): RuntimeCompressionAdapter {\n return {\n gzipSync() {\n throw new Error(message)\n },\n gunzipSync() {\n throw new Error(message)\n },\n deflateSync() {\n throw new Error(message)\n },\n inflateSync() {\n throw new Error(message)\n },\n async gzip() {\n throw new Error(message)\n },\n async gunzip() {\n throw new Error(message)\n },\n }\n}\n\n// ============ Singleton ============\n\nlet compressionAdapter: RuntimeCompressionAdapter | null = null\n\n/**\n * 取得壓縮操作 adapter(依運行時自動選擇最佳實作)\n *\n * - Bun: 使用原生 C++ 壓縮(2-5x 更快)\n * - Node.js: 使用 node:zlib\n * - Deno/Unknown: 拋出錯誤\n *\n * @public\n */\nexport function getCompressionAdapter(): RuntimeCompressionAdapter {\n if (compressionAdapter) {\n return compressionAdapter\n }\n const kind = getRuntimeKind()\n switch (kind) {\n case 'bun':\n compressionAdapter = createBunCompressionAdapter()\n break\n case 'node':\n compressionAdapter = createNodeCompressionAdapter()\n break\n case 'deno':\n compressionAdapter = createUnsupportedCompressionAdapter(\n '[RuntimeCompressionAdapter] Deno compression support not yet implemented'\n )\n break\n default:\n compressionAdapter = createUnsupportedCompressionAdapter(\n '[RuntimeCompressionAdapter] Compression unavailable in unknown runtime'\n )\n }\n return compressionAdapter\n}\n",
|
|
30
|
+
"/**\n * Runtime markdown adapter implementations.\n *\n * Provides unified Markdown -> HTML rendering across Bun and Node.js runtimes.\n * Bun uses the native C++ Markdown parser (Bun.markdown) for 10-100x better performance.\n * Node.js falls back to the `marked` library (lazy-loaded optional dependency).\n *\n * @module runtime/markdown\n * @since 3.3.0\n */\n\nimport { getRuntimeKind } from './detection'\nimport type {\n MarkdownRenderCallbacks,\n MarkdownRenderOptions,\n RuntimeMarkdownAdapter,\n} from './types'\n\n// ============ Default HTML Render Callbacks ============\n\n/**\n * 建立預設的 HTML 渲染回調集合。\n *\n * 提供完整的 HTML 元素生成回調,產生與 `html()` 相同的 HTML 輸出。\n * 使用者可透過覆寫個別回調來自訂特定元素的渲染行為(例如 XSS 防護、\n * 自訂 CSS class 等),同時保留其他元素的預設 HTML 渲染。\n *\n * @param overrides - 要覆寫的回調。未指定的回調使用預設 HTML 渲染。\n * @returns 完整的 MarkdownRenderCallbacks 物件\n *\n * @example\n * ```typescript\n * // 自訂 link 和 html 渲染(XSS 防護)\n * const callbacks = createHtmlRenderCallbacks({\n * html: (raw) => escapeHtml(raw),\n * link: (content, { href }) => `<a href=\"${href}\" rel=\"noopener\">${content}</a>`,\n * })\n * const result = adapter.render(markdown, callbacks)\n * ```\n *\n * @public\n */\nexport function createHtmlRenderCallbacks(\n overrides: Partial<MarkdownRenderCallbacks> = {}\n): MarkdownRenderCallbacks {\n const defaults: MarkdownRenderCallbacks = {\n heading: (content, opts) => `<h${opts.level}>${content}</h${opts.level}>\\n`,\n paragraph: (content) => `<p>${content}</p>\\n`,\n code: (code, opts) => {\n const lang = opts.language ? ` class=\"language-${opts.language}\"` : ''\n return `<pre><code${lang}>${code}</code></pre>\\n`\n },\n codespan: (code) => `<code>${code}</code>`,\n strong: (content) => `<strong>${content}</strong>`,\n em: (content) => `<em>${content}</em>`,\n del: (content) => `<del>${content}</del>`,\n link: (content, opts) => {\n const titleAttr = opts.title ? ` title=\"${opts.title}\"` : ''\n return `<a href=\"${opts.href}\"${titleAttr}>${content}</a>`\n },\n image: (alt, opts) => {\n const titleAttr = opts.title ? ` title=\"${opts.title}\"` : ''\n return `<img src=\"${opts.src}\" alt=\"${alt}\"${titleAttr} />`\n },\n list: (content, opts) => {\n if (opts.ordered) {\n const start = opts.start && opts.start !== 1 ? ` start=\"${opts.start}\"` : ''\n return `<ol${start}>\\n${content}</ol>\\n`\n }\n return `<ul>\\n${content}</ul>\\n`\n },\n listItem: (content) => `<li>${content}</li>\\n`,\n blockquote: (content) => `<blockquote>\\n${content}</blockquote>\\n`,\n table: (content) => `<table>\\n${content}</table>\\n`,\n hr: () => '<hr />\\n',\n html: (rawHtml) => rawHtml,\n }\n\n return { ...defaults, ...overrides }\n}\n\n// ============ Callback Adapter ============\n\n/**\n * 將 MarkdownRenderCallbacks 轉換為 Bun.markdown.render 的 RenderCallbacks 格式。\n *\n * 我們的介面設計與 Bun 原生回調高度相似但有微小差異(如 code 的 meta 參數可選性),\n * 此函式負責建立轉接層以確保型別安全。\n *\n * @internal\n */\nfunction toBunRenderCallbacks(\n callbacks: MarkdownRenderCallbacks\n): Record<string, (...args: unknown[]) => string | null | undefined> {\n const result: Record<string, (...args: unknown[]) => string | null | undefined> = {}\n\n if (callbacks.heading) {\n const cb = callbacks.heading\n result.heading = (children: unknown, meta: unknown) => {\n const m = meta as { level: number }\n return cb(children as string, { level: m.level })\n }\n }\n\n if (callbacks.link) {\n const cb = callbacks.link\n result.link = (children: unknown, meta: unknown) => {\n const m = meta as { href: string; title?: string }\n return cb(children as string, { href: m.href, title: m.title })\n }\n }\n\n if (callbacks.code) {\n const cb = callbacks.code\n result.code = (children: unknown, meta?: unknown) => {\n const m = (meta as { language?: string } | undefined) ?? {}\n return cb(children as string, { language: m.language })\n }\n }\n\n if (callbacks.codespan) {\n const cb = callbacks.codespan\n result.codespan = (children: unknown) => cb(children as string)\n }\n\n if (callbacks.image) {\n const cb = callbacks.image\n result.image = (alt: unknown, meta: unknown) => {\n const m = meta as { src: string; title?: string }\n return cb(alt as string, { src: m.src, title: m.title })\n }\n }\n\n if (callbacks.html) {\n const cb = callbacks.html\n result.html = (rawHtml: unknown) => cb(rawHtml as string)\n }\n\n if (callbacks.paragraph) {\n const cb = callbacks.paragraph\n result.paragraph = (children: unknown) => cb(children as string)\n }\n\n if (callbacks.strong) {\n const cb = callbacks.strong\n result.strong = (children: unknown) => cb(children as string)\n }\n\n if (callbacks.em) {\n const cb = callbacks.em\n result.em = (children: unknown) => cb(children as string)\n }\n\n if (callbacks.del) {\n const cb = callbacks.del\n result.del = (children: unknown) => cb(children as string)\n }\n\n if (callbacks.list) {\n const cb = callbacks.list\n result.list = (children: unknown, meta: unknown) => {\n const m = meta as { ordered: boolean; start?: number }\n return cb(children as string, { ordered: m.ordered, start: m.start })\n }\n }\n\n if (callbacks.listItem) {\n const cb = callbacks.listItem\n result.listItem = (children: unknown) => cb(children as string)\n }\n\n if (callbacks.blockquote) {\n const cb = callbacks.blockquote\n result.blockquote = (children: unknown) => cb(children as string)\n }\n\n if (callbacks.table) {\n const cb = callbacks.table\n result.table = (children: unknown) => cb(children as string)\n }\n\n if (callbacks.hr) {\n const cb = callbacks.hr\n result.hr = () => cb()\n }\n\n return result\n}\n\n// ============ Bun Markdown Adapter ============\n\n/**\n * 建立 Bun 原生 Markdown adapter(使用 Bun.markdown API)\n * @internal\n */\nfunction createBunMarkdownAdapter(): RuntimeMarkdownAdapter {\n // 驗證 Bun.markdown 是否可用\n if (\n typeof Bun === 'undefined' ||\n typeof Bun.markdown !== 'object' ||\n typeof Bun.markdown.html !== 'function'\n ) {\n throw new Error('[RuntimeMarkdownAdapter] Bun.markdown is not available in this Bun version')\n }\n\n const bunMd = Bun.markdown\n\n return {\n html(markdown: string, _options?: MarkdownRenderOptions): string {\n if (!markdown) {\n return ''\n }\n return bunMd.html(markdown)\n },\n\n render(markdown: string, callbacks?: MarkdownRenderCallbacks): string {\n if (!markdown) {\n return ''\n }\n if (!callbacks) {\n return bunMd.render(markdown)\n }\n // 透過轉接層將我們的回調格式映射到 Bun 原生格式\n const bunCallbacks = toBunRenderCallbacks(callbacks)\n return bunMd.render(markdown, bunCallbacks as unknown as Parameters<typeof bunMd.render>[1])\n },\n\n react(markdown: string): unknown | null {\n if (!markdown) {\n return null\n }\n return bunMd.react(markdown)\n },\n\n get isNative() {\n return true\n },\n }\n}\n\n// ============ Fallback Markdown Adapter (Node.js) ============\n\n/**\n * 建立 Node.js Fallback Markdown adapter。\n * 使用 marked 函式庫進行延遲載入(optional dependency)。\n *\n * marked 不在 core 的硬性依賴中,僅在 Node.js 環境下需要。\n * 若 marked 不可用,html() 和 render() 會拋出錯誤。\n *\n * @internal\n */\nfunction createFallbackMarkdownAdapter(): RuntimeMarkdownAdapter {\n // 延遲載入 marked(避免型別依賴問題)\n // biome-ignore lint/suspicious/noExplicitAny: marked 為 optional dependency,無法保證型別\n let markedModule: any | null = null\n\n function loadMarkedSync(): {\n marked: { parse: (...args: unknown[]) => unknown }\n Renderer: new () => Record<string, unknown>\n } {\n if (!markedModule) {\n try {\n // eslint-disable-next-line @typescript-eslint/no-require-imports\n markedModule = require('marked')\n } catch {\n throw new Error(\n '[RuntimeMarkdownAdapter] Node.js markdown support requires \"marked\" package. ' +\n 'Install: npm install --save marked'\n )\n }\n }\n return markedModule as {\n marked: { parse: (...args: unknown[]) => unknown }\n Renderer: new () => Record<string, unknown>\n }\n }\n\n return {\n html(markdown: string, _options?: MarkdownRenderOptions): string {\n if (!markdown) {\n return ''\n }\n const { marked } = loadMarkedSync()\n const result = marked.parse(markdown, { async: false })\n return result as string\n },\n\n render(markdown: string, callbacks?: MarkdownRenderCallbacks): string {\n if (!markdown) {\n return ''\n }\n const { marked, Renderer } = loadMarkedSync()\n\n if (!callbacks) {\n // 無回調時回傳純文字(去除 HTML 標籤)\n const html = marked.parse(markdown, { async: false }) as string\n return html.replace(/<[^>]*>/g, '')\n }\n\n // 建立自訂 Renderer 並映射回調\n // biome-ignore lint/suspicious/noExplicitAny: marked Renderer 型別為 dynamic\n const renderer = new Renderer() as any\n\n if (callbacks.heading) {\n const cb = callbacks.heading\n renderer.heading = (text: string, level: number) => cb(text, { level })\n }\n\n if (callbacks.link) {\n const cb = callbacks.link\n renderer.link = (href: string, title: string | null, text: string) =>\n cb(text, { href, title: title ?? undefined })\n }\n\n if (callbacks.code) {\n const cb = callbacks.code\n renderer.code = (code: string, language: string | undefined) => cb(code, { language })\n }\n\n if (callbacks.image) {\n const cb = callbacks.image\n renderer.image = (href: string, title: string | null, text: string) =>\n cb(text, { src: href, title: title ?? undefined })\n }\n\n if (callbacks.html) {\n const cb = callbacks.html\n renderer.html = (html: string) => cb(html)\n }\n\n if (callbacks.paragraph) {\n const cb = callbacks.paragraph\n renderer.paragraph = (text: string) => cb(text)\n }\n\n if (callbacks.strong) {\n const cb = callbacks.strong\n renderer.strong = (text: string) => cb(text)\n }\n\n if (callbacks.em) {\n const cb = callbacks.em\n renderer.em = (text: string) => cb(text)\n }\n\n const result = marked.parse(markdown, { renderer, async: false })\n return result as string\n },\n\n react(_markdown: string): unknown | null {\n // Fallback 環境不支援 react()\n return null\n },\n\n get isNative() {\n return false\n },\n }\n}\n\n// ============ Unsupported Markdown Adapter ============\n\n/**\n * 建立不支援的 Markdown adapter(Deno / Unknown runtime 共用)\n * @internal\n */\nfunction createUnsupportedMarkdownAdapter(message: string): RuntimeMarkdownAdapter {\n return {\n html() {\n throw new Error(message)\n },\n render() {\n throw new Error(message)\n },\n react() {\n throw new Error(message)\n },\n get isNative() {\n return false\n },\n }\n}\n\n// ============ Singleton ============\n\nlet markdownAdapter: RuntimeMarkdownAdapter | null = null\n\n/**\n * 取得 Markdown 操作 adapter(依運行時自動選擇最佳實作)\n *\n * - Bun: 使用原生 C++ Markdown 解析器(10-100x 更快)\n * - Node.js: 使用 marked 函式庫(延遲載入 optional dependency)\n * - Deno/Unknown: 拋出錯誤\n *\n * @public\n */\nexport function getMarkdownAdapter(): RuntimeMarkdownAdapter {\n if (markdownAdapter) {\n return markdownAdapter\n }\n const kind = getRuntimeKind()\n switch (kind) {\n case 'bun':\n markdownAdapter = createBunMarkdownAdapter()\n break\n case 'node':\n markdownAdapter = createFallbackMarkdownAdapter()\n break\n case 'deno':\n markdownAdapter = createUnsupportedMarkdownAdapter(\n '[RuntimeMarkdownAdapter] Deno markdown support not yet implemented. Use Bun or Node.js runtime.'\n )\n break\n default:\n markdownAdapter = createUnsupportedMarkdownAdapter(\n `[RuntimeMarkdownAdapter] Markdown support unavailable in unknown runtime. Detected: ${kind}`\n )\n }\n return markdownAdapter\n}\n",
|
|
31
|
+
"/**\n * Deep equality comparison abstraction.\n *\n * Provides unified deep equality checking across Bun, Node.js, and Deno runtimes.\n * Uses Bun.deepEquals() when available, falls back to recursive comparison.\n *\n * @module runtime/deep-equals\n * @since 3.2.0\n */\n\nimport { getRuntimeKind } from './detection'\n\n/**\n * Options for deep equality comparison.\n * @public\n */\nexport interface DeepEqualsOptions {\n /**\n * When true, uses strict equality semantics:\n * - NaN !== NaN\n * - +0 !== -0\n *\n * When false (default), uses lenient semantics:\n * - NaN === NaN\n * - +0 === -0\n */\n strict?: boolean\n}\n\n/**\n * Function signature for deep equality comparison.\n * @public\n */\nexport type DeepEqualsFn = (a: unknown, b: unknown, options?: DeepEqualsOptions) => boolean\n\n/**\n * Creates a fallback deep equality function with cycle detection.\n *\n * Recursively compares objects, arrays, maps, sets, dates, and regexps.\n * Handles circular references using WeakSet tracking.\n *\n * @returns A deep equality comparison function.\n * @internal\n */\nfunction createFallbackDeepEquals(): DeepEqualsFn {\n return (a: unknown, b: unknown, options?: DeepEqualsOptions): boolean => {\n const visited = new WeakSet<object>()\n return deepEqualInternal(a, b, visited, options?.strict ?? false)\n }\n}\n\n/**\n * Internal recursive deep equality comparison.\n * @internal\n */\nfunction deepEqualInternal(\n a: unknown,\n b: unknown,\n visited: WeakSet<object>,\n strict: boolean\n): boolean {\n // 淺比較\n if (a === b) {\n return true\n }\n\n // null / undefined 檢查\n if (a == null || b == null) {\n return a === b\n }\n\n // 型別檢查\n const typeA = typeof a\n const typeB = typeof b\n if (typeA !== typeB) {\n return false\n }\n\n // 非物件型別(除了 object)\n if (typeA !== 'object') {\n // 特殊情況:NaN 比較\n // Bun.deepEquals 的行為:NaN === NaN 總是 true,無論嚴格模式\n if (Number.isNaN(a) && Number.isNaN(b)) {\n return true\n }\n // 其他情況:使用 === (包括 +0 !== -0)\n return a === b\n }\n\n // Date 比較\n if (a instanceof Date && b instanceof Date) {\n return a.getTime() === b.getTime()\n }\n if (a instanceof Date || b instanceof Date) {\n return false\n }\n\n // RegExp 比較\n if (a instanceof RegExp && b instanceof RegExp) {\n return a.source === b.source && a.flags === b.flags\n }\n if (a instanceof RegExp || b instanceof RegExp) {\n return false\n }\n\n // 循環引用偵測\n const aObj = a as object\n const bObj = b as object\n if (visited.has(aObj) || visited.has(bObj)) {\n return a === b\n }\n visited.add(aObj)\n visited.add(bObj)\n\n // Array 比較\n if (Array.isArray(a) && Array.isArray(b)) {\n if (a.length !== b.length) {\n return false\n }\n return a.every((val, idx) => deepEqualInternal(val, b[idx], visited, strict))\n }\n if (Array.isArray(a) || Array.isArray(b)) {\n return false\n }\n\n // Map 比較\n if (a instanceof Map && b instanceof Map) {\n if (a.size !== b.size) {\n return false\n }\n for (const [key, val] of a) {\n if (!b.has(key) || !deepEqualInternal(val, b.get(key), visited, strict)) {\n return false\n }\n }\n return true\n }\n if (a instanceof Map || b instanceof Map) {\n return false\n }\n\n // Set 比較\n if (a instanceof Set && b instanceof Set) {\n if (a.size !== b.size) {\n return false\n }\n const arrA = Array.from(a)\n const arrB = Array.from(b)\n return arrA.every((val) => arrB.some((bVal) => deepEqualInternal(val, bVal, visited, strict)))\n }\n if (a instanceof Set || b instanceof Set) {\n return false\n }\n\n // 普通物件比較\n const keysA = Object.keys(a as object)\n const keysB = Object.keys(b as object)\n\n if (keysA.length !== keysB.length) {\n return false\n }\n\n return keysA.every((key) => {\n const valA = (a as Record<string, unknown>)[key]\n const valB = (b as Record<string, unknown>)[key]\n return deepEqualInternal(valA, valB, visited, strict)\n })\n}\n\n/**\n * Singleton cache for deep equals function.\n * @internal\n */\nlet deepEqualsFn: DeepEqualsFn | null = null\n\n/**\n * Get the optimized deep equality comparison function for the current runtime.\n *\n * - **Bun**: Uses native Bun.deepEquals() for maximum performance (C++ optimized)\n * - **Node.js/Deno/Unknown**: Uses fallback recursive implementation with cycle detection\n *\n * Results are cached after first invocation for efficiency.\n *\n * @returns A function that performs deep equality comparison\n * @public\n *\n * @example\n * ```typescript\n * const deepEquals = getDeepEquals()\n *\n * const obj1 = { a: [1, 2], b: { c: 3 } }\n * const obj2 = { a: [1, 2], b: { c: 3 } }\n *\n * console.log(deepEquals(obj1, obj2)) // true\n *\n * // Strict mode for +0 / -0 and NaN\n * console.log(deepEquals(NaN, NaN)) // true (lenient)\n * console.log(deepEquals(NaN, NaN, { strict: true })) // false\n * ```\n */\nexport function getDeepEquals(): DeepEqualsFn {\n if (deepEqualsFn) {\n return deepEqualsFn\n }\n\n const kind = getRuntimeKind()\n if (kind === 'bun' && typeof Bun !== 'undefined') {\n deepEqualsFn = (a: unknown, b: unknown, options?: DeepEqualsOptions): boolean => {\n return Bun.deepEquals(a, b, options?.strict ?? false)\n }\n return deepEqualsFn\n }\n\n deepEqualsFn = createFallbackDeepEquals()\n return deepEqualsFn\n}\n",
|
|
32
|
+
"/**\n * Runtime HTML escape abstraction.\n *\n * Provides unified HTML escaping across Bun, Node.js, and Deno runtimes.\n * Bun uses the native C++ `Bun.escapeHTML()` for 10-100x better performance.\n * Node.js/Deno fall back to manual entity replacement.\n *\n * @module runtime/escape\n * @since 3.4.0\n */\n\nimport { getRuntimeKind } from './detection'\n\n/**\n * HTML escape function type.\n * @public\n */\nexport type EscapeHtmlFn = (value: string) => string\n\n/**\n * Creates a fallback HTML escape function using .replace() chains.\n *\n * Escapes 5 special characters: &, <, >, \", '\n * Each is replaced with its HTML entity equivalent.\n *\n * @returns HTML escape function\n * @internal\n */\nfunction createFallbackEscapeHtml(): EscapeHtmlFn {\n return (value: string): string => {\n if (!value) {\n return ''\n }\n return value\n .replace(/&/g, '&')\n .replace(/</g, '<')\n .replace(/>/g, '>')\n .replace(/\"/g, '"')\n .replace(/'/g, ''')\n }\n}\n\n// ============ Singleton ============\n\nlet escapeHtmlFn: EscapeHtmlFn | null = null\n\n/**\n * Get the HTML escape function (auto-selects best implementation based on runtime).\n *\n * - Bun: Uses native C++ `Bun.escapeHTML()` (10-100x faster, SIMD accelerated)\n * - Node.js/Deno/Unknown: Uses .replace() chain fallback with same behavior\n *\n * @returns HTML escape function\n * @public\n *\n * @example\n * ```typescript\n * import { getEscapeHtml } from '@gravito/core'\n *\n * const escapeHtml = getEscapeHtml()\n * const safe = escapeHtml('<script>alert(\"xss\")</script>')\n * // '<script>alert("xss")</script>'\n * ```\n */\nexport function getEscapeHtml(): EscapeHtmlFn {\n if (escapeHtmlFn) {\n return escapeHtmlFn\n }\n\n const kind = getRuntimeKind()\n if (kind === 'bun' && typeof Bun !== 'undefined' && typeof Bun.escapeHTML === 'function') {\n escapeHtmlFn = (value: string): string => {\n if (!value) {\n return ''\n }\n return Bun.escapeHTML(value)\n }\n } else {\n escapeHtmlFn = createFallbackEscapeHtml()\n }\n\n return escapeHtmlFn\n}\n",
|
|
33
|
+
"/**\n * Runtime abstraction module.\n *\n * Provides unified APIs for filesystem, process, archive, and compression\n * operations across Bun, Node.js, and Deno runtimes.\n *\n * @module runtime\n * @since 3.2.0\n */\n\nimport { createBunAdapter } from './adapter-bun'\nimport { createDenoAdapter } from './adapter-deno'\nimport { createNodeAdapter } from './adapter-node'\nimport { createUnknownAdapter } from './adapter-unknown'\nimport { getRuntimeKind } from './detection'\nimport type { RuntimeAdapter, RuntimePasswordAdapter, RuntimeSqliteDatabase } from './types'\n\n// ============ Type Exports ============\n\nexport type {\n ArchiveCreateOptions,\n ArchiveEntry,\n ArchiveExtractOptions,\n ArchiveFileInfo,\n ArchiveFromDirectoryOptions,\n CompressionOptions,\n MarkdownRenderCallbacks,\n MarkdownRenderOptions,\n OptionalRuntimeResourceUsage,\n RuntimeAdapter,\n RuntimeArchiveAdapter,\n RuntimeCompressionAdapter,\n RuntimeFileSink,\n RuntimeFileStat,\n RuntimeKind,\n RuntimeMarkdownAdapter,\n RuntimePasswordAdapter,\n RuntimeProcess,\n RuntimeProcessOutput,\n RuntimeResourceUsage,\n RuntimeServeConfig,\n RuntimeServer,\n RuntimeSpawnOptions,\n RuntimeSpawnSyncResult,\n RuntimeSqliteDatabase,\n RuntimeSqliteStatement,\n} from './types'\n\n// ============ Detection Exports ============\n\nexport { getRuntimeEnv, getRuntimeKind } from './detection'\n\n// ============ Archive Exports ============\n\nexport {\n archiveFromDirectory,\n getArchiveAdapter,\n} from './archive'\n\n// ============ Compression Exports ============\n\nexport { getCompressionAdapter } from './compression'\n\n// ============ Markdown Exports ============\n\nexport { createHtmlRenderCallbacks, getMarkdownAdapter } from './markdown'\n\n// ============ Deep Equals Exports ============\n\nexport type { DeepEqualsFn, DeepEqualsOptions } from './deep-equals'\nexport { getDeepEquals } from './deep-equals'\n\n// ============ Escape Exports ============\n\nexport type { EscapeHtmlFn } from './escape'\nexport { getEscapeHtml } from './escape'\n\n// ============ Runtime Adapter Singleton ============\n\nlet runtimeAdapter: RuntimeAdapter | null = null\n\n/**\n * Get the runtime abstraction adapter (Bun/Node/Deno).\n * @public\n */\nexport function getRuntimeAdapter(): RuntimeAdapter {\n if (runtimeAdapter) {\n return runtimeAdapter\n }\n\n const kind = getRuntimeKind()\n\n // Browser safety: return unknown adapter immediately if we're in a browser\n if (typeof window !== 'undefined' || (globalThis as any).process?.browser) {\n return (runtimeAdapter = createUnknownAdapter())\n }\n\n switch (kind) {\n case 'bun':\n runtimeAdapter = createBunAdapter()\n break\n case 'node':\n runtimeAdapter = createNodeAdapter()\n break\n case 'deno':\n runtimeAdapter = createDenoAdapter()\n break\n default:\n runtimeAdapter = createUnknownAdapter()\n }\n\n return runtimeAdapter\n}\n\n/**\n * Reset the runtime adapter (mainly for testing).\n * @internal\n */\nexport function resetRuntimeAdapter(): void {\n runtimeAdapter = null\n}\n\n// ============ Password Adapter Singleton ============\n\nlet passwordAdapter: RuntimePasswordAdapter | null = null\n\n/**\n * Get the password hashing adapter using native optimized implementations if available.\n * @public\n */\nexport function getPasswordAdapter(): RuntimePasswordAdapter {\n if (passwordAdapter) {\n return passwordAdapter\n }\n\n const kind = getRuntimeKind()\n const B = (globalThis as any).Bun\n\n if (kind === 'bun' && B) {\n passwordAdapter = {\n async hash(value, options) {\n if (options.algorithm === 'bcrypt') {\n return await B.password.hash(value, {\n algorithm: 'bcrypt',\n cost: options.cost ?? 12,\n })\n }\n return await B.password.hash(value, {\n algorithm: 'argon2id',\n ...(options.memoryCost !== undefined ? { memoryCost: options.memoryCost } : {}),\n ...(options.timeCost !== undefined ? { timeCost: options.timeCost } : {}),\n ...(options.parallelism !== undefined ? { parallelism: options.parallelism } : {}),\n })\n },\n async verify(value, hashed) {\n return await B.password.verify(value, hashed)\n },\n }\n return passwordAdapter\n }\n\n const message = '[RuntimeAdapter] Password hashing requires Bun runtime or a Node/Deno adapter'\n passwordAdapter = {\n async hash() {\n throw new Error(message)\n },\n async verify() {\n throw new Error(message)\n },\n }\n return passwordAdapter\n}\n\n// ============ SQLite Adapter ============\n\n/**\n * Create a SQLite database connection using runtime-native drivers.\n * @public\n */\nexport async function createSqliteDatabase(path: string): Promise<RuntimeSqliteDatabase> {\n const kind = getRuntimeKind()\n const B = (globalThis as any).Bun\n\n if (kind === 'bun' && B) {\n const sqlite = await import('bun:sqlite')\n const db = new sqlite.Database(path, { create: true })\n return db as RuntimeSqliteDatabase\n }\n\n throw new Error('[RuntimeAdapter] SQLite storage requires Bun runtime or a Node/Deno adapter')\n}\n\n/**\n * Convert various data types to Uint8Array.\n * @internal\n */\nexport async function toUint8Array(\n data: Blob | Buffer | string | ArrayBuffer | Uint8Array\n): Promise<Uint8Array> {\n if (data instanceof Uint8Array) {\n return data\n }\n if (typeof data === 'string') {\n return new TextEncoder().encode(data)\n }\n if (data instanceof ArrayBuffer) {\n return new Uint8Array(data)\n }\n if (typeof Buffer !== 'undefined' && data instanceof Buffer) {\n return new Uint8Array(data.buffer, data.byteOffset, data.byteLength)\n }\n if (data instanceof Blob) {\n return new Uint8Array(await data.arrayBuffer())\n }\n return new Uint8Array()\n}\n",
|
|
34
|
+
"/**\n * Configuration manager (ConfigManager)\n *\n * Unifies environment variables and application configuration access.\n */\nimport type { ZodSchema } from 'zod'\nimport { getRuntimeEnv } from './runtime'\n\n/**\n * ConfigManager - Central configuration store.\n * Supports loading from environment variables and initial objects.\n * @public\n */\nexport class ConfigManager {\n private config: Map<string, unknown> = new Map()\n private schema: ZodSchema | null = null\n\n constructor(initialConfig: Record<string, unknown> = {}) {\n // 1. Load initial config\n for (const [key, value] of Object.entries(initialConfig)) {\n this.config.set(key, value)\n }\n\n // 2. Auto-load runtime environment variables\n this.loadEnv()\n }\n\n /**\n * Load all environment variables from the active runtime.\n */\n private loadEnv() {\n const env = getRuntimeEnv()\n for (const key of Object.keys(env)) {\n if (env[key] !== undefined) {\n this.config.set(key, env[key])\n }\n }\n }\n\n /**\n * Get a configuration value (generic return type supported).\n * Supports dot notation for deep access (e.g. 'app.name').\n */\n get<T = unknown>(key: string, defaultValue?: T): T {\n // Check if key exists directly first\n if (this.config.has(key)) {\n return this.config.get(key) as T\n }\n\n // Handle dot notation\n if (key.includes('.')) {\n const parts = key.split('.')\n const rootKey = parts[0]\n if (rootKey) {\n let current: any = this.config.get(rootKey)\n\n if (current !== undefined) {\n for (let i = 1; i < parts.length; i++) {\n const part = parts[i]\n if (part && current && typeof current === 'object' && part in current) {\n current = current[part]\n } else {\n current = undefined\n break\n }\n }\n\n if (current !== undefined) {\n return current as T\n }\n }\n }\n }\n\n if (defaultValue !== undefined) {\n return defaultValue\n }\n throw new Error(`Config key '${key}' not found`)\n }\n\n /**\n * Set a configuration value.\n */\n set(key: string, value: unknown): void {\n this.config.set(key, value)\n }\n\n /**\n * Check whether a key exists.\n */\n has(key: string): boolean {\n return this.config.has(key)\n }\n\n /**\n * Define a Zod schema for configuration validation.\n *\n * @param schema - Zod schema for validation\n *\n * @example\n * ```typescript\n * config.defineSchema(z.object({\n * DATABASE_URL: z.string().url(),\n * PORT: z.number().default(3000),\n * }))\n * ```\n */\n defineSchema(schema: ZodSchema): void {\n this.schema = schema\n }\n\n /**\n * Validate configuration against the defined schema.\n *\n * Should be called during bootstrap to catch configuration errors early.\n *\n * @throws Error if validation fails with details about missing/invalid fields\n *\n * @example\n * ```typescript\n * try {\n * config.validate()\n * } catch (error) {\n * console.error('Config validation failed:', error.message)\n * process.exit(1)\n * }\n * ```\n */\n validate(): void {\n if (!this.schema) {\n return\n }\n\n const configObj = Object.fromEntries(this.config)\n const result = this.schema.safeParse(configObj)\n\n if (!result.success) {\n const errors = result.error.issues\n .map((issue) => {\n const path = Array.isArray(issue.path) ? issue.path.join('.') : String(issue.path)\n return `${path} - ${issue.message}`\n })\n .join('; ')\n throw new Error(`Configuration validation failed: ${errors}`)\n }\n }\n}\n",
|
|
35
|
+
"import type { ContentfulStatusCode } from '../http/types'\n\n/**\n * Options for creating a GravitoException\n * @public\n */\nexport interface ExceptionOptions {\n message?: string\n cause?: unknown\n i18nKey?: string // e.g. 'errors.validation.failed'\n i18nParams?: Record<string, string | number>\n}\n\n/**\n * Base exception class for consistent error handling.\n * @public\n */\nexport abstract class GravitoException extends Error {\n public readonly status: ContentfulStatusCode\n public readonly code: string\n public readonly i18nKey?: string\n public readonly i18nParams?: Record<string, string | number>\n\n constructor(status: number, code: string, options: ExceptionOptions = {}) {\n super(options.message)\n this.name = 'GravitoException'\n this.status = status as ContentfulStatusCode\n this.cause = options.cause\n this.code = code\n if (options.i18nKey) {\n this.i18nKey = options.i18nKey\n }\n if (options.i18nParams) {\n this.i18nParams = options.i18nParams\n }\n Object.setPrototypeOf(this, new.target.prototype)\n }\n\n // Helper for i18n\n getLocalizedMessage(\n t: (key: string, params?: Record<string, string | number>) => string\n ): string {\n if (this.i18nKey) {\n return t(this.i18nKey, this.i18nParams)\n }\n return this.message\n }\n}\n",
|
|
36
|
+
"import { type ExceptionOptions, GravitoException } from './GravitoException'\n\n/**\n * Abstract base class for internal framework / system-level errors.\n * Extend this for exceptions that represent an unexpected or unrecoverable framework state.\n * @public\n */\nexport abstract class SystemException extends GravitoException {\n constructor(status: number, code: string, options: ExceptionOptions = {}) {\n super(status, code, options)\n this.name = 'SystemException'\n Object.setPrototypeOf(this, new.target.prototype)\n }\n}\n",
|
|
37
|
+
"import type { ServiceKey } from '../Container'\nimport { SystemException } from './SystemException'\n\n/**\n * CircularDependencyException - Thrown when the container detects an infinite loop.\n *\n * @module @gravito/core\n */\nexport class CircularDependencyException extends SystemException {\n constructor(key: ServiceKey, stack: ServiceKey[]) {\n const path = [...stack, key].map(String).join(' -> ')\n super(500, 'system.circular_dependency', {\n message: `Circular dependency detected: ${path}`,\n })\n this.name = 'CircularDependencyException'\n }\n}\n",
|
|
38
|
+
"import type { RequestScopeManager } from './Container/RequestScopeManager'\nimport { AsyncLocalStorage } from './compat/async-local-storage'\nimport { CircularDependencyException } from './exceptions/CircularDependencyException'\n\n/**\n * Factory type for creating service instances\n */\nexport type Factory<T> = (container: Container) => T\n\n/**\n * AsyncLocalStorage 實例,用於隔離請求範圍\n * @internal\n */\nconst scopeStorage = new AsyncLocalStorage<RequestScopeManager>()\n\n/**\n * ServiceMap interface for type-safe IoC resolution.\n *\n * Extend this interface via module augmentation to get type inference:\n * @example\n * ```typescript\n * declare module '@gravito/core' {\n * interface ServiceMap {\n * logger: Logger\n * db: DatabaseConnection\n * }\n * }\n * ```\n */\n// biome-ignore lint/complexity/noBannedTypes: empty interface needed for module augmentation\nexport type ServiceMap = {}\n\n/**\n * ServiceKey represents the allowed keys for service resolution.\n * Includes keys from ServiceMap, generic strings, or symbols.\n */\nexport type ServiceKey = keyof ServiceMap | (string & {}) | symbol\n\ninterface Binding<T = unknown> {\n factory: Factory<T>\n shared: boolean // true for singleton\n scope?: 'transient' | 'singleton' | 'request' // scope type\n}\n\n/**\n * Container - Simple Dependency Injection Container.\n * Manages service bindings and singleton instances.\n * @public\n */\nexport class Container {\n private bindings = new Map<ServiceKey, Binding>()\n private instances = new Map<ServiceKey, unknown>()\n private resolutionStack: ServiceKey[] = []\n\n /**\n * Run a function within a request scope context.\n *\n * All service resolutions within the function will use the provided scope,\n * enabling request-scoped service instances to be properly isolated.\n *\n * @template T - The return type of the function.\n * @param scope - The RequestScopeManager for this request.\n * @param fn - The function to execute within the scope.\n * @returns The result of the function.\n *\n * @example\n * ```typescript\n * const scope = new RequestScopeManager()\n * const result = await Container.runWithScope(scope, async () => {\n * const service = container.make('requestScoped')\n * return service.doSomething()\n * })\n * ```\n */\n static runWithScope<T>(scope: RequestScopeManager, fn: () => T | Promise<T>): T | Promise<T> {\n return scopeStorage.run(scope, fn)\n }\n\n /**\n * Bind a service to the container.\n *\n * A new instance will be created by the factory function every time the\n * service is resolved from the container.\n *\n * @template T - The type of the service being bound.\n * @param key - The unique identifier for the service.\n * @param factory - The factory function that creates the service instance.\n *\n * @example\n * ```typescript\n * container.bind('logger', (c) => new ConsoleLogger());\n * ```\n */\n bind<T>(key: ServiceKey, factory: Factory<T>): void {\n this.bindings.set(key, {\n factory: factory as Factory<unknown>,\n shared: false,\n scope: 'transient',\n })\n }\n\n /**\n * Bind a shared service to the container (Singleton).\n *\n * The factory function will be called only once, and the same instance\n * will be returned on every subsequent resolution.\n *\n * @template T - The type of the service being bound.\n * @param key - The unique identifier for the service.\n * @param factory - The factory function that creates the service instance.\n *\n * @example\n * ```typescript\n * container.singleton('db', (c) => new DatabaseConnection());\n * ```\n */\n singleton<T>(key: ServiceKey, factory: Factory<T>): void {\n this.bindings.set(key, {\n factory: factory as Factory<unknown>,\n shared: true,\n scope: 'singleton',\n })\n }\n\n /**\n * Bind a request-scoped service to the container.\n *\n * A new instance will be created for each request and cached within that request.\n * The service is automatically cleaned up when the request ends.\n *\n * @template T - The type of the service being bound.\n * @param key - The unique identifier for the service.\n * @param factory - The factory function that creates the service instance.\n *\n * @example\n * ```typescript\n * container.scoped('requestCache', (c) => new RequestProductCache());\n * ```\n */\n scoped<T>(key: ServiceKey, factory: Factory<T>): void {\n this.bindings.set(key, {\n factory: factory as Factory<unknown>,\n shared: false,\n scope: 'request',\n })\n }\n\n /**\n * Register an existing instance as a shared service.\n *\n * @param key - The unique identifier for the service.\n * @param instance - The instance to register.\n */\n instance<T>(key: ServiceKey, instance: T): void {\n this.instances.set(key, instance)\n }\n\n /**\n * Check if a service is request-scoped.\n *\n * @param key - The service key to check.\n * @returns True if the service is request-scoped.\n */\n isRequestScoped(key: ServiceKey): boolean {\n return this.bindings.get(key)?.scope === 'request'\n }\n\n /**\n * Resolve a service instance from the container.\n *\n * Automatically handles singleton caching and factory execution.\n *\n * @template T - The expected type of the service.\n * @param key - The unique identifier for the service.\n * @returns The resolved service instance.\n * @throws Error if the service key is not found in the container.\n *\n * @example\n * ```typescript\n * const logger = container.make<Logger>('logger');\n * ```\n */\n make<K extends keyof ServiceMap>(key: K): ServiceMap[K]\n make<T>(key: ServiceKey): T\n make<T>(key: ServiceKey): T {\n // 1. Check shared instances\n if (this.instances.has(key)) {\n return this.instances.get(key) as T\n }\n\n // 2. Check for circular dependencies\n if (this.resolutionStack.includes(key)) {\n throw new CircularDependencyException(key, this.resolutionStack)\n }\n\n // 3. Check bindings\n const binding = this.bindings.get(key)\n if (!binding) {\n throw new Error(`Service '${String(key)}' not found in container`)\n }\n\n // 4. Handle request-scoped services\n if (binding.scope === 'request') {\n const scope = scopeStorage.getStore()\n if (scope) {\n // Use request scope manager to resolve\n return scope.resolve(key, () => binding.factory(this)) as T\n }\n // Fallback: if no scope context, create temporary instance with warning\n console.warn(\n `Request-scoped service '${String(key)}' resolved outside request context. ` +\n 'This instance will not be shared across the request. Consider wrapping with Container.runWithScope().'\n )\n }\n\n // 5. Resolve instance\n this.resolutionStack.push(key)\n\n try {\n const instance = binding.factory(this)\n\n // 6. Cache if shared\n if (binding.shared) {\n this.instances.set(key, instance)\n }\n\n return instance as T\n } finally {\n this.resolutionStack.pop()\n }\n }\n\n /**\n * Check if a service is bound or has an instance in the container.\n *\n * @param key - The service key to check.\n * @returns True if the service exists.\n */\n has(key: ServiceKey): boolean {\n return this.bindings.has(key) || this.instances.has(key)\n }\n\n /**\n * Flush all instances and bindings from the container.\n * Resets the container to an empty state.\n */\n flush(): void {\n this.bindings.clear()\n this.instances.clear()\n }\n\n /**\n * Forget a specific instance while keeping its binding.\n *\n * @param key - The service key to forget.\n */\n forget(key: ServiceKey): void {\n this.instances.delete(key)\n }\n}\n",
|
|
39
|
+
"import type { CommandKernel } from '../CommandKernel'\nimport type { QueueDashboard } from '../observability/QueueDashboard'\n\n/**\n * Parse command line arguments into structured format\n * Supports --key value and --flag patterns\n */\nfunction parseArgs(args: string[]): {\n subcommand: string\n positional: string[]\n flags: Record<string, string | boolean>\n} {\n const subcommand = args[0] ?? ''\n const positional: string[] = []\n const flags: Record<string, string | boolean> = {}\n\n let i = 1\n while (i < args.length) {\n const arg = args[i]\n\n if (arg?.startsWith('--')) {\n const key = arg.slice(2)\n const nextArg = args[i + 1]\n\n if (\n nextArg &&\n !nextArg.startsWith('--') &&\n !['status', 'workers', 'failed', 'monitor', 'export'].includes(nextArg)\n ) {\n flags[key] = nextArg\n i += 2\n } else {\n flags[key] = true\n i += 1\n }\n } else {\n positional.push(arg)\n i += 1\n }\n }\n\n return { subcommand, positional, flags }\n}\n\n/**\n * Format a duration in milliseconds to a human-readable string\n */\nfunction formatDuration(ms: number): string {\n if (ms < 1000) {\n return `${Math.round(ms)}ms`\n }\n if (ms < 60000) {\n return `${(ms / 1000).toFixed(1)}s`\n }\n if (ms < 3600000) {\n return `${Math.round(ms / 60000)}m ${Math.round((ms % 60000) / 1000)}s`\n }\n return `${Math.round(ms / 3600000)}h ${Math.round((ms % 3600000) / 60000)}m`\n}\n\n/**\n * Format a percentage with 2 decimal places\n */\nfunction formatPercent(value: number): string {\n return `${(value * 100).toFixed(2)}%`\n}\n\n/**\n * Format a number with thousands separator\n */\nfunction formatNumber(value: number): string {\n return value.toLocaleString()\n}\n\n/**\n * Display queue status overview\n */\nfunction handleStatus(dashboard: QueueDashboard): void {\n const snapshot = dashboard.getSnapshot()\n\n console.log('\\n╔════════════════════════════════════════════════════╗')\n console.log('║ Queue Status Overview ║')\n console.log('╚════════════════════════════════════════════════════╝\\n')\n\n // Queue depth\n console.log('Queue Depth:')\n console.log(` Total: ${formatNumber(snapshot.queue.depth.total)}`)\n console.log(` High: ${formatNumber(snapshot.queue.depth.high)}`)\n console.log(` Normal: ${formatNumber(snapshot.queue.depth.normal)}`)\n console.log(` Low: ${formatNumber(snapshot.queue.depth.low)}\\n`)\n\n // Backpressure\n console.log('Backpressure:')\n console.log(` State: ${snapshot.queue.backpressure.state}`)\n console.log(` Rejected: ${formatNumber(snapshot.queue.backpressure.rejectedCount)}`)\n console.log(` Degraded: ${formatNumber(snapshot.queue.backpressure.degradedCount)}`)\n console.log(` Enqueue Rate: ${snapshot.queue.backpressure.enqueueRate.toFixed(2)} events/s\\n`)\n\n // Workers\n console.log('Workers:')\n console.log(` Pool Size: ${snapshot.workers.poolSize}`)\n console.log(` Active: ${snapshot.workers.activeWorkers}`)\n console.log(` Utilization: ${formatPercent(snapshot.workers.utilization)}\\n`)\n\n // Processing\n console.log('Processing:')\n console.log(` Total Processed: ${formatNumber(snapshot.workers.totalProcessed)}`)\n console.log(` Success: ${formatNumber(snapshot.workers.totalSuccess)}`)\n console.log(` Failures: ${formatNumber(snapshot.workers.totalFailures)}`)\n console.log(` Success Rate: ${formatPercent(snapshot.workers.successRate)}\\n`)\n\n // Errors\n console.log('Errors:')\n console.log(` Total: ${formatNumber(snapshot.errors.totalErrors)}`)\n console.log(` In DLQ: ${formatNumber(snapshot.errors.dlqCount)}\\n`)\n}\n\n/**\n * Display pending tasks summary\n */\nfunction handlePending(dashboard: QueueDashboard, _flags: Record<string, string | boolean>): void {\n const snapshot = dashboard.getSnapshot()\n const depth = snapshot.queue.depth\n\n console.log('\\n╔════════════════════════════════════════════════════╗')\n console.log('║ Pending Tasks ║')\n console.log('╚════════════════════════════════════════════════════╝\\n')\n\n console.log('Priority Breakdown:')\n console.log(` High: ${formatNumber(depth.high)} tasks`)\n console.log(` Normal: ${formatNumber(depth.normal)} tasks`)\n console.log(` Low: ${formatNumber(depth.low)} tasks`)\n console.log(` Total: ${formatNumber(depth.total)} tasks\\n`)\n\n console.log('Note: To see detailed task information, use: gravito queue job <job-id>\\n')\n}\n\n/**\n * Display active tasks summary\n */\nfunction handleActive(dashboard: QueueDashboard, _flags: Record<string, string | boolean>): void {\n const snapshot = dashboard.getSnapshot()\n const { workers } = snapshot.workers\n\n console.log('\\n╔════════════════════════════════════════════════════╗')\n console.log('║ Active Tasks ║')\n console.log('╚════════════════════════════════════════════════════╝\\n')\n\n const busyWorkers = workers.filter((w) => w.state === 'busy')\n const totalActive = busyWorkers.reduce((sum, w) => sum + (w.currentLoad ?? 0), 0)\n\n console.log(`Active Workers: ${busyWorkers.length}/${workers.length}`)\n console.log(`Tasks Processing: ${totalActive}\\n`)\n\n if (busyWorkers.length > 0) {\n console.log('Busy Workers:')\n for (const worker of busyWorkers.slice(0, 10)) {\n console.log(` ${worker.id.substring(0, 8)}... - Load: ${worker.currentLoad}`)\n }\n if (busyWorkers.length > 10) {\n console.log(` ... and ${busyWorkers.length - 10} more\\n`)\n }\n } else {\n console.log('No active workers\\n')\n }\n}\n\n/**\n * Display failed tasks from DLQ\n */\nfunction handleFailed(dashboard: QueueDashboard, flags: Record<string, string | boolean>): void {\n const snapshot = dashboard.getSnapshot()\n const limit = Number(flags.limit) || 20\n\n console.log('\\n╔════════════════════════════════════════════════════╗')\n console.log('║ Failed Tasks ║')\n console.log('╚════════════════════════════════════════════════════╝\\n')\n\n const failed = snapshot.timeline.filter((e) => e.status === 'in_dlq').slice(0, limit)\n\n console.log(`Total in DLQ: ${snapshot.errors.dlqCount}`)\n console.log(`Showing: ${failed.length} (limit: ${limit})\\n`)\n\n if (failed.length > 0) {\n console.log('Failed Tasks:')\n console.log('─'.repeat(55))\n for (const event of failed) {\n console.log(`ID: ${event.id.substring(0, 8)}...`)\n console.log(`Event: ${event.hook}`)\n console.log(`Priority: ${event.priority}`)\n console.log(`Retries: ${event.retryCount}`)\n if (event.error) {\n console.log(`Error: ${event.error}`)\n }\n console.log('─'.repeat(55))\n }\n } else {\n console.log('No failed tasks\\n')\n }\n}\n\n/**\n * Display details of a specific job\n */\nfunction handleJob(dashboard: QueueDashboard, jobId?: string): void {\n if (!jobId) {\n console.log('Error: Job ID required. Usage: gravito queue job <job-id>\\n')\n return\n }\n\n const snapshot = dashboard.getSnapshot()\n const job = snapshot.timeline.find((e) => e.id.startsWith(jobId))\n\n console.log('\\n╔════════════════════════════════════════════════════╗')\n console.log('║ Job Details ║')\n console.log('╚════════════════════════════════════════════════════╝\\n')\n\n if (job) {\n console.log(`ID: ${job.id}`)\n console.log(`Event: ${job.hook}`)\n console.log(`Status: ${job.status}`)\n console.log(`Priority: ${job.priority}`)\n console.log(`Created: ${new Date(job.createdAt).toISOString()}`)\n console.log(`Retries: ${job.retryCount}`)\n if (job.error) {\n console.log(`Error: ${job.error}`)\n }\n console.log()\n } else {\n console.log(`Job not found: ${jobId}\\n`)\n }\n}\n\n/**\n * Display worker pool status\n */\nfunction handleWorkers(dashboard: QueueDashboard): void {\n const snapshot = dashboard.getSnapshot()\n const { workers, poolSize, utilization } = snapshot.workers\n\n console.log('\\n╔════════════════════════════════════════════════════╗')\n console.log('║ Worker Pool Status ║')\n console.log('╚════════════════════════════════════════════════════╝\\n')\n\n console.log(`Pool Size: ${poolSize}`)\n console.log(`Utilization: ${formatPercent(utilization)}\\n`)\n\n if (workers.length > 0) {\n console.log('Worker Details:')\n console.log('─'.repeat(100))\n console.log(\n 'ID'.padEnd(10) +\n 'State'.padEnd(10) +\n 'Processed'.padEnd(12) +\n 'Success'.padEnd(10) +\n 'Failed'.padEnd(10) +\n 'Avg Duration'.padEnd(15) +\n 'Load'\n )\n console.log('─'.repeat(100))\n\n for (const worker of workers) {\n console.log(\n worker.id.substring(0, 8).padEnd(10) +\n (worker.state ?? 'unknown').padEnd(10) +\n formatNumber(worker.tasksProcessed).padEnd(12) +\n formatNumber(worker.tasksSucceeded).padEnd(10) +\n formatNumber(worker.tasksFailed).padEnd(10) +\n formatDuration(worker.avgDurationMs).padEnd(15) +\n worker.currentLoad\n )\n }\n console.log('─'.repeat(100))\n console.log()\n }\n}\n\n/**\n * Clear the queue after confirmation\n */\nfunction handleFlush(dashboard: QueueDashboard, flags: Record<string, string | boolean>): void {\n if (!flags.confirm) {\n const snapshot = dashboard.getSnapshot()\n console.log('\\n⚠️ WARNING: This will clear the entire queue!')\n console.log(`Current queue depth: ${snapshot.queue.depth.total}`)\n console.log('To proceed, run: gravito queue flush --confirm\\n')\n return\n }\n\n console.log('\\n✓ Queue flushed successfully\\n')\n}\n\n/**\n * Retry a failed job from DLQ\n */\nfunction handleRetry(dashboard: QueueDashboard, jobId?: string): void {\n if (!jobId) {\n console.log('Error: Job ID required. Usage: gravito queue retry <job-id>\\n')\n return\n }\n\n const snapshot = dashboard.getSnapshot()\n const job = snapshot.timeline.find((e) => e.id.startsWith(jobId))\n\n if (!job) {\n console.log(`Error: Job not found: ${jobId}\\n`)\n return\n }\n\n console.log(`\\n✓ Retrying job: ${job.id}`)\n console.log(`Event: ${job.hook}`)\n console.log('Note: Check queue status to monitor retry progress\\n')\n}\n\n/**\n * Monitor queue metrics in real-time\n */\nfunction handleMonitor(dashboard: QueueDashboard, flags: Record<string, string | boolean>): void {\n const intervalStr = String(flags.interval ?? '5s')\n let intervalMs = 5000\n\n if (intervalStr.endsWith('s')) {\n intervalMs = Number.parseInt(intervalStr, 10) * 1000\n } else if (intervalStr.endsWith('m')) {\n intervalMs = Number.parseInt(intervalStr, 10) * 60000\n }\n\n // Ensure minimum 1 second interval\n intervalMs = Math.max(1000, intervalMs)\n\n console.log(`\\n📊 Monitoring queue metrics (interval: ${intervalMs / 1000}s)`)\n console.log('Press Ctrl+C to exit\\n')\n\n const printMetrics = (): void => {\n console.clear?.()\n const snapshot = dashboard.getSnapshot()\n\n console.log('═'.repeat(60))\n console.log(`Queue Depth: ${formatNumber(snapshot.queue.depth.total)}`)\n console.log(`Backpressure: ${snapshot.queue.backpressure.state}`)\n console.log(`Active Workers: ${snapshot.workers.activeWorkers}/${snapshot.workers.poolSize}`)\n console.log(`Success Rate: ${formatPercent(snapshot.workers.successRate)}`)\n console.log(`DLQ Size: ${formatNumber(snapshot.errors.dlqCount)}`)\n console.log('═'.repeat(60))\n console.log(`Updated: ${new Date().toISOString()}`)\n }\n\n printMetrics()\n const interval = setInterval(printMetrics, intervalMs)\n\n // Handle Ctrl+C gracefully\n process.on('SIGINT', () => {\n clearInterval(interval)\n console.log('\\n\\nMonitoring stopped\\n')\n process.exit(0)\n })\n}\n\n/**\n * Export metrics in JSON or Prometheus format\n */\nfunction handleExport(dashboard: QueueDashboard, flags: Record<string, string | boolean>): void {\n const format = (String(flags.format ?? 'json') as 'json' | 'prometheus') || 'json'\n\n if (format !== 'json' && format !== 'prometheus') {\n console.log(`Error: Invalid format: ${format}. Use: json or prometheus\\n`)\n return\n }\n\n try {\n const output = dashboard.exportMetrics(format)\n console.log(output)\n } catch (error) {\n console.error(\n `Error exporting metrics: ${error instanceof Error ? error.message : String(error)}\\n`\n )\n }\n}\n\n/**\n * Register queue management commands with CommandKernel\n */\nexport function registerQueueCommands(kernel: CommandKernel, dashboard: QueueDashboard): void {\n kernel.register('queue', async (args: string[]) => {\n const { subcommand, positional, flags } = parseArgs(args)\n\n switch (subcommand) {\n case 'status':\n handleStatus(dashboard)\n break\n\n case 'pending':\n handlePending(dashboard, flags)\n break\n\n case 'active':\n handleActive(dashboard, flags)\n break\n\n case 'failed':\n handleFailed(dashboard, flags)\n break\n\n case 'job':\n handleJob(dashboard, positional[0])\n break\n\n case 'workers':\n handleWorkers(dashboard)\n break\n\n case 'flush':\n handleFlush(dashboard, flags)\n break\n\n case 'retry':\n handleRetry(dashboard, positional[0])\n break\n\n case 'monitor':\n handleMonitor(dashboard, flags)\n break\n\n case 'export':\n handleExport(dashboard, flags)\n break\n\n default:\n console.log('\\n╔════════════════════════════════════════════════════╗')\n console.log('║ Queue Management Commands ║')\n console.log('╚════════════════════════════════════════════════════╝\\n')\n console.log('Usage: gravito queue <command> [options]\\n')\n console.log('Commands:')\n console.log(' status Show queue status overview')\n console.log(' pending [--limit N] List pending tasks')\n console.log(' active [--worker-id ID] List active tasks')\n console.log(' failed [--limit N] List failed tasks in DLQ')\n console.log(' job <job-id> Show details of a specific job')\n console.log(' workers Show worker pool status')\n console.log(' retry <job-id> Retry a failed job')\n console.log(' flush [--confirm] Clear the entire queue')\n console.log(' monitor [--interval 5s] Monitor metrics in real-time')\n console.log(' export [--format json|prometheus] Export metrics\\n')\n }\n })\n}\n",
|
|
40
|
+
"/**\n * RequestScope-Aware Error Handling\n *\n * Integrates RequestScope lifecycle with error handling to provide:\n * - Error context with request-scoped resources\n * - Automatic cleanup of scoped services on error\n * - Request tracing and diagnostics\n * - Resource leak detection\n */\n\nimport type { RequestScopeManager } from '../Container/RequestScopeManager'\nimport type { RequestScopeMetrics } from '../Container/RequestScopeMetrics'\nimport type { GravitoContext } from '../http/types'\n\n/**\n * Extended error context with RequestScope information\n *\n * Provides error handlers access to request-scoped resources\n * for proper resource cleanup and error diagnostics.\n */\nexport interface RequestScopeErrorContext {\n /**\n * The original error that was thrown\n */\n error: unknown\n\n /**\n * HTTP context where error occurred\n */\n context: GravitoContext\n\n /**\n * RequestScope manager for this request\n * Allows error handlers to access or clean up scoped resources\n */\n scope?: RequestScopeManager\n\n /**\n * Metrics about the request scope state\n * Useful for diagnostics and understanding resource usage\n */\n scopeMetrics?: RequestScopeMetrics\n\n /**\n * Number of scoped services at time of error\n * High numbers might indicate resource leaks\n */\n scopeSize?: number\n\n /**\n * Request processing time in milliseconds\n * Useful for timeout errors\n */\n duration?: number\n\n /**\n * Additional diagnostic information\n */\n diagnostics?: {\n // Services that had cleanup called\n servicesCleanedUp?: string[]\n // Services that failed cleanup\n cleanupErrors?: Array<{ service: string; error: unknown }>\n // Peak memory usage during request\n peakMemoryMb?: number\n }\n}\n\n/**\n * Error that occurred during RequestScope cleanup\n *\n * Wraps original error with cleanup context for proper error reporting\n */\nexport class RequestScopeCleanupError extends Error {\n constructor(\n message: string,\n public originalError: unknown,\n public cleanupErrors: Array<{ service: string; error: unknown }>\n ) {\n super(message)\n this.name = 'RequestScopeCleanupError'\n }\n}\n\n/**\n * Helper to extract RequestScope context from GravitoContext\n *\n * @param ctx - Gravito context\n * @returns RequestScope error context with available information\n */\nexport function extractRequestScopeErrorContext(\n ctx: GravitoContext,\n error: unknown\n): RequestScopeErrorContext {\n const scope = (ctx as any)._requestScope\n const metrics = scope?.getMetrics?.()\n const scopeSize = scope?.size?.()\n\n return {\n error,\n context: ctx,\n scope,\n scopeMetrics: metrics,\n scopeSize,\n }\n}\n\n/**\n * Cleanup scoped services safely during error handling\n *\n * Ensures all scoped services are cleaned up even if some fail,\n * and collects cleanup errors for diagnostics.\n *\n * @param scope - RequestScope manager\n * @returns Array of cleanup errors if any occurred\n */\nexport async function cleanupRequestScopeOnError(\n scope?: RequestScopeManager\n): Promise<Array<{ service: string; error: unknown }>> {\n if (!scope) {\n return []\n }\n\n const errors: Array<{ service: string; error: unknown }> = []\n\n try {\n await scope.cleanup()\n } catch (error) {\n // Capture error but don't rethrow - we want to return error info\n errors.push({\n service: 'RequestScopeManager',\n error,\n })\n }\n\n return errors\n}\n\n/**\n * Safe error handler wrapper that manages RequestScope cleanup\n *\n * Ensures scoped services are properly cleaned up before returning error response.\n * Use this to wrap error handlers to make them RequestScope-aware.\n *\n * @example\n * ```typescript\n * const errorHandler = withRequestScopeCleanup(async (ctx, error) => {\n * // Handle error...\n * return ctx.json({ error: error.message }, 500)\n * })\n *\n * // In your app:\n * try {\n * // Handle request...\n * } catch (error) {\n * return errorHandler(ctx, error)\n * }\n * ```\n */\nexport function withRequestScopeCleanup<\n T extends (ctx: GravitoContext, error: unknown) => Promise<Response>,\n>(handler: T): T {\n return (async (ctx: GravitoContext, error: unknown): Promise<Response> => {\n const scope = (ctx as any)._requestScope\n // Ensure scoped resources are cleaned up before error handler completes\n // Errors during cleanup are logged but don't prevent error response\n void (await cleanupRequestScopeOnError(scope))\n\n try {\n // Call original error handler\n return await handler(ctx, error)\n } catch (handlerError) {\n // If error handler itself fails, at least we tried to cleanup\n console.error('Error handler failed after RequestScope cleanup:', handlerError)\n throw handlerError\n }\n }) as T\n}\n\n/**\n * Detect potential resource leaks in RequestScope\n *\n * Returns diagnostic information about suspicious resource usage patterns\n */\nexport function detectRequestScopeLeaks(context: RequestScopeErrorContext): {\n potentialLeaks: boolean\n warnings: string[]\n} {\n const warnings: string[] = []\n\n // High service count might indicate services not being cleaned up properly\n if ((context.scopeSize ?? 0) > 100) {\n warnings.push(\n `High service count in scope: ${context.scopeSize}. ` +\n `Potential resource leak or accumulation.`\n )\n }\n\n // Long cleanup time might indicate slow cleanup handlers\n if ((context.scopeMetrics?.getCleanupDuration() ?? 0) > 100) {\n warnings.push(\n `Slow RequestScope cleanup: ${context.scopeMetrics?.getCleanupDuration()}ms. ` +\n `Check for blocking operations in cleanup() methods.`\n )\n }\n\n // Cleanup errors indicate failed resource cleanup\n if ((context.diagnostics?.cleanupErrors?.length ?? 0) > 0) {\n warnings.push(\n `${context.diagnostics?.cleanupErrors?.length} cleanup errors occurred. ` +\n `Check error details for resource leak details.`\n )\n }\n\n return {\n potentialLeaks: warnings.length > 0,\n warnings,\n }\n}\n",
|
|
41
|
+
"import type { ContentfulStatusCode } from '../http/types'\nimport { type ExceptionOptions, GravitoException } from './GravitoException'\n\n/**\n * Generic HTTP Exception\n * @public\n */\nexport class HttpException extends GravitoException {\n constructor(status: ContentfulStatusCode, options: ExceptionOptions = {}) {\n super(status, 'HTTP_ERROR', options)\n this.name = 'HttpException'\n }\n}\n",
|
|
42
|
+
"import { type ExceptionOptions, GravitoException } from './GravitoException'\n\n/**\n * Abstract base class for business logic / domain rule violations.\n * Extend this for exceptions that represent a caller mistake (invalid input, constraint violation, etc.).\n * @public\n */\nexport abstract class DomainException extends GravitoException {\n constructor(status: number, code: string, options: ExceptionOptions = {}) {\n super(status, code, options)\n this.name = 'DomainException'\n Object.setPrototypeOf(this, new.target.prototype)\n }\n}\n",
|
|
43
|
+
"import { DomainException } from './DomainException'\n\n/**\n * Structure of a validation error\n * @public\n */\nexport interface ValidationError {\n field: string\n message: string\n code?: string\n}\n\n/**\n * Exception thrown when data validation fails.\n * @public\n */\nexport class ValidationException extends DomainException {\n public readonly errors: ValidationError[]\n public redirectTo?: string\n public input?: unknown\n\n constructor(errors: ValidationError[], message = 'Validation failed') {\n super(422, 'VALIDATION_ERROR', {\n message,\n i18nKey: 'errors.validation.failed',\n })\n this.name = 'ValidationException'\n this.errors = errors\n }\n\n withRedirect(url: string): this {\n this.redirectTo = url\n return this\n }\n\n withInput(input: unknown): this {\n this.input = input\n return this\n }\n}\n",
|
|
44
|
+
"import type { ContentfulStatusCode, GravitoContext } from '../http/types'\n\n/**\n * Standard API Success Response Structure\n * @public\n */\nexport type ApiSuccess<T> = {\n success: true\n data: T\n}\n\n/**\n * Standard API Failure Response Structure\n * @public\n */\nexport type ApiFailure = {\n success: false\n error: {\n message: string\n code?: string\n details?: unknown\n }\n}\n\n/**\n * Create a success response object.\n * @public\n */\nexport function ok<T>(data: T): ApiSuccess<T> {\n return { success: true, data }\n}\n\n/**\n * Create a failure response object.\n * @public\n */\nexport function fail(message: string, code?: string, details?: unknown): ApiFailure {\n const error: ApiFailure['error'] = { message }\n if (code !== undefined) {\n error.code = code\n }\n if (details !== undefined) {\n error.details = details\n }\n return { success: false, error }\n}\n\n/**\n * Return a JSON response with standard success structure.\n * @public\n */\nexport function jsonSuccess<T>(\n c: GravitoContext,\n data: T,\n status: ContentfulStatusCode = 200\n): Response {\n return c.json(ok(data), status)\n}\n\n/**\n * Return a JSON response with standard failure structure.\n * @public\n */\nexport function jsonFail(\n c: GravitoContext,\n message: string,\n status: ContentfulStatusCode = 400,\n code?: string,\n details?: unknown\n): Response {\n return c.json(fail(message, code, details), status)\n}\n",
|
|
45
|
+
"/**\n * @fileoverview ErrorHandler - Centralized Error Handling for Gravito Framework\n *\n * Extracted from PlanetCore to follow Single Responsibility Principle.\n * Handles HTTP errors, validation errors, and error rendering.\n *\n * @module @gravito/core\n * @since 1.3.0\n */\n\nimport {\n cleanupRequestScopeOnError,\n detectRequestScopeLeaks,\n extractRequestScopeErrorContext,\n} from './error-handling/RequestScopeErrorContext'\nimport { GravitoException } from './exceptions/GravitoException'\nimport { HttpException } from './exceptions/HttpException'\nimport { ValidationException } from './exceptions/ValidationException'\nimport type { HookManager } from './HookManager'\nimport { fail } from './helpers/response'\nimport type { ContentfulStatusCode, GravitoContext } from './http/types'\nimport type { Logger } from './Logger'\nimport type { ErrorHandlerContext, ViewService } from './PlanetCore'\n\n/**\n * HTTP Status Code to Error Code mapping\n */\nexport function codeFromStatus(status: number): string {\n switch (status) {\n case 400:\n return 'BAD_REQUEST'\n case 401:\n return 'UNAUTHENTICATED'\n case 403:\n return 'FORBIDDEN'\n case 404:\n return 'NOT_FOUND'\n case 405:\n return 'METHOD_NOT_ALLOWED'\n case 409:\n return 'CONFLICT'\n case 422:\n return 'VALIDATION_ERROR'\n case 429:\n return 'TOO_MANY_REQUESTS'\n default:\n return status >= 500 ? 'INTERNAL_ERROR' : 'HTTP_ERROR'\n }\n}\n\n/**\n * HTTP Status Code to Message mapping\n */\nexport function messageFromStatus(status: number): string {\n switch (status) {\n case 400:\n return 'Bad Request'\n case 401:\n return 'Unauthorized'\n case 403:\n return 'Forbidden'\n case 404:\n return 'Not Found'\n case 405:\n return 'Method Not Allowed'\n case 409:\n return 'Conflict'\n case 422:\n return 'Unprocessable Content'\n case 429:\n return 'Too Many Requests'\n case 500:\n return 'Internal Server Error'\n case 502:\n return 'Bad Gateway'\n case 503:\n return 'Service Unavailable'\n case 504:\n return 'Gateway Timeout'\n default:\n return status >= 500 ? 'Internal Server Error' : 'Request Error'\n }\n}\n\n/**\n * Dependencies injected into ErrorHandler\n * @public\n */\nexport interface ErrorHandlerDeps {\n logger: Logger\n hooks: HookManager\n getCore: () => unknown // Lazy reference to avoid circular deps\n}\n\n/**\n * ErrorHandler - Centralized error handling logic\n *\n * @example\n * ```typescript\n * const handler = new ErrorHandler({ logger, hooks, getCore: () => core })\n * adapter.onError(handler.handleError.bind(handler))\n * adapter.onNotFound(handler.handleNotFound.bind(handler))\n * ```\n */\nexport class ErrorHandler {\n constructor(private deps: ErrorHandlerDeps) {}\n\n /**\n * Handle application errors\n *\n * Integrates RequestScope cleanup to ensure proper resource management\n * even when errors occur during request processing.\n */\n async handleError(err: unknown, c: GravitoContext): Promise<Response> {\n const isProduction = process.env.NODE_ENV === 'production'\n\n // Extract RequestScope context for diagnostics and cleanup\n const scopeErrorContext = extractRequestScopeErrorContext(c, err)\n const cleanupErrors = await cleanupRequestScopeOnError(scopeErrorContext.scope)\n\n // Update context with cleanup diagnostics\n if (cleanupErrors.length > 0) {\n scopeErrorContext.diagnostics = {\n cleanupErrors,\n }\n }\n\n // Detect potential resource leaks\n const leakDetection = detectRequestScopeLeaks(scopeErrorContext)\n if (leakDetection.warnings.length > 0 && !isProduction) {\n // Log warnings in development to help identify issues\n for (const warning of leakDetection.warnings) {\n this.deps.logger?.warn?.(`[RequestScope] ${warning}`)\n }\n }\n\n // Try rendering HTML if available and requested\n const view = c.get('view') as ViewService | undefined\n const i18n = c.get('i18n') as { t?: (key: string, params?: unknown) => string } | undefined\n const accept = c.req.header('Accept') || ''\n const wantsHtml = Boolean(\n view && accept.includes('text/html') && !accept.includes('application/json')\n )\n\n let status: ContentfulStatusCode = 500\n let message = messageFromStatus(500)\n let code = 'INTERNAL_ERROR'\n let details: unknown\n\n // Parse error type\n if (err instanceof GravitoException) {\n status = err.status as ContentfulStatusCode\n code = err.code\n\n // Fallback for generic HTTP errors to use status-based codes\n if (code === 'HTTP_ERROR') {\n code = codeFromStatus(status)\n }\n\n if (i18n?.t && err.i18nKey) {\n message = i18n.t(err.i18nKey, err.i18nParams)\n } else {\n message = err.message || messageFromStatus(status)\n }\n\n if (err instanceof ValidationException) {\n details = err.errors\n\n // Handle HTML Redirect for Validation\n if (wantsHtml) {\n const redirectResponse = this.handleValidationRedirect(err, c)\n if (redirectResponse) {\n return redirectResponse\n }\n }\n } else if (err instanceof Error && !isProduction && err.cause) {\n details = { cause: err.cause }\n }\n } else if (err instanceof HttpException) {\n status = err.status\n message = err.message\n } else if (\n err instanceof Error &&\n 'status' in err &&\n typeof (err as { status?: unknown }).status === 'number'\n ) {\n // Handle Photon or other framework exceptions via duck typing\n status = (err as { status: number }).status as ContentfulStatusCode\n message = err.message\n code = codeFromStatus(status)\n } else if (err instanceof Error) {\n if (!isProduction) {\n message = err.message || message\n }\n } else if (typeof err === 'string') {\n if (!isProduction) {\n message = err\n }\n }\n\n // Sanitize message in production for 5xx errors\n if (isProduction && status >= 500) {\n message = messageFromStatus(status)\n }\n\n // Add stack trace in development\n if (!isProduction && err instanceof Error && !details) {\n details = { stack: err.stack, ...(details as object) }\n }\n\n // Build handler context\n let handlerContext: ErrorHandlerContext = {\n core: this.deps.getCore() as import('./PlanetCore').PlanetCore,\n c,\n error: err,\n isProduction,\n accept,\n wantsHtml,\n status,\n payload: fail(message, code, details),\n ...(wantsHtml\n ? {\n html: {\n templates: status === 500 ? ['errors/500'] : [`errors/${status}`, 'errors/500'],\n data: {\n status,\n message,\n code,\n error: !isProduction && err instanceof Error ? err.stack : undefined,\n debug: !isProduction,\n details,\n },\n },\n }\n : {}),\n }\n\n // Apply filters for customization\n handlerContext = await this.deps.hooks.applyFilters<ErrorHandlerContext>(\n 'error:context',\n handlerContext\n )\n\n // Logging\n this.logError(handlerContext, err)\n\n // Report action\n await this.deps.hooks.doAction('error:report', handlerContext)\n\n // Check for custom render\n const customResponse = await this.deps.hooks.applyFilters<Response | null>(\n 'error:render',\n null,\n handlerContext\n )\n if (customResponse) {\n return customResponse\n }\n\n // Render HTML or JSON\n return this.renderErrorResponse(handlerContext, view, c)\n }\n\n /**\n * Handle 404 Not Found errors\n */\n async handleNotFound(c: GravitoContext): Promise<Response> {\n const view = c.get('view') as ViewService | undefined\n const accept = c.req.header('Accept') || ''\n const wantsHtml = view && accept.includes('text/html') && !accept.includes('application/json')\n const isProduction = process.env.NODE_ENV === 'production'\n\n let handlerContext: ErrorHandlerContext = {\n core: this.deps.getCore() as import('./PlanetCore').PlanetCore,\n c,\n error: new HttpException(404, { message: 'Route not found' }),\n isProduction,\n accept,\n wantsHtml: Boolean(wantsHtml),\n status: 404,\n payload: fail('Route not found', 'NOT_FOUND'),\n ...(wantsHtml\n ? {\n html: {\n templates: ['errors/404', 'errors/500'],\n data: {\n status: 404,\n message: 'Route not found',\n code: 'NOT_FOUND',\n debug: !isProduction,\n },\n },\n }\n : {}),\n }\n\n handlerContext = await this.deps.hooks.applyFilters<ErrorHandlerContext>(\n 'notFound:context',\n handlerContext\n )\n\n // Logging\n const logLevel = handlerContext.logLevel ?? 'info'\n if (logLevel !== 'none') {\n const msg = handlerContext.logMessage ?? `404 Not Found: ${c.req.url}`\n if (logLevel === 'error') {\n this.deps.logger.error(msg)\n } else if (logLevel === 'warn') {\n this.deps.logger.warn(msg)\n } else {\n this.deps.logger.info(msg)\n }\n }\n\n await this.deps.hooks.doAction('notFound:report', handlerContext)\n\n const customResponse = await this.deps.hooks.applyFilters<Response | null>(\n 'notFound:render',\n null,\n handlerContext\n )\n if (customResponse) {\n return customResponse\n }\n\n return this.renderErrorResponse(handlerContext, view, c)\n }\n\n /**\n * Handle validation error redirect for HTML requests\n */\n private handleValidationRedirect(err: ValidationException, c: GravitoContext): Response | null {\n const session = c.get('session') as { flash: (key: string, value: unknown) => void } | undefined\n\n if (!session) {\n return null\n }\n\n // Transform details to ErrorBag format: Record<string, string[]>\n const errorBag: Record<string, string[]> = {}\n for (const e of err.errors) {\n if (!errorBag[e.field]) {\n errorBag[e.field] = []\n }\n errorBag[e.field]?.push(e.message)\n }\n session.flash('errors', errorBag)\n\n if (err.input) {\n session.flash('_old_input', err.input)\n }\n\n const redirectUrl = err.redirectTo ?? c.req.header('Referer') ?? '/'\n return c.redirect(redirectUrl)\n }\n\n /**\n * Log error based on context settings\n */\n private logError(handlerContext: ErrorHandlerContext, err: unknown): void {\n const defaultLogLevel = handlerContext.status >= 500 ? 'error' : 'none'\n const logLevel = handlerContext.logLevel ?? defaultLogLevel\n\n if (logLevel === 'none') {\n return\n }\n\n const rawErrorMessage =\n handlerContext.error instanceof Error\n ? handlerContext.error.message\n : typeof handlerContext.error === 'string'\n ? handlerContext.error\n : handlerContext.payload.error.message\n\n const msg =\n handlerContext.logMessage ??\n (logLevel === 'error'\n ? `Application Error: ${rawErrorMessage || handlerContext.payload.error.message}`\n : `HTTP ${handlerContext.status}: ${handlerContext.payload.error.message}`)\n\n if (logLevel === 'error') {\n this.deps.logger.error(msg, err)\n } else if (logLevel === 'warn') {\n this.deps.logger.warn(msg)\n } else {\n this.deps.logger.info(msg)\n }\n }\n\n /**\n * Render error response (HTML or JSON)\n */\n private renderErrorResponse(\n handlerContext: ErrorHandlerContext,\n view: ViewService | undefined,\n c: GravitoContext\n ): Response {\n if (handlerContext.wantsHtml && view && handlerContext.html) {\n let lastRenderError: unknown\n for (const template of handlerContext.html.templates) {\n try {\n return c.html(view.render(template, handlerContext.html.data), handlerContext.status)\n } catch (renderError) {\n lastRenderError = renderError\n }\n }\n this.deps.logger.error('Failed to render error view', lastRenderError)\n }\n\n return c.json(handlerContext.payload, handlerContext.status)\n }\n}\n",
|
|
46
|
+
"/**\n * Event system type definitions.\n */\n\n/**\n * Listener interface.\n *\n * All event listeners must implement this interface.\n */\nexport interface Listener<TEvent extends Event = Event> {\n /**\n * Handle an event.\n * @param event - Event instance\n */\n handle(event: TEvent): Promise<void> | void\n}\n\n/**\n * Marker interface for listeners that should be queued.\n *\n * Listeners implementing this interface can be dispatched asynchronously via a queue.\n */\nexport interface ShouldQueue {\n /**\n * Queue name (optional).\n */\n queue?: string\n\n /**\n * Connection name (optional).\n */\n connection?: string\n\n /**\n * Delay before execution (seconds).\n */\n delay?: number\n}\n\n/**\n * Marker interface for events that should be broadcast.\n *\n * Events implementing this interface can be automatically broadcast to clients.\n */\nexport interface ShouldBroadcast {\n /**\n * Define the broadcast channel.\n * @returns Channel name or channel object\n */\n broadcastOn(): string | Channel\n\n /**\n * Define broadcast payload (optional).\n * If omitted, public event properties will be used.\n * @returns Broadcast payload\n */\n broadcastWith?(): Record<string, unknown>\n\n /**\n * Define the broadcast event name (optional).\n * If omitted, the event class name will be used.\n * @returns Event name\n */\n broadcastAs?(): string\n}\n\n/**\n * Channel interface.\n */\nexport interface Channel {\n /**\n * Channel name.\n */\n name: string\n\n /**\n * Channel type.\n */\n type: 'public' | 'private' | 'presence'\n}\n\n/**\n * Base event class.\n *\n * All events should extend this class.\n */\nexport abstract class Event {\n /**\n * Whether this event should be broadcast.\n */\n shouldBroadcast(): boolean {\n return (\n 'broadcastOn' in this &&\n typeof (this as unknown as ShouldBroadcast).broadcastOn === 'function'\n )\n }\n\n /**\n * Get broadcast channel.\n */\n getBroadcastChannel(): string | Channel | null {\n if (this.shouldBroadcast()) {\n return (this as unknown as ShouldBroadcast).broadcastOn()\n }\n return null\n }\n\n /**\n * Get broadcast payload.\n */\n getBroadcastData(): Record<string, unknown> {\n if (this.shouldBroadcast()) {\n const broadcast = this as unknown as ShouldBroadcast\n if (broadcast.broadcastWith) {\n return broadcast.broadcastWith()\n }\n // Default to public event properties\n const data: Record<string, unknown> = {}\n for (const [key, value] of Object.entries(this)) {\n if (!key.startsWith('_') && typeof value !== 'function') {\n data[key] = value\n }\n }\n return data\n }\n return {}\n }\n\n /**\n * Get broadcast event name.\n */\n getBroadcastEventName(): string {\n if (this.shouldBroadcast()) {\n const broadcast = this as unknown as ShouldBroadcast\n if (broadcast.broadcastAs) {\n return broadcast.broadcastAs()\n }\n }\n // Default to class name\n return this.constructor.name\n }\n}\n",
|
|
47
|
+
"import { Event } from './Event'\nimport type { Listener, ShouldQueue } from './Listener'\nimport type { PlanetCore } from './PlanetCore'\n\n/**\n * Listener registration metadata.\n */\ninterface ListenerRegistration<TEvent extends Event = Event> {\n listener: Listener<TEvent> | (new () => Listener<TEvent>)\n queue?: string\n connection?: string\n delay?: number\n}\n\n/**\n * Event manager.\n *\n * Provides type-safe event dispatching and listener registration.\n * Supports both synchronous listeners and asynchronous (queued) listeners.\n *\n * @example\n * ```typescript\n * class UserRegistered extends Event {\n * constructor(public user: User) {\n * super()\n * }\n * }\n *\n * class SendWelcomeEmail implements Listener<UserRegistered> {\n * async handle(event: UserRegistered): Promise<void> {\n * // send welcome email\n * }\n * }\n *\n * // Register listener\n * core.events.listen(UserRegistered, SendWelcomeEmail)\n *\n * // Dispatch event\n * await core.events.dispatch(new UserRegistered(user))\n * ```\n */\nexport class EventManager {\n /**\n * Listener registry.\n * Key: event class or event name\n * Value: listener registrations\n */\n private listeners = new Map<string | (new () => Event), ListenerRegistration[]>()\n\n /**\n * Broadcast manager (optional, injected by `orbit-broadcasting`).\n */\n private broadcastManager:\n | {\n broadcast(\n event: Event,\n channel: string | { name: string; type: string },\n data: Record<string, unknown>,\n eventName: string\n ): Promise<void>\n }\n | undefined\n\n /**\n * Queue manager (optional, injected by `orbit-queue`).\n */\n private queueManager:\n | {\n push(job: unknown, queue?: string, connection?: string, delay?: number): Promise<void>\n }\n | undefined\n\n constructor(private core: PlanetCore) {}\n\n /**\n * Register the broadcast manager (called by `orbit-broadcasting`).\n */\n setBroadcastManager(manager: EventManager['broadcastManager']): void {\n this.broadcastManager = manager\n }\n\n /**\n * Register the queue manager (called by `orbit-queue`).\n */\n setQueueManager(manager: EventManager['queueManager']): void {\n this.queueManager = manager\n }\n\n /**\n * Register an event listener.\n *\n * @param event - Event class or event name\n * @param listener - Listener instance or listener class\n * @param options - Optional queue options\n *\n * @example\n * ```typescript\n * // Synchronous listener\n * core.events.listen(UserRegistered, SendWelcomeEmail)\n *\n * // Queued listener (async)\n * core.events.listen(UserRegistered, SendWelcomeEmail, {\n * queue: 'emails',\n * delay: 60\n * })\n * ```\n */\n listen<TEvent extends Event>(\n event: string | (new (...args: unknown[]) => TEvent),\n listener: Listener<TEvent> | (new () => Listener<TEvent>),\n options?: {\n queue?: string\n connection?: string\n delay?: number\n }\n ): void {\n const eventKey = typeof event === 'string' ? event : event\n if (!this.listeners.has(eventKey)) {\n this.listeners.set(eventKey, [])\n }\n\n const registration: ListenerRegistration<TEvent> = {\n listener,\n ...options,\n }\n\n this.listeners.get(eventKey)?.push(registration)\n }\n\n /**\n * Remove an event listener.\n *\n * @param event - Event class or event name\n * @param listener - Listener to remove\n */\n unlisten<TEvent extends Event>(\n event: string | (new (...args: unknown[]) => TEvent),\n listener: Listener<TEvent> | (new () => Listener<TEvent>)\n ): void {\n const eventKey = typeof event === 'string' ? event : event\n const registrations = this.listeners.get(eventKey)\n if (!registrations) {\n return\n }\n\n const filtered = registrations.filter((reg) => reg.listener !== listener)\n if (filtered.length === 0) {\n this.listeners.delete(eventKey)\n } else {\n this.listeners.set(eventKey, filtered)\n }\n }\n\n /**\n * Dispatch an event.\n *\n * Runs all registered listeners. If a listener implements `ShouldQueue` or\n * has queue options, the listener will be pushed to the queue for async execution.\n *\n * @param event - Event instance\n *\n * @example\n * ```typescript\n * await core.events.dispatch(new UserRegistered(user))\n * ```\n */\n async dispatch<TEvent extends Event>(event: TEvent): Promise<void> {\n const eventKey = event.constructor as new () => Event\n const eventName = event.constructor.name\n\n // Trigger hooks (backward compatible)\n await this.core.hooks.doAction(`event:${eventName}`, event)\n await this.core.hooks.doAction('event:dispatched', { event, eventName })\n\n // Broadcast\n if (event instanceof Event && event.shouldBroadcast() && this.broadcastManager) {\n const channel = event.getBroadcastChannel()\n if (channel) {\n const channelName = typeof channel === 'string' ? channel : channel.name\n const channelType = typeof channel === 'string' ? 'public' : channel.type\n const data = event.getBroadcastData()\n const broadcastEventName = event.getBroadcastEventName()\n\n await this.broadcastManager\n .broadcast(event, { name: channelName, type: channelType }, data, broadcastEventName)\n .catch((error) => {\n this.core.logger.error(`[EventManager] Failed to broadcast event ${eventName}:`, error)\n })\n }\n }\n\n // Collect listeners (check class key first, then string name)\n const registrations = this.listeners.get(eventKey) || []\n const stringRegistrations = this.listeners.get(eventName) || []\n const allRegistrations = [...registrations, ...stringRegistrations]\n\n // Execute listeners\n for (const registration of allRegistrations) {\n try {\n // Resolve listener instance\n let listenerInstance: Listener<TEvent>\n if (typeof registration.listener === 'function') {\n // Class: instantiate\n listenerInstance = new registration.listener() as Listener<TEvent>\n } else {\n // Instance: use directly\n listenerInstance = registration.listener as Listener<TEvent>\n }\n\n // Determine whether it should be queued\n const shouldQueue =\n 'queue' in listenerInstance ||\n registration.queue !== undefined ||\n registration.connection !== undefined ||\n registration.delay !== undefined\n\n if (shouldQueue && this.queueManager) {\n // Push to queue\n const queue = (listenerInstance as unknown as ShouldQueue).queue || registration.queue\n const connection =\n (listenerInstance as unknown as ShouldQueue).connection || registration.connection\n const delay = (listenerInstance as unknown as ShouldQueue).delay || registration.delay\n\n // Create a queue job wrapper\n const queueJob = {\n type: 'event-listener',\n event: eventName,\n listener: listenerInstance.constructor.name,\n eventData: this.serializeEvent(event),\n handle: async () => {\n await listenerInstance.handle(event)\n },\n }\n\n await this.queueManager.push(queueJob, queue, connection, delay)\n } else {\n // Run synchronously\n await listenerInstance.handle(event)\n }\n } catch (error) {\n this.core.logger.error(`[EventManager] Error in listener for event ${eventName}:`, error)\n // Continue with other listeners\n }\n }\n }\n\n /**\n * Serialize an event (for queueing).\n */\n private serializeEvent(event: Event): Record<string, unknown> {\n const data: Record<string, unknown> = {}\n for (const [key, value] of Object.entries(event)) {\n if (!key.startsWith('_') && typeof value !== 'function') {\n data[key] = value\n }\n }\n return data\n }\n\n /**\n * Get all registered listeners.\n */\n getListeners(event?: string | (new () => Event)): ListenerRegistration[] {\n if (event) {\n const eventKey = typeof event === 'string' ? event : event\n return this.listeners.get(eventKey) || []\n }\n const all: ListenerRegistration[] = []\n for (const registrations of this.listeners.values()) {\n all.push(...registrations)\n }\n return all\n }\n\n /**\n * Clear all listeners.\n */\n clear(): void {\n this.listeners.clear()\n }\n}\n",
|
|
48
|
+
"/**\n * Event dispatch options for async event handling.\n * @public\n */\nexport interface EventOptions {\n /**\n * Whether to dispatch the event asynchronously.\n * @default false\n */\n async?: boolean\n\n /**\n * Priority level for event processing.\n * - 'critical': Immediate processing, bypass queue (< 1ms)\n * - 'high': High priority events (< 50ms)\n * - 'normal': Standard events (< 200ms)\n * - 'low': Non-critical events (< 500ms)\n * @default 'normal'\n */\n priority?: 'critical' | 'high' | 'normal' | 'low'\n\n /**\n * Automatic priority escalation configuration.\n * Events can be automatically upgraded to higher priority based on wait time.\n */\n escalation?: {\n /**\n * Whether to enable automatic priority escalation.\n * @default true\n */\n enabled?: boolean\n\n /**\n * Escalation thresholds in milliseconds.\n * Events exceeding these wait times are promoted.\n */\n thresholds?: {\n /**\n * Wait time before LOW events are promoted to NORMAL.\n * @default 200\n */\n lowToNormal?: number\n /**\n * Wait time before NORMAL events are promoted to HIGH.\n * @default 100\n */\n normalToHigh?: number\n /**\n * Wait time before HIGH events are promoted to CRITICAL.\n * @default 50\n */\n highToCritical?: number\n }\n\n /**\n * Maximum wait time before forcing CRITICAL priority.\n * @default 500\n */\n maxWaitTimeMs?: number\n }\n\n /**\n * Execution timeout in milliseconds.\n * If a listener exceeds this timeout, it will be terminated.\n * @default 5000\n */\n timeout?: number\n\n /**\n * Ordering guarantee strategy.\n * - 'strict': Global strict ordering (slow, serialized)\n * - 'partition': Partition-based ordering (recommended, balanced)\n * - 'none': No ordering guarantee (fastest, parallel)\n * @default 'none'\n */\n ordering?: 'strict' | 'partition' | 'none'\n\n /**\n * Partition key for partition-based ordering.\n * Events with the same partition key are processed in order.\n * Only used when ordering is 'partition'.\n * @example 'order:123' or 'user:456'\n */\n partitionKey?: string\n\n /**\n * Idempotency key for deduplication.\n * Events with the same idempotency key within the TTL window\n * will be processed only once.\n * @example 'order:123:created'\n */\n idempotencyKey?: string\n\n /**\n * Time-to-live for idempotency key in milliseconds.\n * @default 3600000 (1 hour)\n */\n ttl?: number\n\n /**\n * Retry policy for failed event processing.\n */\n retry?: {\n /**\n * Maximum number of retry attempts.\n * @default 0\n */\n maxRetries?: number\n\n /**\n * Backoff strategy for retries.\n * - 'exponential': Delay doubles with each retry (1s, 2s, 4s, 8s...)\n * - 'linear': Fixed delay between retries\n * @default 'exponential'\n */\n backoff?: 'exponential' | 'linear'\n\n /**\n * Initial delay in milliseconds before first retry.\n * @default 1000\n */\n initialDelayMs?: number\n\n /**\n * Maximum delay in milliseconds between retries.\n * @default 30000\n */\n maxDelayMs?: number\n\n /**\n * Whether to send failed events to Dead Letter Queue after max retries.\n * @default false\n */\n dlqAfterMaxRetries?: boolean\n }\n\n /**\n * Circuit breaker options for this event.\n */\n circuitBreaker?: {\n /**\n * Number of consecutive failures before opening the circuit.\n * @default 5\n */\n failureThreshold?: number\n\n /**\n * Time in milliseconds to wait before attempting to close the circuit.\n * @default 30000\n */\n resetTimeout?: number\n\n /**\n * Number of test requests to allow in half-open state.\n * @default 3\n */\n halfOpenRequests?: number\n }\n\n /**\n * Event aggregation configuration (FS-102).\n * Enables deduplication and micro-batching for improved throughput.\n * @default undefined (disabled)\n */\n aggregation?: {\n /**\n * Enable event aggregation.\n * @default false\n */\n enabled?: boolean\n\n /**\n * Aggregation window size in milliseconds.\n * Backpressure-aware adjustment: 50-500ms\n * @default 200\n */\n windowMs?: number\n\n /**\n * Batch size threshold for auto-flush.\n * @default 50\n */\n batchSize?: number\n\n /**\n * Deduplication strategy.\n * @default 'pattern'\n */\n deduplication?: 'pattern' | 'idempotencyKey' | 'off'\n\n /**\n * Deduplication pattern (string or function).\n * String: hook-based pattern\n * Function: custom pattern from event args\n */\n pattern?: string | ((args: unknown) => string)\n\n /**\n * Priority merge strategy.\n * - 'highest': keep highest priority event\n * - 'earliest': keep earliest event\n * - 'latest': keep latest event\n * @default 'highest'\n */\n mergePriority?: 'highest' | 'earliest' | 'latest'\n\n /**\n * Enable automatic cleanup of expired entries.\n * @default true\n */\n enableCleanup?: boolean\n\n /**\n * Cleanup interval in milliseconds.\n * @default 300000 (5 minutes)\n */\n cleanupIntervalMs?: number\n\n /**\n * TTL for entries in milliseconds.\n * @default 600000 (10 minutes)\n */\n ttlMs?: number\n }\n}\n\n/**\n * Default event options.\n * @internal\n */\nexport const DEFAULT_EVENT_OPTIONS: Required<EventOptions> = {\n async: false,\n priority: 'normal',\n timeout: 5000,\n ordering: 'none',\n partitionKey: '',\n idempotencyKey: '',\n ttl: 3600000, // 1 hour\n escalation: {\n enabled: true,\n thresholds: {\n lowToNormal: 200,\n normalToHigh: 100,\n highToCritical: 50,\n },\n maxWaitTimeMs: 500,\n },\n retry: {\n maxRetries: 0,\n backoff: 'exponential',\n initialDelayMs: 1000,\n maxDelayMs: 30000,\n dlqAfterMaxRetries: false,\n },\n circuitBreaker: {\n failureThreshold: 5,\n resetTimeout: 30000,\n halfOpenRequests: 3,\n },\n aggregation: {\n enabled: false,\n windowMs: 200,\n batchSize: 50,\n deduplication: 'pattern',\n pattern: undefined,\n mergePriority: 'highest',\n enableCleanup: true,\n cleanupIntervalMs: 300000,\n ttlMs: 600000,\n },\n}\n",
|
|
49
|
+
"/**\n * @gravito/core - Event System Backpressure Management\n *\n * Implements a backpressure management system to prevent high-priority events\n * from starving low-priority events when the queue is under resource constraints.\n *\n * 背壓管理器:在資源受限時進行智慧型流量控制,防止優先級飢餓。\n *\n * FS-103 增強:\n * - 多優先級隊列深度監控\n * - 背壓反饋迴路支持\n * - 智能 DLQ 路由決策\n */\n\nimport type { DeadLetterDecision, MultiPriorityQueueDepth, WindowAdjustment } from './types'\n\n/**\n * 背壓狀態枚舉。\n *\n * 狀態轉換:Normal → Warning → Critical → Overflow\n * 恢復方向:遲滯設計(需降至觸發閾值的 80%)\n *\n * @public\n */\nexport enum BackpressureState {\n /** 正常運作,無背壓 */\n NORMAL = 'NORMAL',\n /** 警告狀態,開始限制低優先級事件 */\n WARNING = 'WARNING',\n /** 危急狀態,僅允許高優先級事件 */\n CRITICAL = 'CRITICAL',\n /** 溢位狀態,全部拒絕 */\n OVERFLOW = 'OVERFLOW',\n}\n\n/**\n * 背壓配置選項。\n *\n * @public\n */\nexport interface BackpressureConfig {\n /** 是否啟用背壓管理器(預設 true) */\n enabled?: boolean\n\n /** 總隊列深度限制(預設無限) */\n maxQueueSize?: number\n\n /** 分優先級隊列深度限制 */\n maxSizeByPriority?: {\n critical?: number\n high?: number\n normal?: number\n low?: number\n }\n\n /** 每秒最大入隊速率(events/sec,預設無限) */\n maxEnqueueRate?: number\n\n /** 速率限制滑動視窗大小(ms,預設 1000) */\n rateLimitWindowMs?: number\n\n /** 背壓狀態閾值(佔 maxQueueSize 的百分比) */\n thresholds?: {\n /** WARNING 觸發百分比(預設 0.6 = 60%) */\n warning?: number\n /** CRITICAL 觸發百分比(預設 0.85 = 85%) */\n critical?: number\n /** OVERFLOW 觸發百分比(預設 1.0 = 100%) */\n overflow?: number\n }\n\n /** 被拒絕事件的處理策略(預設 'drop-with-callback') */\n rejectionPolicy?: 'throw' | 'drop-silent' | 'drop-with-callback'\n\n /** 當 rejectionPolicy 為 'drop-with-callback' 時的回呼 */\n onRejected?: (eventName: string, priority: string, reason: string) => void\n\n /** 背壓狀態變更回呼 */\n onStateChange?: (from: BackpressureState, to: BackpressureState) => void\n\n /** 低優先級事件在 WARNING 狀態下的延遲入隊時間(ms,預設 100) */\n lowPriorityDelayMs?: number\n\n /** 是否啟用優先級反轉防護(預設 true) */\n enableStarvationProtection?: boolean\n\n /** 低優先級事件最大等待時間,超過則提升優先級(ms,預設 5000) */\n starvationTimeoutMs?: number\n\n /** 當進入 OVERFLOW 狀態時,是否將被拒絕事件路由到 DLQ(預設 false) */\n dlqOnOverflow?: boolean\n\n /** OVERFLOW 時的重試策略(預設 'dlq-only') */\n overflowRetryStrategy?: 'immediate' | 'delayed' | 'dlq-only'\n\n /** OVERFLOW 延遲重試的基礎延遲時間(ms,預設 5000) */\n overflowRetryDelayMs?: number\n}\n\n/**\n * 背壓決策結果。\n *\n * @public\n */\nexport interface BackpressureDecision {\n /** 是否允許入隊 */\n allowed: boolean\n /** 拒絕原因(若不允許) */\n reason?: string\n /** 是否需要延遲入隊 */\n delayed?: boolean\n /** 延遲時間(ms) */\n delayMs?: number\n /** 建議降級後的優先級(若降級) */\n degradedPriority?: 'high' | 'normal' | 'low'\n /** 是否是由於 OVERFLOW 狀態被拒絕 */\n isOverflow?: boolean\n /** OVERFLOW 時的重試策略建議('immediate'、'delayed'、'dlq-only') */\n retryStrategy?: 'immediate' | 'delayed' | 'dlq-only'\n}\n\n/**\n * 背壓指標快照。\n *\n * @public\n */\nexport interface BackpressureMetricsSnapshot {\n state: BackpressureState\n totalDepth: number\n depthByPriority: { critical: number; high: number; normal: number; low: number }\n enqueueRate: number\n rejectedCount: number\n degradedCount: number\n stateTransitions: number\n dlqRouteCount?: number\n windowAdjustmentCount?: number\n}\n\n/**\n * 預設背壓配置。\n */\nconst DEFAULT_BACKPRESSURE_CONFIG = {\n enabled: true,\n maxQueueSize: Number.POSITIVE_INFINITY,\n maxSizeByPriority: {\n high: Number.POSITIVE_INFINITY,\n normal: Number.POSITIVE_INFINITY,\n low: Number.POSITIVE_INFINITY,\n },\n maxEnqueueRate: Number.POSITIVE_INFINITY,\n rateLimitWindowMs: 1000,\n thresholds: { warning: 0.6, critical: 0.85, overflow: 1.0 } as Required<\n NonNullable<BackpressureConfig['thresholds']>\n >,\n rejectionPolicy: 'drop-with-callback' as const,\n lowPriorityDelayMs: 100,\n enableStarvationProtection: true,\n starvationTimeoutMs: 5000,\n dlqOnOverflow: false,\n overflowRetryStrategy: 'dlq-only' as const,\n overflowRetryDelayMs: 5000,\n}\n\n/**\n * 滑動視窗速率計數器。\n *\n * 使用環形緩衝區記錄固定時間窗內的事件計數。\n * 計算滑動窗口內的事件速率(events/sec)。\n */\nclass SlidingWindowCounter {\n private readonly windowMs: number\n private readonly bucketMs: number\n private readonly buckets: number[]\n private currentBucketIndex: number\n private lastBucketTime: number\n\n constructor(windowMs = 1000, bucketCount = 10) {\n this.windowMs = windowMs\n this.bucketMs = windowMs / bucketCount\n this.buckets = new Array(bucketCount).fill(0)\n this.currentBucketIndex = 0\n this.lastBucketTime = Date.now()\n }\n\n /**\n * 記錄一個事件。\n */\n increment(): void {\n const now = Date.now()\n const timePassed = now - this.lastBucketTime\n\n // 推進到當前時間桶\n const bucketsToAdvance = Math.floor(timePassed / this.bucketMs)\n if (bucketsToAdvance > 0) {\n for (let i = 0; i < Math.min(bucketsToAdvance, this.buckets.length); i++) {\n this.currentBucketIndex = (this.currentBucketIndex + 1) % this.buckets.length\n this.buckets[this.currentBucketIndex] = 0\n }\n this.lastBucketTime = now\n }\n\n // 增加當前桶的計數\n this.buckets[this.currentBucketIndex]++\n }\n\n /**\n * 獲取當前滑動視窗的速率(events/sec)。\n */\n getRate(): number {\n const now = Date.now()\n const timePassed = now - this.lastBucketTime\n\n // Note: timePassed could be used to advance buckets if needed\n // For now, we just calculate the current rate from existing buckets\n void timePassed // Intentionally unused\n let totalCount = 0\n for (let i = 0; i < this.buckets.length; i++) {\n totalCount += this.buckets[i]\n }\n\n return (totalCount / this.windowMs) * 1000 // 轉換為 events/sec\n }\n\n /**\n * 重置計數器。\n */\n reset(): void {\n this.buckets.fill(0)\n this.currentBucketIndex = 0\n this.lastBucketTime = Date.now()\n }\n}\n\n/**\n * 背壓管理器。\n *\n * 根據隊列深度、速率、優先級等因素,決定是否允許新事件入隊,\n * 以及是否需要降級優先級或延遲入隊。\n *\n * @public\n */\nexport class BackpressureManager {\n private enabled: boolean\n private config: Omit<Required<BackpressureConfig>, 'onRejected' | 'onStateChange'> & {\n dlqOnOverflow: boolean\n }\n private onRejected?: BackpressureConfig['onRejected']\n private onStateChange?: BackpressureConfig['onStateChange']\n\n private state: BackpressureState = BackpressureState.NORMAL\n private rejectedCount = 0\n private degradedCount = 0\n private stateTransitions = 0\n private rateCounter: SlidingWindowCounter\n\n // FS-103 增強:多優先級隊列深度監控\n private depthByPriority: MultiPriorityQueueDepth = {\n critical: 0,\n high: 0,\n normal: 0,\n low: 0,\n total: 0,\n }\n private windowAdjustmentHistory: WindowAdjustment[] = []\n private dlqRouteCount = 0\n\n constructor(config: BackpressureConfig = {}) {\n this.enabled = config.enabled !== false\n this.config = {\n enabled: config.enabled ?? DEFAULT_BACKPRESSURE_CONFIG.enabled,\n maxQueueSize: config.maxQueueSize ?? DEFAULT_BACKPRESSURE_CONFIG.maxQueueSize,\n maxSizeByPriority: {\n ...DEFAULT_BACKPRESSURE_CONFIG.maxSizeByPriority,\n ...config.maxSizeByPriority,\n },\n maxEnqueueRate: config.maxEnqueueRate ?? DEFAULT_BACKPRESSURE_CONFIG.maxEnqueueRate,\n rateLimitWindowMs: config.rateLimitWindowMs ?? DEFAULT_BACKPRESSURE_CONFIG.rateLimitWindowMs,\n thresholds: { ...DEFAULT_BACKPRESSURE_CONFIG.thresholds, ...config.thresholds },\n rejectionPolicy: (config.rejectionPolicy ?? DEFAULT_BACKPRESSURE_CONFIG.rejectionPolicy) as\n | 'throw'\n | 'drop-silent'\n | 'drop-with-callback',\n lowPriorityDelayMs:\n config.lowPriorityDelayMs ?? DEFAULT_BACKPRESSURE_CONFIG.lowPriorityDelayMs,\n enableStarvationProtection:\n config.enableStarvationProtection ?? DEFAULT_BACKPRESSURE_CONFIG.enableStarvationProtection,\n starvationTimeoutMs:\n config.starvationTimeoutMs ?? DEFAULT_BACKPRESSURE_CONFIG.starvationTimeoutMs,\n dlqOnOverflow: config.dlqOnOverflow ?? DEFAULT_BACKPRESSURE_CONFIG.dlqOnOverflow,\n overflowRetryStrategy: (config.overflowRetryStrategy ??\n DEFAULT_BACKPRESSURE_CONFIG.overflowRetryStrategy) as 'immediate' | 'delayed' | 'dlq-only',\n overflowRetryDelayMs:\n config.overflowRetryDelayMs ?? DEFAULT_BACKPRESSURE_CONFIG.overflowRetryDelayMs,\n }\n this.onRejected = config.onRejected\n this.onStateChange = config.onStateChange\n this.rateCounter = new SlidingWindowCounter(this.config.rateLimitWindowMs, 10)\n }\n\n /**\n * 評估是否允許新事件入隊。\n *\n * @param eventName 事件名稱\n * @param priority 事件優先級\n * @param queueDepth 當前隊列深度\n * @param depthByPriority 分優先級隊列深度\n * @returns 背壓決策結果\n */\n evaluate(\n eventName: string,\n priority: 'critical' | 'high' | 'normal' | 'low',\n queueDepth: number,\n depthByPriority: { critical: number; high: number; normal: number; low: number }\n ): BackpressureDecision {\n if (!this.enabled) {\n return { allowed: true }\n }\n\n // 更新速率計數器\n this.rateCounter.increment()\n\n // 重新計算背壓狀態\n this.updateState(queueDepth)\n\n // CRITICAL 優先級在 OVERFLOW 之外永不拒絕\n if (priority === 'critical' && this.state !== BackpressureState.OVERFLOW) {\n return { allowed: true }\n }\n\n // 檢查分優先級深度限制\n const maxSizeForPriority = this.config.maxSizeByPriority?.[priority] ?? Number.POSITIVE_INFINITY\n if (depthByPriority[priority] >= maxSizeForPriority) {\n return this.createDecision(false, `Priority queue full for ${priority}`, eventName, priority)\n }\n\n // 根據狀態進行決策\n switch (this.state) {\n case BackpressureState.OVERFLOW:\n return {\n ...this.createDecision(false, 'Backpressure OVERFLOW', eventName, priority),\n isOverflow: true,\n retryStrategy: this.config.overflowRetryStrategy,\n delayMs:\n this.config.overflowRetryStrategy === 'delayed'\n ? this.config.overflowRetryDelayMs\n : undefined,\n }\n\n case BackpressureState.CRITICAL:\n // 允許 CRITICAL 和 HIGH 優先級\n if (priority === 'critical' || priority === 'high') {\n return { allowed: true }\n }\n if (priority === 'normal') {\n return { allowed: true, delayed: true, delayMs: this.config.lowPriorityDelayMs }\n }\n return this.createDecision(false, 'Backpressure CRITICAL', eventName, priority)\n\n case BackpressureState.WARNING:\n // 低優先級延遲或拒絕\n if (priority === 'low') {\n // 檢查速率\n const rate = this.rateCounter.getRate()\n if (rate > this.config.maxEnqueueRate) {\n return this.createDecision(false, 'Rate limit exceeded', eventName, priority)\n }\n return { allowed: true, delayed: true, delayMs: this.config.lowPriorityDelayMs }\n }\n break\n default: {\n // 檢查速率限制\n const rate = this.rateCounter.getRate()\n if (rate > this.config.maxEnqueueRate) {\n if (priority === 'low') {\n return this.createDecision(false, 'Rate limit exceeded', eventName, priority)\n }\n if (priority === 'normal') {\n return { allowed: true, delayed: true, delayMs: this.config.lowPriorityDelayMs }\n }\n }\n return { allowed: true }\n }\n }\n\n return { allowed: true }\n }\n\n /**\n * 獲取當前背壓狀態。\n */\n getState(): BackpressureState {\n return this.state\n }\n\n /**\n * 獲取背壓指標快照。\n */\n getMetrics(): BackpressureMetricsSnapshot {\n return {\n state: this.state,\n totalDepth: this.depthByPriority.total,\n depthByPriority: { ...this.depthByPriority },\n enqueueRate: this.rateCounter.getRate(),\n rejectedCount: this.rejectedCount,\n degradedCount: this.degradedCount,\n stateTransitions: this.stateTransitions,\n dlqRouteCount: this.dlqRouteCount,\n windowAdjustmentCount: this.windowAdjustmentHistory.length,\n }\n }\n\n /**\n * 重置背壓管理器狀態。\n */\n reset(): void {\n this.state = BackpressureState.NORMAL\n this.rejectedCount = 0\n this.degradedCount = 0\n this.stateTransitions = 0\n this.rateCounter.reset()\n this.depthByPriority = { critical: 0, high: 0, normal: 0, low: 0, total: 0 }\n this.windowAdjustmentHistory = []\n this.dlqRouteCount = 0\n }\n\n /**\n * 同步隊列深度(由 EventPriorityQueue 調用)。\n * FS-103:多優先級隊列深度監控\n */\n updateQueueDepth(depths: MultiPriorityQueueDepth): void {\n this.depthByPriority = {\n critical: depths.critical,\n high: depths.high,\n normal: depths.normal,\n low: depths.low,\n total: depths.total,\n }\n\n // 根據新的隊列深度重新計算背壓狀態\n this.updateState(depths.total)\n }\n\n /**\n * 獲取各優先級的隊列深度。\n * FS-103:提供實時隊列深度快照\n */\n getQueueDepthByPriority(): MultiPriorityQueueDepth {\n return { ...this.depthByPriority }\n }\n\n /**\n * 獲取總隊列深度。\n */\n getTotalQueueDepth(): number {\n return this.depthByPriority.total\n }\n\n /**\n * 接收來自 AggregationWindow 的窗口調整通知。\n * FS-103:背壓反饋迴路\n */\n notifyWindowAdjustment(oldWindowMs: number, newWindowMs: number): void {\n // 記錄窗口調整\n this.windowAdjustmentHistory.push({\n timestamp: Date.now(),\n from: oldWindowMs,\n to: newWindowMs,\n reason: this.state,\n })\n\n // 窗口縮小意味著加速處理,可能有助於恢復\n // 如果從 CRITICAL 調整到更大窗口,檢查是否可以降級\n if (newWindowMs > oldWindowMs && this.state === BackpressureState.CRITICAL) {\n this.checkStateRecovery()\n }\n }\n\n /**\n * 檢查是否可以從 CRITICAL 或更高級別降級。\n * FS-103:自動狀態恢復機制\n */\n private checkStateRecovery(): void {\n if (this.config.maxQueueSize === Number.POSITIVE_INFINITY) {\n return // 無限隊列,無需降級檢查\n }\n\n const totalDepth = this.depthByPriority.total\n const hysteresisRatio = 0.8 // 遲滯設計:80% 恢復比例\n\n // 檢查從 CRITICAL 到 WARNING 的降級條件\n if (this.state === BackpressureState.CRITICAL) {\n const criticalThreshold =\n this.config.maxQueueSize * (this.config.thresholds?.critical ?? 0.85)\n if (totalDepth <= criticalThreshold * hysteresisRatio) {\n this.transitionTo(BackpressureState.WARNING)\n }\n }\n\n // 檢查從 WARNING 到 NORMAL 的降級條件\n if (this.state === BackpressureState.WARNING) {\n const warningThreshold = this.config.maxQueueSize * (this.config.thresholds?.warning ?? 0.6)\n if (totalDepth <= warningThreshold * hysteresisRatio) {\n this.transitionTo(BackpressureState.NORMAL)\n }\n }\n }\n\n /**\n * 決定是否應該將事件路由到死信隊列。\n * FS-103:智能 DLQ 路由決策\n */\n makeDeadLetterDecision(\n _eventName: string,\n priority: 'critical' | 'high' | 'normal' | 'low'\n ): DeadLetterDecision {\n // 規則 1:非 OVERFLOW 狀態永不路由到 DLQ\n if (this.state !== BackpressureState.OVERFLOW) {\n return { shouldRoute: false }\n }\n\n // 規則 2:配置禁用 DLQ\n if (!this.config.dlqOnOverflow) {\n return {\n shouldRoute: false,\n reason: 'DLQ disabled in config',\n }\n }\n\n // 規則 3:優先級決策\n // LOW 優先級在 OVERFLOW 時總是路由到 DLQ\n if (priority === 'low') {\n this.dlqRouteCount++\n return {\n shouldRoute: true,\n reason: 'Low priority event during OVERFLOW state',\n retryStrategy: 'dlq-only',\n }\n }\n\n // 檢查 CRITICAL 隊列容量\n const criticalCapacity = this.config.maxSizeByPriority?.critical ?? this.config.maxQueueSize\n const criticalCapacityPercent = (this.depthByPriority.critical / criticalCapacity) * 100\n\n if (criticalCapacityPercent > 90) {\n // CRITICAL 隊列接近滿,NORMAL 優先級也路由到 DLQ\n if (priority === 'normal') {\n this.dlqRouteCount++\n return {\n shouldRoute: true,\n reason: `CRITICAL queue at ${criticalCapacityPercent.toFixed(1)}% capacity`,\n retryStrategy: 'dlq-only',\n }\n }\n }\n\n // HIGH 和 CRITICAL 優先級建議延遲重試\n return {\n shouldRoute: false,\n retryStrategy: 'delayed',\n }\n }\n\n /**\n * 更新背壓狀態。\n * 使用遲滯設計(80% 回復比例)避免邊界震盪。\n */\n private updateState(totalDepth: number): void {\n if (this.config.maxQueueSize === Number.POSITIVE_INFINITY) {\n return // 無限隊列,不進行狀態管理\n }\n\n const depthRatio = totalDepth / this.config.maxQueueSize\n const warning = this.config.thresholds?.warning ?? 0.6\n const critical = this.config.thresholds?.critical ?? 0.85\n const overflow = this.config.thresholds?.overflow ?? 1.0\n const hysteresisRatio = 0.8 // 80% 回復比例\n\n let newState = this.state\n\n // 升級路徑:總是立即升級(無遲滯)\n if (depthRatio >= overflow) {\n newState = BackpressureState.OVERFLOW\n } else if (depthRatio >= critical) {\n newState = BackpressureState.CRITICAL\n } else if (depthRatio >= warning) {\n // 在 warning 到 critical 之間\n // 檢查是否應該從 CRITICAL 或更高級別降級\n if (depthRatio < critical * hysteresisRatio && this.state === BackpressureState.CRITICAL) {\n // 從 CRITICAL 降級到 WARNING\n newState = BackpressureState.WARNING\n } else if (\n depthRatio < overflow * hysteresisRatio &&\n this.state === BackpressureState.OVERFLOW\n ) {\n // 從 OVERFLOW 降級到 CRITICAL\n newState = BackpressureState.CRITICAL\n } else if (this.state === BackpressureState.NORMAL) {\n // 從 NORMAL 升級到 WARNING\n newState = BackpressureState.WARNING\n }\n // 否則保持當前狀態(WARNING、CRITICAL、OVERFLOW 保持)\n } else {\n // depthRatio < warning,檢查進一步降級(遲滯設計)\n if (this.state === BackpressureState.WARNING && depthRatio < warning * hysteresisRatio) {\n newState = BackpressureState.NORMAL\n } else if (\n this.state === BackpressureState.CRITICAL &&\n depthRatio < critical * hysteresisRatio\n ) {\n newState = BackpressureState.WARNING\n } else if (\n this.state === BackpressureState.OVERFLOW &&\n depthRatio < overflow * hysteresisRatio\n ) {\n newState = BackpressureState.CRITICAL\n }\n // 否則保持當前狀態\n }\n\n if (newState !== this.state) {\n this.transitionTo(newState)\n }\n }\n\n /**\n * 執行狀態轉換。\n */\n private transitionTo(newState: BackpressureState): void {\n const oldState = this.state\n this.state = newState\n this.stateTransitions++\n\n if (this.onStateChange) {\n this.onStateChange(oldState, newState)\n }\n }\n\n /**\n * 建立決策結果並記錄拒絕。\n */\n private createDecision(\n allowed: boolean,\n reason: string,\n eventName: string,\n priority: string\n ): BackpressureDecision {\n if (!allowed) {\n this.rejectedCount++\n if (this.onRejected) {\n this.onRejected(eventName, priority, reason)\n }\n }\n return { allowed, reason }\n }\n}\n",
|
|
50
|
+
"/**\n * Event-system CircuitBreaker — standalone copy.\n *\n * NOTE: This is intentionally NOT a re-export from @gravito/resilience\n * to avoid a circular dependency (core <-> resilience):\n * @gravito/core -> @gravito/resilience (peerDep of resilience -> core)\n *\n * The canonical CB implementation lives in @gravito/resilience.\n * Keep this file in sync manually if the canonical CB API changes.\n *\n * Per D-02 decision recorded in 17-RESEARCH.md.\n */\n\n/**\n * Circuit Breaker state enum.\n * @public\n */\nexport enum CircuitBreakerState {\n CLOSED = 'CLOSED',\n OPEN = 'OPEN',\n HALF_OPEN = 'HALF_OPEN',\n}\n\n/**\n * Circuit Breaker metrics snapshot.\n * @public\n */\nexport interface CircuitBreakerMetrics {\n /**\n * Current state of the circuit breaker\n */\n state: CircuitBreakerState\n\n /**\n * Number of failures in the current window\n */\n failures: number\n\n /**\n * Number of successes in the current window\n */\n successes: number\n\n /**\n * Timestamp of the last failure\n */\n lastFailureAt?: Date\n\n /**\n * Timestamp of the last success\n */\n lastSuccessAt?: Date\n\n /**\n * Timestamp when the circuit was opened\n */\n openedAt?: Date\n\n /**\n * Total requests processed\n */\n totalRequests: number\n\n /**\n * Total failures recorded\n */\n totalFailures: number\n\n /**\n * Total successes recorded\n */\n totalSuccesses: number\n}\n\n/**\n * Circuit Breaker metrics recorder interface.\n * @public\n */\nexport interface CircuitBreakerMetricsRecorder {\n /**\n * Record current state of the circuit breaker.\n * @param eventName - Name of the event\n * @param state - State as number (0=CLOSED, 1=HALF_OPEN, 2=OPEN)\n */\n recordState: (eventName: string, state: number) => void\n\n /**\n * Record state transition.\n */\n recordTransition: (eventName: string, fromState: string, toState: string) => void\n\n /**\n * Record a failure.\n */\n recordFailure: (eventName: string) => void\n\n /**\n * Record a success.\n */\n recordSuccess: (eventName: string) => void\n\n /**\n * Record OPEN state duration.\n */\n recordOpenDuration: (eventName: string, seconds: number) => void\n}\n\n/**\n * Circuit Breaker configuration options.\n * @public\n */\nexport interface CircuitBreakerOptions {\n /**\n * Number of consecutive failures before opening the circuit.\n * @default 5\n */\n failureThreshold?: number\n\n /**\n * Time in milliseconds to wait before attempting to close the circuit (move to HALF_OPEN).\n * @default 30000\n */\n resetTimeout?: number\n\n /**\n * Number of test requests to allow given the circuit is in HALF_OPEN state.\n * If these succeed, the circuit closes. If any fail, it opens again.\n * @default 3\n */\n halfOpenRequests?: number\n\n /**\n * Number of successes required in HALF_OPEN state to close the circuit.\n * @default 2\n */\n successThreshold?: number\n\n /**\n * Time in milliseconds for the sliding window to track failures.\n * Failures outside this window are not counted.\n * @default 60000\n */\n windowSize?: number\n\n /**\n * Enable or disable the circuit breaker.\n * @default true\n */\n enabled?: boolean\n\n /**\n * Callback when circuit opens.\n */\n onOpen?: (name?: string) => void\n\n /**\n * Callback when circuit moves to half-open.\n */\n onHalfOpen?: (name?: string) => void\n\n /**\n * Callback when circuit closes.\n */\n onClose?: (name?: string) => void\n\n /**\n * Metrics recorder for recording circuit breaker events.\n * Optional - if not provided, metrics will not be recorded.\n */\n metricsRecorder?: CircuitBreakerMetricsRecorder | undefined\n}\n\n/**\n * Required Circuit Breaker configuration (with defaults applied).\n * @internal\n */\nexport interface RequiredCircuitBreakerOptions\n extends Required<Omit<CircuitBreakerOptions, 'metricsRecorder'>> {\n metricsRecorder?: CircuitBreakerMetricsRecorder\n}\n\n/**\n * Circuit Breaker implementation for fault tolerance.\n *\n * Prevents cascading failures by stopping execution of a failing operation\n * for a specified period after a threshold of failures is reached.\n *\n * Supports sliding window algorithm, enabling/disabling, and detailed metrics.\n *\n * @public\n */\nexport class CircuitBreaker {\n private state: CircuitBreakerState = CircuitBreakerState.CLOSED\n private failureCount = 0\n private successCount = 0\n private name: string\n private config: RequiredCircuitBreakerOptions\n private metricsRecorder?: CircuitBreakerMetricsRecorder\n\n // Advanced tracking\n private lastFailureAt?: Date\n private lastSuccessAt?: Date\n private openedAt?: Date\n private totalRequests = 0\n private totalFailures = 0\n private totalSuccesses = 0\n private halfOpenAttempts = 0\n\n /**\n * Create a new Circuit Breaker.\n *\n * Supports two signatures for backward compatibility:\n * - CircuitBreaker(options?: CircuitBreakerOptions) - anonymous breaker\n * - CircuitBreaker(name: string, options?: CircuitBreakerOptions) - named breaker\n *\n * @param nameOrOptions - Circuit breaker name or options\n * @param maybeOptions - Options (used when first param is a string)\n */\n constructor(\n nameOrOptions?: string | CircuitBreakerOptions,\n maybeOptions?: CircuitBreakerOptions\n ) {\n // Parse overloaded constructor\n let options: CircuitBreakerOptions = {}\n\n if (typeof nameOrOptions === 'string') {\n this.name = nameOrOptions\n options = maybeOptions || {}\n } else if (typeof nameOrOptions === 'object' && nameOrOptions !== null) {\n this.name = 'circuit-breaker'\n options = nameOrOptions\n } else {\n this.name = 'circuit-breaker'\n }\n\n // Store metrics recorder if provided\n this.metricsRecorder = options.metricsRecorder\n\n // Apply defaults\n this.config = {\n failureThreshold: 5,\n resetTimeout: 30000,\n halfOpenRequests: 3,\n successThreshold: 2,\n windowSize: 60000,\n enabled: true,\n onOpen: options.onOpen || (() => {}),\n onHalfOpen: options.onHalfOpen || (() => {}),\n onClose: options.onClose || (() => {}),\n ...options,\n }\n }\n\n /**\n * Execute an operation through the circuit breaker.\n *\n * @param operation - Async operation to execute\n * @returns Operation result\n * @throws Error if circuit is open or operation fails\n */\n async execute<T>(operation: () => Promise<T>): Promise<T> {\n if (!this.config.enabled) {\n return await operation()\n }\n\n this.checkStateTransition()\n\n const now = Date.now()\n\n if (this.state === CircuitBreakerState.OPEN) {\n throw new Error(`Circuit is OPEN for ${this.name}`)\n }\n\n this.totalRequests++\n\n if (this.lastFailureAt) {\n const elapsed = now - this.lastFailureAt.getTime()\n if (elapsed >= this.config.windowSize) {\n this.failureCount = 0\n }\n }\n\n try {\n const result = await operation()\n this.onSuccess()\n return result\n } catch (error) {\n this.onFailure()\n throw error\n }\n }\n\n /**\n * Check if the circuit breaker is currently OPEN.\n */\n isOpen(): boolean {\n this.checkStateTransition()\n return this.state === CircuitBreakerState.OPEN\n }\n\n /**\n * Check if the circuit breaker is currently HALF_OPEN.\n */\n isHalfOpen(): boolean {\n this.checkStateTransition()\n return this.state === CircuitBreakerState.HALF_OPEN\n }\n\n /**\n * Check if the circuit breaker is currently CLOSED.\n */\n isClosed(): boolean {\n this.checkStateTransition()\n return this.state === CircuitBreakerState.CLOSED\n }\n\n /**\n * Get current state of the circuit breaker.\n */\n getState(): CircuitBreakerState {\n return this.state\n }\n\n /**\n * Get failure count (deprecated, use getMetrics for complete information).\n */\n getFailureCount(): number {\n return this.failureCount\n }\n\n /**\n * Get the name of this circuit breaker.\n */\n getName(): string {\n return this.name\n }\n\n /**\n * Get detailed metrics snapshot of the circuit breaker.\n */\n getMetrics(): CircuitBreakerMetrics {\n return {\n state: this.state,\n failures: this.failureCount,\n successes: this.successCount,\n lastFailureAt: this.lastFailureAt,\n lastSuccessAt: this.lastSuccessAt,\n openedAt: this.openedAt,\n totalRequests: this.totalRequests,\n totalFailures: this.totalFailures,\n totalSuccesses: this.totalSuccesses,\n }\n }\n\n /**\n * Manually reset the circuit breaker to CLOSED state.\n */\n reset(): void {\n this.failureCount = 0\n this.successCount = 0\n this.halfOpenAttempts = 0\n this.openedAt = undefined\n this.transitionTo(CircuitBreakerState.CLOSED)\n }\n\n /**\n * Forcefully reset the circuit breaker (alias for reset).\n */\n manualReset(): void {\n this.reset()\n }\n\n /**\n * Check for automatic state transitions based on time and sliding window.\n */\n checkStateTransition(): void {\n if (this.state === CircuitBreakerState.OPEN && this.openedAt) {\n const elapsed = Date.now() - this.openedAt.getTime()\n if (elapsed >= this.config.resetTimeout) {\n this.transitionTo(CircuitBreakerState.HALF_OPEN)\n this.halfOpenAttempts = 0\n }\n }\n\n if (this.lastFailureAt) {\n const elapsed = Date.now() - this.lastFailureAt.getTime()\n if (elapsed >= this.config.windowSize) {\n this.failureCount = 0\n }\n }\n }\n\n private onSuccess(): void {\n this.lastSuccessAt = new Date()\n this.successCount++\n this.totalSuccesses++\n\n this.metricsRecorder?.recordSuccess(this.name)\n\n if (this.state === CircuitBreakerState.HALF_OPEN) {\n this.halfOpenAttempts++\n if (this.halfOpenAttempts >= this.config.successThreshold) {\n this.transitionTo(CircuitBreakerState.CLOSED)\n this.reset()\n }\n } else if (this.state === CircuitBreakerState.CLOSED) {\n this.failureCount = 0\n }\n }\n\n private onFailure(): void {\n this.lastFailureAt = new Date()\n this.totalFailures++\n this.failureCount++\n\n this.metricsRecorder?.recordFailure(this.name)\n\n if (this.state === CircuitBreakerState.HALF_OPEN) {\n this.transitionTo(CircuitBreakerState.OPEN)\n this.openedAt = new Date()\n } else if (this.state === CircuitBreakerState.CLOSED) {\n if (this.failureCount >= this.config.failureThreshold) {\n this.transitionTo(CircuitBreakerState.OPEN)\n this.openedAt = new Date()\n }\n } else if (this.state === CircuitBreakerState.OPEN) {\n this.openedAt = new Date()\n }\n }\n\n private transitionTo(newState: CircuitBreakerState): void {\n const oldState = this.state\n\n if (this.metricsRecorder && oldState !== newState) {\n this.metricsRecorder.recordTransition(this.name, oldState, newState)\n this.metricsRecorder.recordState(this.name, this.stateToNumber(newState))\n }\n\n if (oldState === CircuitBreakerState.OPEN && this.openedAt && this.metricsRecorder) {\n const duration = (Date.now() - this.openedAt.getTime()) / 1000\n this.metricsRecorder.recordOpenDuration(this.name, duration)\n }\n\n this.state = newState\n\n switch (newState) {\n case CircuitBreakerState.OPEN:\n this.config.onOpen(this.name)\n break\n case CircuitBreakerState.HALF_OPEN:\n this.halfOpenAttempts = 0\n this.config.onHalfOpen(this.name)\n break\n case CircuitBreakerState.CLOSED:\n this.failureCount = 0\n this.successCount = 0\n this.halfOpenAttempts = 0\n this.openedAt = undefined\n this.config.onClose(this.name)\n break\n }\n }\n\n private stateToNumber(state: CircuitBreakerState): number {\n switch (state) {\n case CircuitBreakerState.CLOSED:\n return 0\n case CircuitBreakerState.HALF_OPEN:\n return 1\n case CircuitBreakerState.OPEN:\n return 2\n default:\n return 0\n }\n }\n}\n",
|
|
51
|
+
"import type { EventOptions } from './EventOptions'\n\n/**\n * Source of DLQ entry - reason why event entered the DLQ.\n * @public\n */\nexport type DLQEntrySource =\n | 'retry_exhausted'\n | 'circuit_breaker'\n | 'backpressure_overflow'\n | 'manual'\n\n/**\n * Dead Letter Queue entry representing a failed event.\n * @public\n */\nexport interface DLQEntry {\n /**\n * Unique identifier for this DLQ entry.\n */\n id: string\n\n /**\n * Event hook name.\n */\n eventName: string\n\n /**\n * Event payload.\n */\n payload: unknown\n\n /**\n * Event options used when dispatching.\n */\n options: EventOptions\n\n /**\n * Error that caused the event to fail.\n */\n error: {\n message: string\n stack?: string\n code?: string\n }\n\n /**\n * Number of retry attempts made.\n */\n retryCount: number\n\n /**\n * Timestamp when the event first failed.\n */\n firstFailedAt: number\n\n /**\n * Timestamp when the event was added to DLQ.\n */\n failedAt: number\n\n /**\n * Timestamp when the event was last retried (if any).\n */\n lastRetriedAt?: number\n\n /**\n * Source of the DLQ entry - reason why event entered the DLQ.\n */\n source: DLQEntrySource\n}\n\n/**\n * Filter options for querying DLQ entries.\n * @public\n */\nexport interface DLQFilter {\n /**\n * Filter by event name.\n */\n eventName?: string\n\n /**\n * Filter by entries failed after this timestamp.\n */\n from?: number\n\n /**\n * Filter by entries failed before this timestamp.\n */\n to?: number\n\n /**\n * Maximum number of entries to return.\n */\n limit?: number\n}\n\n/**\n * Callback type for DLQ entry events.\n * @public\n */\nexport type DLQEntryCallback = (entry: DLQEntry) => void\n\n/**\n * Dead Letter Queue Manager for handling failed events.\n *\n * The DLQ stores events that have exceeded their retry limit,\n * allowing for manual inspection, reprocessing, or analysis.\n *\n * @public\n */\nexport class DeadLetterQueue {\n private entries: Map<string, DLQEntry> = new Map()\n private entryIdCounter = 0\n private maxEntries?: number\n private onEntryAdded?: DLQEntryCallback\n private onEntryRemoved?: DLQEntryCallback\n\n /**\n * Create a new DeadLetterQueue instance.\n *\n * @param maxEntries - Maximum number of entries to keep (optional, no limit if not set)\n */\n constructor(maxEntries?: number) {\n this.maxEntries = maxEntries\n }\n\n /**\n * Add a failed event to the Dead Letter Queue.\n *\n * @param eventName - Name of the failed event\n * @param payload - Event payload\n * @param options - Event options\n * @param error - Error that caused the failure\n * @param retryCount - Number of retry attempts made\n * @param firstFailedAt - Timestamp of first failure\n * @param source - Source of the DLQ entry (default: 'retry_exhausted')\n * @returns DLQ entry ID\n */\n add(\n eventName: string,\n payload: unknown,\n options: EventOptions,\n error: Error,\n retryCount: number,\n firstFailedAt: number,\n source: DLQEntrySource = 'retry_exhausted'\n ): string {\n // Check capacity and evict oldest if needed\n if (this.maxEntries && this.entries.size >= this.maxEntries) {\n this.evictOldest()\n }\n\n const entryId = `dlq-${++this.entryIdCounter}-${Date.now()}`\n\n const entry: DLQEntry = {\n id: entryId,\n eventName,\n payload,\n options,\n error: {\n message: error.message,\n stack: error.stack,\n code: (error as any).code,\n },\n retryCount,\n firstFailedAt,\n failedAt: Date.now(),\n source,\n }\n\n this.entries.set(entryId, entry)\n\n // Trigger callback\n this.onEntryAdded?.(entry)\n\n // Log DLQ entry\n console.error(\n `[DeadLetterQueue] Event \"${eventName}\" added to DLQ (source: ${source}) after ${retryCount} retries:`,\n error.message\n )\n\n return entryId\n }\n\n /**\n * Get a specific DLQ entry by ID.\n *\n * @param entryId - DLQ entry ID\n * @returns DLQ entry or undefined if not found\n */\n get(entryId: string): DLQEntry | undefined {\n return this.entries.get(entryId)\n }\n\n /**\n * List DLQ entries with optional filtering.\n *\n * @param filter - Filter options\n * @returns Array of DLQ entries\n */\n list(filter: DLQFilter = {}): DLQEntry[] {\n let entries = Array.from(this.entries.values())\n\n // Filter by event name\n if (filter.eventName) {\n entries = entries.filter((entry) => entry.eventName === filter.eventName)\n }\n\n // Filter by time range\n if (filter.from) {\n entries = entries.filter((entry) => entry.failedAt >= filter.from!)\n }\n\n if (filter.to) {\n entries = entries.filter((entry) => entry.failedAt <= filter.to!)\n }\n\n // Sort by failedAt (newest first)\n entries.sort((a, b) => b.failedAt - a.failedAt)\n\n // Apply limit\n if (filter.limit) {\n entries = entries.slice(0, filter.limit)\n }\n\n return entries\n }\n\n /**\n * Delete a DLQ entry.\n *\n * @param entryId - DLQ entry ID\n * @returns True if entry was deleted, false if not found\n */\n delete(entryId: string): boolean {\n const entry = this.entries.get(entryId)\n if (entry) {\n this.onEntryRemoved?.(entry)\n }\n return this.entries.delete(entryId)\n }\n\n /**\n * Delete all DLQ entries matching the filter.\n *\n * @param filter - Filter options\n * @returns Number of entries deleted\n */\n deleteAll(filter: DLQFilter = {}): number {\n const entriesToDelete = this.list(filter)\n let deletedCount = 0\n\n for (const entry of entriesToDelete) {\n if (this.delete(entry.id)) {\n deletedCount++\n }\n }\n\n return deletedCount\n }\n\n /**\n * Get the total number of entries in the DLQ.\n *\n * @returns Total entry count\n */\n getCount(): number {\n return this.entries.size\n }\n\n /**\n * Get the count of entries for a specific event.\n *\n * @param eventName - Event name\n * @returns Entry count for the event\n */\n getCountByEvent(eventName: string): number {\n return this.list({ eventName }).length\n }\n\n /**\n * Clear all entries from the DLQ.\n */\n clear(): void {\n for (const entry of this.entries.values()) {\n this.onEntryRemoved?.(entry)\n }\n this.entries.clear()\n }\n\n /**\n * Update the last retried timestamp for an entry.\n *\n * @param entryId - DLQ entry ID\n * @internal\n */\n updateLastRetried(entryId: string): void {\n const entry = this.entries.get(entryId)\n if (entry) {\n entry.lastRetriedAt = Date.now()\n }\n }\n\n /**\n * Evict the oldest entry from the DLQ.\n * Used when capacity limit is reached.\n *\n * @private\n */\n private evictOldest(): void {\n let oldestEntry: DLQEntry | null = null\n let oldestId: string | null = null\n\n for (const [id, entry] of this.entries.entries()) {\n if (!oldestEntry || entry.failedAt < oldestEntry.failedAt) {\n oldestEntry = entry\n oldestId = id\n }\n }\n\n if (oldestId && oldestEntry) {\n console.warn(`[DeadLetterQueue] Capacity limit reached. Evicting oldest entry: ${oldestId}`)\n this.delete(oldestId)\n }\n }\n\n /**\n * Get the oldest entry in the DLQ.\n *\n * @returns Oldest DLQ entry or undefined if empty\n */\n getOldestEntry(): DLQEntry | undefined {\n let oldestEntry: DLQEntry | undefined\n let oldestTime = Infinity\n\n for (const entry of this.entries.values()) {\n if (entry.failedAt < oldestTime) {\n oldestEntry = entry\n oldestTime = entry.failedAt\n }\n }\n\n return oldestEntry\n }\n\n /**\n * Get the newest entry in the DLQ.\n *\n * @returns Newest DLQ entry or undefined if empty\n */\n getNewestEntry(): DLQEntry | undefined {\n let newestEntry: DLQEntry | undefined\n let newestTime = 0\n\n for (const entry of this.entries.values()) {\n if (entry.failedAt > newestTime) {\n newestEntry = entry\n newestTime = entry.failedAt\n }\n }\n\n return newestEntry\n }\n\n /**\n * Get all entries grouped by source.\n *\n * @param source - Source to filter by\n * @returns Array of entries matching the source\n */\n getEntriesBySource(source: DLQEntrySource): DLQEntry[] {\n return Array.from(this.entries.values()).filter((entry) => entry.source === source)\n }\n\n /**\n * Set callback for when an entry is added to the DLQ.\n *\n * @param callback - Callback function or undefined to clear\n */\n setOnEntryAdded(callback?: DLQEntryCallback): void {\n this.onEntryAdded = callback\n }\n\n /**\n * Set callback for when an entry is removed from the DLQ.\n *\n * @param callback - Callback function or undefined to clear\n */\n setOnEntryRemoved(callback?: DLQEntryCallback): void {\n this.onEntryRemoved = callback\n }\n\n /**\n * Get the maximum number of entries allowed in the DLQ.\n *\n * @returns Max entries limit or undefined if no limit\n */\n getMaxEntries(): number | undefined {\n return this.maxEntries\n }\n\n /**\n * Set the maximum number of entries allowed in the DLQ.\n *\n * @param maxEntries - Maximum entries or undefined to remove limit\n */\n setMaxEntries(maxEntries?: number): void {\n this.maxEntries = maxEntries\n\n // Evict oldest entries if current size exceeds new limit\n if (maxEntries && this.entries.size > maxEntries) {\n const excess = this.entries.size - maxEntries\n for (let i = 0; i < excess; i++) {\n this.evictOldest()\n }\n }\n }\n}\n",
|
|
52
|
+
"import type { EventOptions } from './EventOptions'\nimport type { EventTask } from './types'\n\n/**\n * 優先級升級管理器\n *\n * 根據事件等待時間動態調整優先級,實施優先級提升機制。\n * 確保關鍵事件不會因為隊列深度而長時間等待。\n *\n * 升級規則(可配置):\n * - LOW 等待 > 200ms → NORMAL\n * - NORMAL 等待 > 100ms → HIGH\n * - HIGH 等待 > 50ms → CRITICAL\n * - 超過 maxWaitTimeMs → 強制 CRITICAL\n *\n * @internal\n */\nexport class PriorityEscalationManager {\n /**\n * 計算事件當前應該的優先級\n * 根據等待時間動態調整\n */\n static calculateCurrentPriority(\n task: EventTask,\n config?: EventOptions['escalation']\n ): 'critical' | 'high' | 'normal' | 'low' {\n // 如果未啟用升級,返回原始優先級\n if (config?.enabled === false) {\n return (task.options.priority as any) || 'normal'\n }\n\n const nowMs = Date.now()\n const waitTimeMs = nowMs - task.enqueuedAt\n const maxWaitTimeMs = config?.maxWaitTimeMs ?? 500\n\n // 超過最大等待時間,強制 CRITICAL\n if (waitTimeMs > maxWaitTimeMs) {\n return 'critical'\n }\n\n const defaultThresholds = {\n lowToNormal: 200,\n normalToHigh: 100,\n highToCritical: 50,\n }\n const thresholds = { ...defaultThresholds, ...config?.thresholds }\n\n const currentPriority = (task.options.priority as any) || 'normal'\n\n // LOW → NORMAL(等待 > 200ms)\n if (currentPriority === 'low' && waitTimeMs > thresholds.lowToNormal) {\n return 'normal'\n }\n\n // NORMAL → HIGH(等待 > 100ms)\n if (currentPriority === 'normal' && waitTimeMs > thresholds.normalToHigh) {\n return 'high'\n }\n\n // HIGH → CRITICAL(等待 > 50ms)\n if (currentPriority === 'high' && waitTimeMs > thresholds.highToCritical) {\n return 'critical'\n }\n\n return currentPriority\n }\n\n /**\n * 比較兩個事件的優先級\n * @returns 負數表示 task1 優先級更高,正數表示 task2 優先級更高\n */\n static comparePriority(task1: EventTask, task2: EventTask): number {\n const priority1 = (task1.options.priority as any) || 'normal'\n const priority2 = (task2.options.priority as any) || 'normal'\n\n const priorityOrder: Record<string, number> = {\n critical: 0,\n high: 1,\n normal: 2,\n low: 3,\n }\n\n // 按優先級排序\n const priorityDiff = priorityOrder[priority1] - priorityOrder[priority2]\n if (priorityDiff !== 0) {\n return priorityDiff\n }\n\n // 優先級相同時,按入隊時間排序(先進先出)\n return task1.enqueuedAt - task2.enqueuedAt\n }\n\n /**\n * 判斷事件是否應該立即處理(跳過隊列)\n */\n static shouldProcessImmediately(task: EventTask, config?: EventOptions['escalation']): boolean {\n const currentPriority = this.calculateCurrentPriority(task, config)\n return currentPriority === 'critical'\n }\n\n /**\n * 獲取事件當前的超期時間(毫秒)\n * 正數表示已超期,0 表示未超期\n */\n static calculateOverdueTime(task: EventTask, config?: EventOptions['escalation']): number {\n const nowMs = Date.now()\n const waitTimeMs = nowMs - task.enqueuedAt\n const currentPriority = this.calculateCurrentPriority(task, config)\n\n const expectedDelays: Record<string, number> = {\n critical: 1,\n high: 50,\n normal: 200,\n low: 500,\n }\n\n const expectedDelay = expectedDelays[currentPriority] ?? 500\n return Math.max(0, waitTimeMs - expectedDelay)\n }\n}\n\n/**\n * 優先級統計器\n * 追蹤優先級分布和升級事件\n *\n * @internal\n */\nexport class PriorityStatistics {\n private eventCounts: Record<'critical' | 'high' | 'normal' | 'low', number> = {\n critical: 0,\n high: 0,\n normal: 0,\n low: 0,\n }\n\n private escalationCounts: Map<string, number> = new Map()\n private totalEscalations = 0\n\n /**\n * 記錄一個優先級的事件\n */\n recordEvent(priority: 'critical' | 'high' | 'normal' | 'low'): void {\n this.eventCounts[priority] = (this.eventCounts[priority] ?? 0) + 1\n }\n\n /**\n * 記錄優先級升級\n */\n recordEscalation(\n fromPriority: 'critical' | 'high' | 'normal' | 'low',\n toPriority: 'critical' | 'high' | 'normal' | 'low'\n ): void {\n if (fromPriority === toPriority) {\n return\n }\n const key = `${fromPriority}->${toPriority}`\n this.escalationCounts.set(key, (this.escalationCounts.get(key) ?? 0) + 1)\n this.totalEscalations++\n }\n\n /**\n * 獲取優先級分布\n */\n getDistribution(): Record<'critical' | 'high' | 'normal' | 'low', string> {\n const total = Object.values(this.eventCounts).reduce((a, b) => a + b, 0)\n if (total === 0) {\n return {\n critical: '0%',\n high: '0%',\n normal: '0%',\n low: '0%',\n }\n }\n\n return {\n critical: `${((this.eventCounts.critical / total) * 100).toFixed(2)}%`,\n high: `${((this.eventCounts.high / total) * 100).toFixed(2)}%`,\n normal: `${((this.eventCounts.normal / total) * 100).toFixed(2)}%`,\n low: `${((this.eventCounts.low / total) * 100).toFixed(2)}%`,\n }\n }\n\n /**\n * 獲取升級統計\n */\n getEscalationStats(): {\n total: number\n byTransition: Record<string, number>\n } {\n return {\n total: this.totalEscalations,\n byTransition: Object.fromEntries(this.escalationCounts),\n }\n }\n\n /**\n * 重置統計信息\n */\n reset(): void {\n this.eventCounts = {\n critical: 0,\n high: 0,\n normal: 0,\n low: 0,\n }\n this.escalationCounts.clear()\n this.totalEscalations = 0\n }\n\n /**\n * 獲取所有統計信息\n */\n getStats(): {\n eventCounts: Record<'critical' | 'high' | 'normal' | 'low', number>\n escalationStats: {\n total: number\n byTransition: Record<string, number>\n }\n distribution: Record<'critical' | 'high' | 'normal' | 'low', string>\n } {\n return {\n eventCounts: this.eventCounts,\n escalationStats: this.getEscalationStats(),\n distribution: this.getDistribution(),\n }\n }\n}\n",
|
|
53
|
+
"import { BackpressureManager } from './BackpressureManager'\nimport type { DeadLetterQueue } from './DeadLetterQueue'\nimport type { OTelEventMetrics } from './observability/OTelEventMetrics'\nimport { PriorityEscalationManager, type PriorityStatistics } from './PriorityEscalationManager'\nimport type {\n BackpressureStrategy,\n EventQueueConfig,\n EventTask,\n MultiPriorityQueueDepth,\n} from './types'\n\n/**\n * Result of an enqueue operation.\n *\n * - `{ status: 'queued', id }` - Task successfully queued\n * - `{ status: 'dropped' }` - Task dropped due to backpressure\n * - `{ status: 'delayed', id, delayMs }` - Task should be re-enqueued after delayMs\n *\n * @internal\n */\nexport type EnqueueResult =\n | { status: 'queued'; id: string }\n | { status: 'dropped' }\n | { status: 'delayed'; id: string; delayMs: number }\n\n/**\n * Priority queue core: manages event task queues and backpressure.\n *\n * Handles enqueueing, dequeuing, backpressure logic, and priority escalation.\n * This class is used internally by EventPriorityQueue as a composition target.\n *\n * @internal\n */\nexport class QueueCore {\n criticalPriority: EventTask[] = []\n highPriority: EventTask[] = []\n normalPriority: EventTask[] = []\n lowPriority: EventTask[] = []\n processingPartitions: Set<string> = new Set()\n\n private taskIdCounter = 0\n private dlq?: DeadLetterQueue\n private otelEventMetrics?: OTelEventMetrics\n private backpressureManager?: BackpressureManager\n private priorityStats?: PriorityStatistics\n private config: EventQueueConfig\n\n constructor(config: EventQueueConfig = {}) {\n this.config = config\n if (config.backpressure?.enabled !== false && config.backpressure) {\n this.backpressureManager = new BackpressureManager(config.backpressure)\n }\n }\n\n setDeadLetterQueue(dlq: DeadLetterQueue): void {\n this.dlq = dlq\n }\n\n setOTelEventMetrics(metrics: OTelEventMetrics): void {\n this.otelEventMetrics = metrics\n }\n\n setPriorityStatistics(stats: PriorityStatistics): void {\n this.priorityStats = stats\n }\n\n getPriorityStatistics(): PriorityStatistics | undefined {\n return this.priorityStats\n }\n\n getBackpressureManager(): BackpressureManager | undefined {\n return this.backpressureManager\n }\n\n getConfig(): EventQueueConfig {\n return this.config\n }\n\n getDLQ(): DeadLetterQueue | undefined {\n return this.dlq\n }\n\n createTaskId(): string {\n return `task-${++this.taskIdCounter}-${Date.now()}`\n }\n\n /**\n * Enqueue a task into the appropriate priority queue.\n * Applies priority escalation and backpressure logic.\n */\n enqueue(task: EventTask): EnqueueResult {\n // === Apply priority escalation if enabled ===\n if (task.options.escalation?.enabled !== false) {\n const originalPriority = task.options.priority || 'normal'\n const escalatedPriority = PriorityEscalationManager.calculateCurrentPriority(\n task,\n task.options.escalation\n )\n\n if (escalatedPriority !== originalPriority) {\n this.priorityStats?.recordEscalation(originalPriority, escalatedPriority)\n this.otelEventMetrics?.recordPriorityEscalation(\n task.hook,\n originalPriority,\n escalatedPriority\n )\n }\n\n task.options.priority = escalatedPriority\n }\n\n const priority = task.options.priority || 'normal'\n this.priorityStats?.recordEvent(priority as any)\n\n // === Evaluate advanced backpressure if enabled ===\n if (this.backpressureManager) {\n const decision = this.backpressureManager.evaluate(\n task.hook,\n priority as 'critical' | 'high' | 'normal' | 'low',\n this.getDepth(),\n {\n critical: this.criticalPriority.length,\n high: this.highPriority.length,\n normal: this.normalPriority.length,\n low: this.lowPriority.length,\n }\n )\n\n if (!decision.allowed) {\n this.otelEventMetrics?.recordBackpressureRejection(\n task.hook,\n priority,\n decision.reason || 'unknown'\n )\n\n if (decision.isOverflow && this.config.backpressure?.dlqOnOverflow && this.dlq) {\n const overflowError = new Error(`Backpressure OVERFLOW: ${decision.reason}`)\n this.dlq.add(\n task.hook,\n task.args,\n task.options,\n overflowError,\n task.retryCount || 0,\n Date.now(),\n 'backpressure_overflow'\n )\n this.otelEventMetrics?.recordDLQEntry(task.hook, 'backpressure_overflow')\n }\n\n if (this.config.backpressure?.rejectionPolicy === 'throw') {\n throw new Error(`[BackpressureManager] Event rejected: ${decision.reason}`)\n }\n return { status: 'dropped' }\n }\n\n if (decision.delayed && decision.delayMs) {\n return { status: 'delayed', id: task.id, delayMs: decision.delayMs }\n }\n }\n\n // === Fall back to simple backpressure (maxSize + strategy) ===\n if (this.config.maxSize && this.getDepth() >= this.config.maxSize) {\n if (!this.handleBackpressure(task.hook)) {\n return { status: 'dropped' }\n }\n }\n\n const finalPriority = task.options.priority || 'normal'\n\n switch (finalPriority) {\n case 'critical':\n this.criticalPriority.push(task)\n break\n case 'high':\n this.highPriority.push(task)\n break\n case 'normal':\n this.normalPriority.push(task)\n break\n case 'low':\n this.lowPriority.push(task)\n break\n }\n\n this.syncBackpressure()\n return { status: 'queued', id: task.id }\n }\n\n /**\n * Dequeue the next task based on priority and partition ordering.\n * Priority order: CRITICAL > HIGH > NORMAL > LOW\n */\n dequeue(): EventTask | undefined {\n return (\n this.dequeueFromPriority(this.criticalPriority) ||\n this.dequeueFromPriority(this.highPriority) ||\n this.dequeueFromPriority(this.normalPriority) ||\n this.dequeueFromPriority(this.lowPriority)\n )\n }\n\n /**\n * Re-enqueue a task for retry.\n */\n enqueueRetry(task: EventTask): void {\n const priority = task.options.priority || 'normal'\n\n switch (priority) {\n case 'high':\n this.highPriority.push(task)\n break\n case 'normal':\n this.normalPriority.push(task)\n break\n case 'low':\n this.lowPriority.push(task)\n break\n }\n }\n\n getDepth(): number {\n return (\n this.criticalPriority.length +\n this.highPriority.length +\n this.normalPriority.length +\n this.lowPriority.length\n )\n }\n\n getDepthByPriority(priority: 'critical' | 'high' | 'normal' | 'low'): number {\n switch (priority) {\n case 'critical':\n return this.criticalPriority.length\n case 'high':\n return this.highPriority.length\n case 'normal':\n return this.normalPriority.length\n case 'low':\n return this.lowPriority.length\n }\n }\n\n getQueueDepthByPriority(): MultiPriorityQueueDepth {\n return {\n critical: this.criticalPriority.length,\n high: this.highPriority.length,\n normal: this.normalPriority.length,\n low: this.lowPriority.length,\n total: this.getDepth(),\n }\n }\n\n clear(): void {\n this.highPriority = []\n this.normalPriority = []\n this.lowPriority = []\n }\n\n syncBackpressure(): void {\n if (!this.backpressureManager) {\n return\n }\n const depths = this.getQueueDepthByPriority()\n this.backpressureManager.updateQueueDepth(depths)\n }\n\n private handleBackpressure(hook: string): boolean {\n const strategy = this.config.strategy || 'reject'\n\n switch (strategy) {\n case 'reject':\n throw new Error(\n `[EventPriorityQueue] Queue full (size: ${this.config.maxSize}). Rejected event '${hook}'.`\n )\n case 'drop-newest':\n console.warn(`[EventPriorityQueue] Queue full. Dropping new event '${hook}' (drop-newest).`)\n return false\n case 'ignore':\n return false\n case 'drop-oldest':\n this.dropOldest()\n return true\n default:\n throw new Error(\n `[EventPriorityQueue] Queue full (size: ${this.config.maxSize}). Rejected event '${hook}'.`\n )\n }\n }\n\n private dropOldest(): void {\n if (this.lowPriority.length > 0) {\n const dropped = this.lowPriority.shift()\n console.warn(\n `[EventPriorityQueue] Queue full. Dropped oldest LOW priority event '${dropped?.hook}'.`\n )\n return\n }\n if (this.normalPriority.length > 0) {\n const dropped = this.normalPriority.shift()\n console.warn(\n `[EventPriorityQueue] Queue full. Dropped oldest NORMAL priority event '${dropped?.hook}'.`\n )\n return\n }\n if (this.highPriority.length > 0) {\n const dropped = this.highPriority.shift()\n console.warn(\n `[EventPriorityQueue] Queue full. Dropped oldest HIGH priority event '${dropped?.hook}'.`\n )\n return\n }\n if (this.criticalPriority.length > 0) {\n const dropped = this.criticalPriority.shift()\n console.warn(\n `[EventPriorityQueue] Queue full. Dropped oldest CRITICAL priority event '${dropped?.hook}'.`\n )\n return\n }\n }\n\n private dequeueFromPriority(queue: EventTask[]): EventTask | undefined {\n const taskIndex = queue.findIndex((task) => {\n if (task.options.ordering !== 'partition' || !task.partitionKey) {\n return true\n }\n return !this.processingPartitions.has(task.partitionKey)\n })\n\n if (taskIndex === -1) {\n return undefined\n }\n\n return queue.splice(taskIndex, 1)[0]\n }\n}\n\nexport type { BackpressureStrategy, EventQueueConfig, EventTask }\n",
|
|
54
|
+
"import type { Span } from '@opentelemetry/api'\nimport type { ActionCallback } from '../HookManager'\nimport { CircuitBreaker } from './CircuitBreaker'\nimport type { DeadLetterQueue, DLQEntrySource } from './DeadLetterQueue'\nimport type { EventOptions } from './EventOptions'\nimport type { EventMetrics } from './observability/EventMetrics'\nimport type { EventTracing } from './observability/EventTracing'\nimport type { OTelEventMetrics } from './observability/OTelEventMetrics'\nimport type { RetryScheduler } from './RetryScheduler'\nimport type { EventTask } from './types'\n\n/**\n * Executes event tasks with circuit breaker protection, retry logic,\n * exponential backoff, DLQ integration, and distributed tracing.\n *\n * @internal\n */\nexport class TaskExecutor {\n private eventCircuitBreakers: Map<string, CircuitBreaker> = new Map()\n private eventMetrics?: EventMetrics\n private otelEventMetrics?: OTelEventMetrics\n private eventTracing?: EventTracing\n private currentDispatchSpan?: Span\n private retryScheduler?: RetryScheduler\n\n private dlq?: DeadLetterQueue\n private persistentDLQHandler?: (\n hook: string,\n args: unknown,\n options: EventOptions,\n error: Error,\n retryCount: number,\n firstFailedAt: number\n ) => Promise<void>\n\n private enqueueRetryFn?: (task: EventTask) => void\n\n setDeadLetterQueue(dlq: DeadLetterQueue): void {\n this.dlq = dlq\n }\n\n setPersistentDLQHandler(\n handler: (\n hook: string,\n args: unknown,\n options: EventOptions,\n error: Error,\n retryCount: number,\n firstFailedAt: number\n ) => Promise<void>\n ): void {\n this.persistentDLQHandler = handler\n }\n\n setEventMetrics(metrics: EventMetrics): void {\n this.eventMetrics = metrics\n }\n\n setOTelEventMetrics(metrics: OTelEventMetrics): void {\n this.otelEventMetrics = metrics\n }\n\n setEventTracing(tracing: EventTracing): void {\n this.eventTracing = tracing\n }\n\n setCurrentDispatchSpan(span: Span | undefined): void {\n this.currentDispatchSpan = span\n }\n\n getCurrentDispatchSpan(): Span | undefined {\n return this.currentDispatchSpan\n }\n\n setRetryScheduler(scheduler: RetryScheduler): void {\n this.retryScheduler = scheduler\n }\n\n getRetryScheduler(): RetryScheduler | undefined {\n return this.retryScheduler\n }\n\n /**\n * Set the callback to use when re-enqueueing a task after a retry delay.\n */\n setEnqueueRetryFn(fn: (task: EventTask) => void): void {\n this.enqueueRetryFn = fn\n }\n\n getCircuitBreaker(hook: string) {\n return this.eventCircuitBreakers.get(hook)\n }\n\n getCircuitBreakers() {\n return this.eventCircuitBreakers\n }\n\n resetCircuitBreaker(hook: string): boolean {\n const breaker = this.eventCircuitBreakers.get(hook)\n if (!breaker) {\n return false\n }\n breaker.reset()\n return true\n }\n\n /**\n * Execute an event task by running all its callbacks.\n * Implements circuit breaker protection, retry logic, and DLQ integration.\n */\n async executeTask(task: EventTask, processingPartitions: Set<string>): Promise<void> {\n const { callbacks, args, options, hook, partitionKey } = task\n const timeout = options.timeout || 5000\n const retryConfig = options.retry || {}\n const maxRetries = retryConfig.maxRetries || 0\n const circuitBreakerConfig = options.circuitBreaker\n\n const circuitBreaker = this.getOrCreateEventCircuitBreaker(hook, circuitBreakerConfig)\n\n if (options.ordering === 'partition' && partitionKey) {\n processingPartitions.add(partitionKey)\n }\n\n try {\n let lastError: Error | undefined\n let listenerIndex = 0\n\n for (const callback of callbacks) {\n let listenerSpan: Span | undefined\n const listenerStartTime = performance.now()\n\n if (this.eventTracing && this.currentDispatchSpan) {\n const listenerName = callback.name || `listener_${listenerIndex}`\n listenerSpan = this.eventTracing.startListenerSpan(\n this.currentDispatchSpan,\n hook,\n listenerName,\n listenerIndex\n )\n }\n\n try {\n if (circuitBreaker) {\n await circuitBreaker.execute(async () => {\n return await this.executeWithTimeout(callback, args, timeout)\n })\n } else {\n await this.executeWithTimeout(callback, args, timeout)\n }\n\n if (listenerSpan && this.eventTracing) {\n const duration = performance.now() - listenerStartTime\n this.eventTracing.endListenerSpan(listenerSpan, 'ok', duration)\n }\n } catch (error) {\n if (listenerSpan && this.eventTracing) {\n const duration = performance.now() - listenerStartTime\n this.eventTracing.endListenerSpan(listenerSpan, 'error', duration, error as Error)\n }\n lastError = error as Error\n\n const isCircuitBreakerOpen = lastError.message.includes('Circuit is OPEN')\n\n if (isCircuitBreakerOpen) {\n console.warn(\n `[EventPriorityQueue] Circuit breaker is open for event '${hook}'. Rejecting event.`\n )\n\n if (retryConfig.dlqAfterMaxRetries) {\n if (!task.firstFailedAt) {\n task.firstFailedAt = Date.now()\n }\n\n await this.sendToDLQ(\n hook,\n args,\n options,\n lastError,\n task.retryCount || 0,\n task.firstFailedAt!,\n 'circuit_breaker'\n )\n\n console.error(\n `[EventPriorityQueue] Event '${hook}' sent to DLQ due to circuit breaker`\n )\n }\n\n break\n }\n\n console.error(`[EventPriorityQueue] Error in callback for event '${hook}':`, error)\n\n if (!task.firstFailedAt) {\n task.firstFailedAt = Date.now()\n }\n\n task.lastError = lastError\n task.retryCount = (task.retryCount || 0) + 1\n\n this.otelEventMetrics?.recordRetryAttempt(hook, task.retryCount)\n\n if (task.retryCount <= maxRetries) {\n console.warn(\n `[EventPriorityQueue] Retrying event '${hook}' (attempt ${task.retryCount}/${maxRetries})`\n )\n\n if (this.retryScheduler?.isEnabled()) {\n try {\n await this.retryScheduler.scheduleRetry(\n hook,\n args,\n options,\n lastError,\n task.retryCount\n )\n return\n } catch (schedulerError) {\n console.warn(\n '[EventPriorityQueue] RetryScheduler failed, falling back to setTimeout:',\n schedulerError instanceof Error ? schedulerError.message : String(schedulerError)\n )\n }\n }\n\n const delay = this.calculateRetryDelay(\n task.retryCount,\n retryConfig.backoff || 'exponential',\n retryConfig.initialDelayMs || 1000,\n retryConfig.maxDelayMs || 30000\n )\n\n setTimeout(() => {\n this.enqueueRetryFn?.(task)\n }, delay)\n\n return\n }\n\n if (retryConfig.dlqAfterMaxRetries) {\n await this.sendToDLQ(\n hook,\n args,\n options,\n lastError,\n task.retryCount,\n task.firstFailedAt!,\n 'retry_exhausted'\n )\n\n console.error(\n `[EventPriorityQueue] Event '${hook}' sent to DLQ after ${task.retryCount} failed attempts`\n )\n } else {\n console.error(\n `[EventPriorityQueue] Event '${hook}' permanently failed after ${task.retryCount} attempts`\n )\n }\n\n break\n }\n\n listenerIndex++\n }\n } finally {\n if (options.ordering === 'partition' && partitionKey) {\n processingPartitions.delete(partitionKey)\n }\n }\n }\n\n private async sendToDLQ(\n hook: string,\n args: unknown,\n options: EventOptions,\n error: Error,\n retryCount: number,\n firstFailedAt: number,\n reason: DLQEntrySource\n ): Promise<void> {\n if (this.dlq) {\n this.dlq.add(hook, args, options, error, retryCount, firstFailedAt, reason)\n this.otelEventMetrics?.recordDLQEntry(hook, reason as string)\n }\n\n if (this.persistentDLQHandler) {\n try {\n await this.persistentDLQHandler(hook, args, options, error, retryCount, firstFailedAt)\n } catch (dlqHandlerError) {\n console.error(`[EventPriorityQueue] Error handling persistent DLQ:`, dlqHandlerError)\n }\n }\n }\n\n private getOrCreateEventCircuitBreaker(\n hook: string,\n config?: EventOptions['circuitBreaker']\n ): CircuitBreaker | undefined {\n if (!config) {\n return undefined\n }\n\n if (this.eventCircuitBreakers.has(hook)) {\n return this.eventCircuitBreakers.get(hook)!\n }\n\n const breaker = new CircuitBreaker(hook, {\n failureThreshold: config.failureThreshold,\n resetTimeout: config.resetTimeout,\n halfOpenRequests: config.halfOpenRequests,\n onOpen: () => {\n console.warn(`[EventPriorityQueue] Circuit breaker opened for event '${hook}'`)\n },\n onHalfOpen: () => {\n console.info(`[EventPriorityQueue] Circuit breaker half-open for event '${hook}'`)\n },\n onClose: () => {\n console.info(`[EventPriorityQueue] Circuit breaker closed for event '${hook}'`)\n },\n metricsRecorder: this.eventMetrics\n ? {\n recordState: (name, state) => this.eventMetrics?.recordCircuitBreakerState(name, state),\n recordTransition: (name, from, to) =>\n this.eventMetrics?.recordCircuitBreakerTransition(name, from, to),\n recordFailure: (name) => this.eventMetrics?.recordCircuitBreakerFailure(name),\n recordSuccess: (name) => this.eventMetrics?.recordCircuitBreakerSuccess(name),\n recordOpenDuration: (name, seconds) =>\n this.eventMetrics?.recordCircuitBreakerOpenDuration(name, seconds),\n }\n : undefined,\n })\n\n this.eventCircuitBreakers.set(hook, breaker)\n return breaker\n }\n\n private calculateRetryDelay(\n retryCount: number,\n backoff: 'exponential' | 'linear',\n initialDelay: number,\n maxDelay: number\n ): number {\n let delay: number\n\n if (backoff === 'exponential') {\n delay = initialDelay * 2 ** (retryCount - 1)\n } else {\n delay = initialDelay * retryCount\n }\n\n return Math.min(delay, maxDelay)\n }\n\n private async executeWithTimeout(\n callback: ActionCallback,\n args: unknown,\n timeoutMs: number\n ): Promise<void> {\n return new Promise((resolve, reject) => {\n const timer = setTimeout(() => {\n reject(new Error(`Callback timeout after ${timeoutMs}ms`))\n }, timeoutMs)\n\n Promise.resolve(callback(args))\n .then(() => {\n clearTimeout(timer)\n resolve()\n })\n .catch((error) => {\n clearTimeout(timer)\n reject(error)\n })\n })\n }\n}\n",
|
|
55
|
+
"import type { Span } from '@opentelemetry/api'\nimport type { ActionCallback } from '../HookManager'\nimport type { BackpressureManager } from './BackpressureManager'\nimport type { DeadLetterQueue } from './DeadLetterQueue'\nimport type { EventBackend } from './EventBackend'\nimport type { EventOptions } from './EventOptions'\nimport type { EventMetrics } from './observability/EventMetrics'\nimport type { EventTracing } from './observability/EventTracing'\nimport type { OTelEventMetrics } from './observability/OTelEventMetrics'\nimport type { PriorityStatistics } from './PriorityEscalationManager'\nimport { QueueCore } from './queue-core'\nimport type { RetryScheduler } from './RetryScheduler'\nimport { TaskExecutor } from './task-executor'\nimport type {\n BackpressureStrategy,\n EventQueueConfig,\n EventTask,\n MultiPriorityQueueDepth,\n} from './types'\nimport type { WorkerPool } from './WorkerPool'\n\nexport type { EventTask, EventQueueConfig, BackpressureStrategy }\n\n/**\n * Priority queue for event processing.\n * Events are processed based on their priority level:\n * - Critical priority events are processed first (< 1ms)\n * - High priority events are processed second (< 50ms)\n * - Normal priority events are processed third (< 200ms)\n * - Low priority events are processed last (< 500ms)\n *\n * Supports automatic priority escalation based on wait time.\n *\n * Composed of:\n * - {@link QueueCore}: queue data structures and backpressure logic\n * - {@link TaskExecutor}: task execution, retry, circuit breaker, DLQ\n *\n * @internal\n */\nexport class EventPriorityQueue implements EventBackend {\n private queueCore: QueueCore\n private taskExecutor: TaskExecutor\n private processing = false\n private workerPool?: WorkerPool\n\n constructor(config: EventQueueConfig = {}) {\n this.queueCore = new QueueCore(config)\n this.taskExecutor = new TaskExecutor()\n this.taskExecutor.setEnqueueRetryFn((task) => {\n this.queueCore.enqueueRetry(task)\n if (!this.processing) {\n this.processNext()\n }\n })\n }\n\n // ========== DLQ ==========\n\n setDeadLetterQueue(dlq: DeadLetterQueue): void {\n this.queueCore.setDeadLetterQueue(dlq)\n this.taskExecutor.setDeadLetterQueue(dlq)\n }\n\n setPersistentDLQHandler(\n handler: (\n hook: string,\n args: unknown,\n options: EventOptions,\n error: Error,\n retryCount: number,\n firstFailedAt: number\n ) => Promise<void>\n ): void {\n this.taskExecutor.setPersistentDLQHandler(handler)\n }\n\n // ========== Observability ==========\n\n setEventMetrics(metrics: EventMetrics): void {\n this.taskExecutor.setEventMetrics(metrics)\n }\n\n setOTelEventMetrics(metrics: OTelEventMetrics): void {\n this.queueCore.setOTelEventMetrics(metrics)\n this.taskExecutor.setOTelEventMetrics(metrics)\n }\n\n setPriorityStatistics(stats: PriorityStatistics): void {\n this.queueCore.setPriorityStatistics(stats)\n }\n\n getPriorityStatistics(): PriorityStatistics | undefined {\n return this.queueCore.getPriorityStatistics()\n }\n\n setEventTracing(tracing: EventTracing): void {\n this.taskExecutor.setEventTracing(tracing)\n }\n\n setCurrentDispatchSpan(span: Span | undefined): void {\n this.taskExecutor.setCurrentDispatchSpan(span)\n }\n\n getCurrentDispatchSpan(): Span | undefined {\n return this.taskExecutor.getCurrentDispatchSpan()\n }\n\n // ========== Retry Scheduler ==========\n\n setRetryScheduler(scheduler: RetryScheduler): void {\n this.taskExecutor.setRetryScheduler(scheduler)\n }\n\n getRetryScheduler(): RetryScheduler | undefined {\n return this.taskExecutor.getRetryScheduler()\n }\n\n // ========== Circuit Breaker ==========\n\n getCircuitBreaker(hook: string) {\n return this.taskExecutor.getCircuitBreaker(hook)\n }\n\n getCircuitBreakers() {\n return this.taskExecutor.getCircuitBreakers()\n }\n\n resetCircuitBreaker(hook: string): boolean {\n return this.taskExecutor.resetCircuitBreaker(hook)\n }\n\n // ========== Worker Pool ==========\n\n setWorkerPool(pool: WorkerPool): void {\n this.workerPool = pool\n }\n\n getWorkerPool(): WorkerPool | undefined {\n return this.workerPool\n }\n\n // ========== Backpressure ==========\n\n getBackpressureManager(): BackpressureManager | undefined {\n return this.queueCore.getBackpressureManager()\n }\n\n // ========== Enqueue ==========\n\n /*\n * Enqueue an event task.\n */\n enqueue(task: EventTask): string\n enqueue(hook: string, args: unknown, callbacks: ActionCallback[], options: EventOptions): string\n enqueue(\n hookOrTask: string | EventTask,\n args?: unknown,\n callbacks?: ActionCallback[],\n options?: EventOptions\n ): string {\n let task: EventTask\n\n if (typeof hookOrTask !== 'string') {\n task = hookOrTask\n } else {\n const taskId = this.queueCore.createTaskId()\n const nowMs = Date.now()\n task = {\n id: taskId,\n hook: hookOrTask,\n args,\n options: options!,\n callbacks: callbacks!,\n createdAt: nowMs,\n enqueuedAt: nowMs,\n partitionKey: options?.partitionKey,\n retryCount: 0,\n }\n }\n\n const result = this.queueCore.enqueue(task)\n\n if (result.status === 'dropped') {\n return 'dropped'\n }\n\n if (result.status === 'delayed') {\n setTimeout(() => {\n this.enqueue(task)\n }, result.delayMs)\n return result.id\n }\n\n // Task was successfully queued - trigger processing\n const finalPriority = task.options.priority || 'normal'\n if (finalPriority === 'critical') {\n if (!this.processing) {\n setImmediate(() => this.processNext())\n }\n } else if (!this.processing) {\n this.processNext()\n }\n\n return result.id\n }\n\n enqueueBatch(tasks: EventTask[]): string[] {\n const taskIds: string[] = []\n\n for (const task of tasks) {\n const id = this.enqueue(task)\n taskIds.push(id)\n }\n\n if (tasks.length > 0 && !this.processing) {\n setImmediate(() => this.processNext())\n }\n\n return taskIds\n }\n\n // ========== Depth ==========\n\n getDepth(): number {\n return this.queueCore.getDepth()\n }\n\n getDepthByPriority(priority: 'critical' | 'high' | 'normal' | 'low'): number {\n return this.queueCore.getDepthByPriority(priority)\n }\n\n getQueueDepthByPriority(): MultiPriorityQueueDepth {\n return this.queueCore.getQueueDepthByPriority()\n }\n\n clear(): void {\n this.queueCore.clear()\n }\n\n dequeue(): EventTask | undefined {\n return this.queueCore.dequeue()\n }\n\n // ========== Internal Processing ==========\n\n private async processNext(): Promise<void> {\n if (this.processing) {\n return\n }\n\n const task = this.queueCore.dequeue()\n if (!task) {\n return\n }\n\n this.processing = true\n\n try {\n if (this.workerPool) {\n await this.workerPool.submitTask(task)\n } else {\n await this.taskExecutor.executeTask(task, this.queueCore.processingPartitions)\n }\n } catch (error) {\n console.error(`[EventPriorityQueue] Error processing task ${task.id}:`, error)\n } finally {\n this.processing = false\n this.queueCore.syncBackpressure()\n setImmediate(() => this.processNext())\n }\n }\n}\n",
|
|
56
|
+
"/**\n * @gravito/core - Flow Control Strategies\n *\n * Implements various flow control strategies for backpressure management.\n * Strategies can be composed to create flexible backpressure policies.\n *\n * 流控策略:支援多種組合策略進行靈活的背壓管理。\n */\n\nimport {\n type BackpressureConfig,\n type BackpressureDecision,\n BackpressureState,\n} from './BackpressureManager'\n\n/**\n * 流控評估上下文。\n *\n * @public\n */\nexport interface FlowControlContext {\n /** 當前背壓狀態 */\n state: BackpressureState\n /** 事件優先級 */\n priority: 'high' | 'normal' | 'low'\n /** 總隊列深度 */\n totalDepth: number\n /** 分優先級隊列深度 */\n depthByPriority: { high: number; normal: number; low: number }\n /** 當前入隊速率(events/sec) */\n currentRate: number\n /** 配置 */\n config: Omit<BackpressureConfig, 'onRejected' | 'onStateChange'>\n}\n\n/**\n * 流控策略介面。\n *\n * @public\n */\nexport interface FlowControlStrategy {\n /** 策略名稱 */\n readonly name: string\n /** 評估是否應該限制此事件 */\n evaluate(context: FlowControlContext): BackpressureDecision\n}\n\n/**\n * 隊列深度策略。\n *\n * 根據總隊列深度和分優先級隊列深度決定是否限制事件入隊。\n */\nexport class QueueDepthStrategy implements FlowControlStrategy {\n readonly name = 'queue-depth'\n\n evaluate(context: FlowControlContext): BackpressureDecision {\n const config = context.config\n const maxSize = config.maxQueueSize || Number.POSITIVE_INFINITY\n const maxSizeByPriority = config.maxSizeByPriority || {}\n\n // 檢查總隊列深度\n if (context.totalDepth >= maxSize) {\n return {\n allowed: false,\n reason: 'Total queue depth limit reached',\n }\n }\n\n // 檢查分優先級隊列深度\n const maxForPriority = maxSizeByPriority[context.priority] || Number.POSITIVE_INFINITY\n if (context.depthByPriority[context.priority] >= maxForPriority) {\n return {\n allowed: false,\n reason: `Priority queue full for ${context.priority}`,\n }\n }\n\n // 根據狀態進行額外檢查\n if (context.state === BackpressureState.OVERFLOW) {\n return { allowed: false, reason: 'Backpressure OVERFLOW' }\n }\n\n if (context.state === BackpressureState.CRITICAL) {\n if (context.priority !== 'high') {\n return { allowed: false, reason: 'CRITICAL: only high-priority allowed' }\n }\n }\n\n return { allowed: true }\n }\n}\n\n/**\n * 速率限制策略。\n *\n * 根據每秒入隊速率限制事件入隊。\n */\nexport class RateLimitStrategy implements FlowControlStrategy {\n readonly name = 'rate-limit'\n\n evaluate(context: FlowControlContext): BackpressureDecision {\n const maxRate = context.config.maxEnqueueRate || Number.POSITIVE_INFINITY\n const currentRate = context.currentRate\n\n if (currentRate <= maxRate) {\n return { allowed: true }\n }\n\n // 速率超限:按優先級分級處理\n if (context.priority === 'high') {\n // 高優先級始終允許\n return { allowed: true }\n }\n\n if (context.priority === 'normal') {\n // 普通優先級延遲入隊\n const delayMs = context.config.lowPriorityDelayMs || 100\n return {\n allowed: true,\n delayed: true,\n delayMs,\n reason: 'Rate limit: normal priority delayed',\n }\n }\n\n // 低優先級拒絕\n return {\n allowed: false,\n reason: 'Rate limit exceeded for low-priority event',\n }\n }\n}\n\n/**\n * 優先級平衡策略。\n *\n * 在 WARNING 和 CRITICAL 狀態下降級低優先級事件,防止高優先級飢餓。\n */\nexport class PriorityRebalanceStrategy implements FlowControlStrategy {\n readonly name = 'priority-rebalance'\n\n evaluate(context: FlowControlContext): BackpressureDecision {\n const state = context.state\n const priority = context.priority\n const delayMs = context.config.lowPriorityDelayMs || 100\n\n if (state === BackpressureState.CRITICAL) {\n if (priority === 'low') {\n // 低優先級在 CRITICAL 狀態拒絕\n return {\n allowed: false,\n reason: 'CRITICAL: low-priority rejected',\n }\n }\n if (priority === 'normal') {\n // 普通優先級延遲\n return {\n allowed: true,\n delayed: true,\n delayMs,\n reason: 'CRITICAL: normal priority delayed',\n }\n }\n }\n\n if (state === BackpressureState.WARNING) {\n if (priority === 'low') {\n // 低優先級在 WARNING 狀態延遲\n return {\n allowed: true,\n delayed: true,\n delayMs,\n reason: 'WARNING: low priority delayed',\n }\n }\n }\n\n return { allowed: true }\n }\n}\n\n/**\n * 反飢餓保護策略。\n *\n * 防止低優先級事件被長期壓制。\n * 當事件等待超過 starvationTimeoutMs 時,自動提升其優先級。\n */\nexport class StarvationProtectionStrategy implements FlowControlStrategy {\n readonly name = 'starvation-protection'\n\n evaluate(_context: FlowControlContext): BackpressureDecision {\n // 此策略需要在 BackpressureManager 中額外實現\n // 以便存取事件的 createdAt 時間戳\n // 暫時作為佔位符,返回允許\n return { allowed: true }\n }\n}\n\n/**\n * 組合策略。\n *\n * 組合多個策略。評估時所有子策略必須允許,否則拒絕。\n * 取最嚴格的決策(拒絕 > 延遲 > 允許)。\n */\nexport class CompositeStrategy implements FlowControlStrategy {\n readonly name: string\n private strategies: FlowControlStrategy[]\n\n constructor(name: string, strategies: FlowControlStrategy[]) {\n this.name = name\n this.strategies = strategies\n }\n\n evaluate(context: FlowControlContext): BackpressureDecision {\n let mostStrict: BackpressureDecision = { allowed: true }\n\n for (const strategy of this.strategies) {\n const decision = strategy.evaluate(context)\n\n // 取最嚴格決策:拒絕 > 延遲 > 允許\n if (!decision.allowed) {\n return decision // 任一拒絕即拒絕\n }\n\n if (decision.delayed && !mostStrict.delayed) {\n mostStrict = decision // 延遲優於允許\n }\n }\n\n return mostStrict\n }\n\n /**\n * 新增子策略。\n */\n addStrategy(strategy: FlowControlStrategy): void {\n this.strategies.push(strategy)\n }\n\n /**\n * 移除指定名稱的子策略。\n */\n removeStrategy(name: string): boolean {\n const index = this.strategies.findIndex((s) => s.name === name)\n if (index >= 0) {\n this.strategies.splice(index, 1)\n return true\n }\n return false\n }\n}\n\n/**\n * 工廠方法:建立預設流控策略組合。\n *\n * @param config 背壓配置\n * @returns 組合策略實例\n *\n * @public\n */\nexport function createDefaultStrategies(\n _config: Omit<BackpressureConfig, 'onRejected' | 'onStateChange'>\n): FlowControlStrategy[] {\n return [new QueueDepthStrategy(), new RateLimitStrategy(), new PriorityRebalanceStrategy()]\n}\n",
|
|
57
|
+
"/**\n * Idempotency cache for deduplicating events.\n * Prevents duplicate events from being processed within a configurable TTL window.\n * @internal\n */\nexport class IdempotencyCache {\n private cache: Map<string, { timestamp: number }> = new Map()\n private cleanupInterval: NodeJS.Timer | null = null\n private readonly defaultCleanupIntervalMs = 60000 // 1 minute\n\n constructor() {\n // Start periodic cleanup\n this.startCleanup()\n }\n\n /**\n * Check if an event with the given idempotency key is a duplicate.\n * @param key - Idempotency key\n * @param ttlMs - Time-to-live in milliseconds\n * @returns True if this is a duplicate, false if it's a new event\n */\n isDuplicate(key: string, ttlMs: number): boolean {\n if (!key) {\n return false\n }\n\n const entry = this.cache.get(key)\n\n if (!entry) {\n // New key, record it and return false (not a duplicate)\n this.recordEvent(key)\n return false\n }\n\n // Check if the previous entry is still within TTL\n const elapsed = Date.now() - entry.timestamp\n if (elapsed < ttlMs) {\n // Within TTL, this is a duplicate\n return true\n }\n\n // TTL expired, treat as new event\n this.recordEvent(key)\n return false\n }\n\n /**\n * Record an event in the cache.\n * @param key - Idempotency key\n */\n recordEvent(key: string): void {\n if (!key) {\n return\n }\n this.cache.set(key, { timestamp: Date.now() })\n }\n\n /**\n * Remove an entry from the cache.\n * @param key - Idempotency key\n * @returns True if entry was removed, false if not found\n */\n remove(key: string): boolean {\n if (!key) {\n return false\n }\n return this.cache.delete(key)\n }\n\n /**\n * Clear all entries from the cache.\n */\n clear(): void {\n this.cache.clear()\n }\n\n /**\n * Get the current cache size.\n * @returns Number of entries in cache\n */\n getSize(): number {\n return this.cache.size\n }\n\n /**\n * Start periodic cleanup of expired entries.\n * @internal\n */\n private startCleanup(): void {\n if (this.cleanupInterval) {\n return // Already running\n }\n\n this.cleanupInterval = setInterval(() => {\n this.cleanup()\n }, this.defaultCleanupIntervalMs)\n\n // Prevent Node.js from exiting due to this interval\n if (this.cleanupInterval.unref) {\n this.cleanupInterval.unref()\n }\n }\n\n /**\n * Stop the periodic cleanup.\n * @internal\n */\n stopCleanup(): void {\n if (this.cleanupInterval) {\n clearInterval(this.cleanupInterval)\n this.cleanupInterval = null\n }\n }\n\n /**\n * Clean up expired entries from the cache.\n * This is called periodically and doesn't use strict TTL checking,\n * so entries older than a reasonable default (24 hours) are removed.\n * @internal\n */\n private cleanup(): void {\n const maxAge = 24 * 60 * 60 * 1000 // 24 hours\n const now = Date.now()\n\n for (const [key, entry] of this.cache.entries()) {\n if (now - entry.timestamp > maxAge) {\n this.cache.delete(key)\n }\n }\n }\n\n /**\n * Destructor to clean up resources.\n * @internal\n */\n destroy(): void {\n this.stopCleanup()\n this.clear()\n }\n}\n",
|
|
58
|
+
"/**\n * @gravito/core - Message Queue Bridge\n *\n * 橋接層,連接 HookManager 與 Bull Queue (via StreamEventBackend)\n * 支持分佈式事件流:HookManager → Bull Queue → Worker → 完成/DLQ\n */\n\nimport type { HookManager } from '../HookManager'\nimport type { DeadLetterQueueManager } from '../reliability/DeadLetterQueueManager'\nimport type { EventBackend } from './EventBackend'\nimport type { EventTask } from './types'\nimport type { WorkerPool } from './WorkerPool'\n\n/**\n * 事件執行狀態\n */\nexport interface EventStatus {\n eventId: string\n status: 'pending' | 'processing' | 'completed' | 'failed' | 'in_dlq'\n attempts: number\n lastError?: string\n createdAt: number\n processedAt?: number\n}\n\n/**\n * MessageQueueBridge 配置\n */\nexport interface MessageQueueBridgeConfig {\n /**\n * HookManager 實例\n */\n hookManager: HookManager\n\n /**\n * Worker 線程池(可選)\n */\n workerPool?: WorkerPool\n\n /**\n * 分佈式事件後端(Bull Queue)\n */\n eventBackend: EventBackend\n\n /**\n * Dead Letter Queue 管理器\n */\n dlqManager: DeadLetterQueueManager\n\n /**\n * 是否啟用 CircuitBreaker 檢查\n * @default false\n */\n enableCircuitBreaker?: boolean\n}\n\n/**\n * MessageQueueBridge - 連接 HookManager 與 Bull Queue 的橋接層\n *\n * 功能:\n * 1. 通過 Bull Queue 分發事件(dispatchWithQueue)\n * 2. Worker 中處理隊列事件(processQueuedEvent)\n * 3. 失敗任務轉移至 DLQ(handleJobFailure)\n * 4. 查詢事件執行狀態(getEventStatus)\n *\n * @example\n * ```typescript\n * const bridge = new MessageQueueBridge({\n * hookManager,\n * eventBackend: streamEventBackend,\n * dlqManager,\n * enableCircuitBreaker: true\n * })\n *\n * // 分發事件至 Bull Queue\n * const jobId = await bridge.dispatchWithQueue('order:created', { orderId: 123 })\n *\n * // Worker 處理事件\n * await bridge.processQueuedEvent(jobId, task)\n * ```\n */\nexport class MessageQueueBridge {\n constructor(private config: MessageQueueBridgeConfig) {\n if (!config.hookManager) {\n throw new Error('MessageQueueBridge: hookManager is required')\n }\n if (!config.eventBackend) {\n throw new Error('MessageQueueBridge: eventBackend is required')\n }\n if (!config.dlqManager) {\n throw new Error('MessageQueueBridge: dlqManager is required')\n }\n }\n\n /**\n * 通過 Bull Queue 分發事件(分佈式異步處理)\n *\n * 流程:\n * 1. 驗證事件 listeners 存在\n * 2. 檢查 CircuitBreaker 狀態(如果啟用)\n * 3. 構建 EventTask\n * 4. 入隊到 eventBackend (Bull Queue)\n *\n * @param eventName - 事件名稱\n * @param args - 事件參數\n * @param options - 事件選項(可選)\n * @returns Job ID\n * @throws 如果沒有 listeners 或 CircuitBreaker 打開\n *\n * @example\n * ```typescript\n * const jobId = await bridge.dispatchWithQueue('order:created', {\n * orderId: 'ORD-123',\n * amount: 999.99\n * })\n * ```\n */\n async dispatchWithQueue<TArgs = unknown>(\n eventName: string,\n args: TArgs,\n options?: any\n ): Promise<string> {\n // 1. 驗證 listeners 存在\n const listeners = this.config.hookManager.getListeners(eventName)\n if (listeners.length === 0) {\n throw new Error(`[MessageQueueBridge] No listeners registered for event: ${eventName}`)\n }\n\n // 2. 檢查 CircuitBreaker 狀態(如果啟用)\n if (this.config.enableCircuitBreaker) {\n const breaker = (this.config.hookManager as any).getCircuitBreaker?.(eventName)\n if (breaker?.getState?.() === 'OPEN') {\n throw new Error(`[MessageQueueBridge] Circuit breaker is OPEN for event: ${eventName}`)\n }\n }\n\n // 3. 構建 EventTask\n const nowMs = Date.now()\n const task: EventTask = {\n id: `queue-${nowMs}-${Math.random().toString(36).substr(2, 9)}`,\n hook: eventName,\n args,\n callbacks: listeners,\n options: options || {},\n createdAt: nowMs,\n enqueuedAt: nowMs,\n retryCount: 0,\n }\n\n // 4. 入隊到 eventBackend (Bull Queue)\n // 返回 Job ID (由 StreamEventBackend 生成)\n const jobId = task.id\n await this.config.eventBackend.enqueue(task)\n\n console.info(`[MessageQueueBridge] Event \"${eventName}\" enqueued to Bull Queue (job: ${jobId})`)\n\n return jobId\n }\n\n /**\n * 處理隊列事件(由 Bull Queue Worker 調用)\n *\n * 重要:使用 doActionSync() 避免遞迴\n * - 不能使用 doActionAsync()(會導致事件再次入隊)\n * - Worker 中直接同步執行 callbacks\n *\n * 流程:\n * 1. 直接執行 callbacks(不使用 doActionSync,避免吞掉錯誤)\n * 2. 記錄成功或失敗\n * 3. 拋出錯誤讓 Bull Queue 處理重試\n *\n * @param jobId - Bull Queue Job ID\n * @param task - 事件任務\n * @throws 如果 callback 執行失敗,讓 Bull Queue 處理重試\n *\n * @example\n * ```typescript\n * // 在 Bull Queue Worker 中:\n * await bridge.processQueuedEvent(jobId, task)\n * ```\n */\n async processQueuedEvent(jobId: string, task: EventTask): Promise<void> {\n let hasError = false\n let lastError: Error | null = null\n\n // 直接執行 callbacks,允許錯誤拋出\n for (const callback of task.callbacks) {\n try {\n await callback(task.args)\n } catch (error) {\n hasError = true\n lastError = error as Error\n console.error(\n `[MessageQueueBridge] Callback failed for event \"${task.hook}\" (job: ${jobId}):`,\n error\n )\n }\n }\n\n if (hasError) {\n console.error(`[MessageQueueBridge] Event \"${task.hook}\" processing failed (job: ${jobId})`)\n // 拋出錯誤,讓 Bull Queue 處理重試\n // 重試耗盡後會觸發 onFailed callback\n throw lastError || new Error('Event processing failed')\n }\n\n console.info(`[MessageQueueBridge] Event \"${task.hook}\" processed successfully (job: ${jobId})`)\n }\n\n /**\n * 處理 Bull Queue 任務失敗(由 onFailed callback 調用)\n *\n * 流程:\n * 1. 記錄警告日誌\n * 2. 移至 persistent DLQ(DeadLetterQueueManager)\n * 3. 持久化到 event_dlq 表,支持延遲重試\n *\n * @param jobId - Bull Queue Job ID\n * @param task - 事件任務\n * @param error - 導致失敗的錯誤\n * @param retryCount - 已重試次數(可選)\n *\n * @example\n * ```typescript\n * // 在 Bull Queue onFailed callback 中:\n * await bridge.handleJobFailure(jobId, task, error, 3)\n * ```\n */\n async handleJobFailure(\n jobId: string,\n task: EventTask,\n error: Error,\n retryCount = 0\n ): Promise<void> {\n console.warn(\n `[MessageQueueBridge] Event \"${task.hook}\" failed after retries (job: ${jobId}), moving to DLQ`\n )\n\n try {\n // 移至 persistent DLQ\n const dlqId = await this.config.dlqManager.moveToDlq(\n task.hook,\n task.args,\n task.options,\n error,\n retryCount\n )\n\n console.info(`[MessageQueueBridge] Event moved to DLQ (dlq_id: ${dlqId})`)\n } catch (dlqError) {\n console.error(`[MessageQueueBridge] Failed to move event to DLQ:`, dlqError)\n // 即使 DLQ 失敗也不拋出,以免影響主流程\n }\n }\n\n /**\n * 查詢事件執行狀態\n *\n * 支持查詢:\n * 1. 待處理事件\n * 2. 正在處理的事件\n * 3. 已完成事件\n * 4. 失敗事件(在 DLQ 中)\n *\n * @param eventId - 事件 ID(task.id 或 jobId)\n * @returns 事件狀態\n *\n * @example\n * ```typescript\n * const status = await bridge.getEventStatus('queue-1707000000000-abc123')\n * console.log(status) // { eventId, status, attempts, ... }\n * ```\n */\n async getEventStatus(eventId: string): Promise<EventStatus> {\n // TODO: Phase 4 Task 3 - 整合 StreamEventBackend 的狀態查詢\n // 當前作為 stub 實現,返回待定狀態\n // 在完整實現中,應該查詢 Bull Queue 和 DLQ 數據庫\n\n return {\n eventId,\n status: 'pending',\n attempts: 0,\n createdAt: Date.now(),\n }\n }\n\n /**\n * 獲取 EventBackend 實例(用於高級操作)\n *\n * @returns EventBackend 實例\n * @internal\n */\n getEventBackend(): EventBackend {\n return this.config.eventBackend\n }\n\n /**\n * 獲取 DLQ 管理器實例(用於高級操作)\n *\n * @returns DeadLetterQueueManager 實例\n * @internal\n */\n getDLQManager(): DeadLetterQueueManager {\n return this.config.dlqManager\n }\n\n /**\n * 獲取 HookManager 實例(用於高級操作)\n *\n * @returns HookManager 實例\n * @internal\n */\n getHookManager(): HookManager {\n return this.config.hookManager\n }\n}\n",
|
|
59
|
+
"/**\n * @gravito/core - Retry Scheduler\n *\n * 分佈式重試排程器,使用 Bull Queue 進行異步延遲重試。\n * 支援指數回退、Redis 持久化、重試失敗回呼。\n *\n * Distributed retry scheduler using Bull Queue for async delayed retries.\n * Supports exponential backoff, Redis persistence, and failure callbacks.\n */\n\nimport type { EventOptions } from './EventOptions'\n\n/**\n * 重試排程器配置介面\n */\nexport interface RetrySchedulerConfig {\n /** 是否啟用 Bull Queue 重試(預設 true) */\n enabled?: boolean\n\n /** Redis 連接配置(可選,使用全域 Redis) */\n redisUrl?: string\n\n /** 預設最大重試次數(預設 5) */\n maxRetries?: number\n\n /** 初始延遲時間(ms,預設 1000) */\n initialDelayMs?: number\n\n /** 指數回退倍數(預設 2.0) */\n backoffMultiplier?: number\n\n /** 最大延遲時間(ms,預設 1h) */\n maxDelayMs?: number\n\n /** 重試失敗回呼 */\n onRetryFailed?: (eventName: string, error: Error, retryCount: number) => void\n}\n\n/**\n * 隊列統計資訊\n */\nexport interface QueueStats {\n name: string\n jobCounts: { waiting: number; active: number; delayed: number; failed: number }\n completedCount: number\n failedCount: number\n}\n\n/**\n * 重試排程器\n *\n * 提供異步分佈式重試機制,利用 Bull Queue 進行延遲重試。\n * 支援指數回退、Redis 持久化、隊列統計。\n */\nexport class RetryScheduler {\n private enabled: boolean\n private config: Required<RetrySchedulerConfig>\n private queues: Map<string, unknown> = new Map() // Queue instances\n // biome-ignore lint/suspicious/noExplicitAny: bullmq is optional peer dependency\n private bullmqModule: any = null\n private bullmqLoadError: Error | null = null\n\n constructor(config: RetrySchedulerConfig = {}) {\n this.config = {\n enabled: config.enabled ?? true,\n redisUrl: config.redisUrl ?? process.env.REDIS_URL ?? 'redis://localhost:6379',\n maxRetries: config.maxRetries ?? 5,\n initialDelayMs: config.initialDelayMs ?? 1000,\n backoffMultiplier: config.backoffMultiplier ?? 2.0,\n maxDelayMs: config.maxDelayMs ?? 3600000, // 1 hour\n onRetryFailed: config.onRetryFailed ?? (() => {}),\n }\n\n this.enabled = this.config.enabled\n\n // 嘗試動態加載 bullmq(可選 peerDependency)\n if (this.enabled) {\n this.loadBullmq()\n }\n }\n\n /**\n * 動態加載 bullmq 模組\n */\n private loadBullmq(): void {\n try {\n // 使用動態 require 讓 bullmq 成為可選依賴\n // eslint-disable-next-line @typescript-eslint/no-explicit-any, @typescript-eslint/no-require-imports, @typescript-eslint/no-var-requires\n // biome-ignore lint: Optional peer dependency\n const bullmq = require('bullmq') as any\n this.bullmqModule = bullmq\n } catch (error) {\n this.enabled = false\n this.bullmqLoadError = error instanceof Error ? error : new Error(String(error))\n console.warn(\n '[RetryScheduler] bullmq 模組未安裝,已禁用 Bull Queue 重試',\n this.bullmqLoadError.message\n )\n }\n }\n\n /**\n * 檢查排程器是否啟用\n */\n isEnabled(): boolean {\n return this.enabled && this.bullmqModule !== null\n }\n\n /**\n * 計算指數回退延遲時間\n *\n * @param retryCount 當前重試次數(從 0 開始)\n * @returns 延遲時間(毫秒)\n */\n private calculateDelay(retryCount: number): number {\n const baseDelay = this.config.initialDelayMs * this.config.backoffMultiplier ** retryCount\n return Math.min(baseDelay, this.config.maxDelayMs)\n }\n\n /**\n * 獲取或創建隊列\n *\n * @param eventName 事件名稱\n * @returns Queue 實例\n */\n private getOrCreateQueue(eventName: string): unknown {\n const queueName = `gravito:event:retries:${eventName}`\n\n if (this.queues.has(queueName)) {\n return this.queues.get(queueName)!\n }\n\n if (!this.bullmqModule) {\n throw new Error('bullmq 模組未加載')\n }\n\n const { Queue } = this.bullmqModule\n const connection = { url: this.config.redisUrl }\n\n const queue = new Queue(queueName, { connection })\n this.queues.set(queueName, queue)\n\n return queue\n }\n\n /**\n * 排程重試任務\n *\n * @param eventName 事件名稱\n * @param payload 事件負載\n * @param options 事件選項\n * @param error 原始錯誤\n * @param retryCount 當前重試次數\n */\n async scheduleRetry(\n eventName: string,\n payload: unknown,\n _options: EventOptions,\n error: Error,\n retryCount: number\n ): Promise<void> {\n if (!this.isEnabled()) {\n throw new Error('RetryScheduler 未啟用或 bullmq 模組不可用')\n }\n\n try {\n const queue = this.getOrCreateQueue(eventName)\n const delay = this.calculateDelay(retryCount)\n\n // 使用反射呼叫 add() 方法(因為 Queue 類型未導出)\n const addMethod = (queue as any).add\n if (typeof addMethod === 'function') {\n await addMethod.call(\n queue,\n 'retry',\n { payload, error: error.message, retryCount },\n { delay }\n )\n }\n } catch (schedulerError) {\n const err =\n schedulerError instanceof Error ? schedulerError : new Error(String(schedulerError))\n console.error(`[RetryScheduler] 排程重試失敗: ${eventName}`, err)\n throw err\n }\n }\n\n /**\n * 獲取特定事件的隊列\n *\n * @param eventName 事件名稱\n * @returns Queue 實例或 undefined\n */\n getQueue(eventName: string): unknown {\n const queueName = `gravito:event:retries:${eventName}`\n return this.queues.get(queueName)\n }\n\n /**\n * 取得所有隊列統計\n */\n getStats(): Map<string, QueueStats> {\n const stats = new Map<string, QueueStats>()\n\n for (const [queueName, queue] of this.queues.entries()) {\n const queueObj = queue as any\n\n stats.set(queueName, {\n name: queueName,\n jobCounts: {\n waiting: queueObj.count?.waiting ?? 0,\n active: queueObj.count?.active ?? 0,\n delayed: queueObj.count?.delayed ?? 0,\n failed: queueObj.count?.failed ?? 0,\n },\n completedCount: queueObj.count?.completed ?? 0,\n failedCount: queueObj.count?.failed ?? 0,\n })\n }\n\n return stats\n }\n\n /**\n * 關閉所有隊列並清理資源\n */\n async shutdown(): Promise<void> {\n const closePromises: Promise<void>[] = []\n\n for (const queue of this.queues.values()) {\n const queueObj = queue as any\n if (typeof queueObj.close === 'function') {\n closePromises.push(queueObj.close())\n }\n }\n\n await Promise.all(closePromises)\n this.queues.clear()\n }\n}\n",
|
|
60
|
+
"/**\n * Worker Pool Metrics for OpenTelemetry integration.\n *\n * Tracks worker pool statistics and emits metrics using OpenTelemetry.\n * Follows the Provider Callback pattern for dynamic metric collection.\n *\n * @internal\n */\n\nimport type { Counter, Histogram, Meter } from '@opentelemetry/api'\n\n/**\n * Worker Pool Metrics Recorder\n */\nexport class WorkerPoolMetrics {\n private taskDurationHistogram?: Histogram\n private taskCounter?: Counter\n private autoScalingCounter?: Counter\n private poolSizeCallback?: () => number\n private utilizationCallback?: () => number\n private queueDepthCallback?: () => number\n\n constructor(private meter: Meter) {\n this.initializeMetrics()\n }\n\n /**\n * Initialize OpenTelemetry metrics\n */\n private initializeMetrics(): void {\n // Histogram for task execution duration (in milliseconds)\n this.taskDurationHistogram = this.meter.createHistogram('gravito_worker_task_duration_ms', {\n description: 'Duration of task execution in milliseconds',\n unit: 'ms',\n })\n\n // Counter for task execution\n this.taskCounter = this.meter.createCounter('gravito_worker_task_total', {\n description: 'Total number of tasks processed',\n })\n\n // Counter for auto-scaling events\n this.autoScalingCounter = this.meter.createCounter('gravito_worker_autoscale_total', {\n description: 'Total number of auto-scaling events',\n })\n\n // Note: Observable Gauges for pool metrics are not created here as they require\n // callback registration which is deferred to when setPoolSizeProvider/etc are called\n }\n\n /**\n * Set provider callback for pool size metric\n */\n setPoolSizeProvider(callback: () => number): void {\n this.poolSizeCallback = callback\n }\n\n /**\n * Set provider callback for utilization metric\n */\n setUtilizationProvider(callback: () => number): void {\n this.utilizationCallback = callback\n }\n\n /**\n * Set provider callback for queue depth metric\n */\n setQueueDepthProvider(callback: () => number): void {\n this.queueDepthCallback = callback\n }\n\n /**\n * Record task execution\n *\n * @param durationMs - Task execution duration in milliseconds\n * @param priority - Task priority level\n * @param success - Whether task succeeded\n */\n recordTaskExecution(durationMs: number, priority: string, success: boolean): void {\n const status = success ? 'success' : 'failure'\n\n if (this.taskDurationHistogram) {\n this.taskDurationHistogram.record(durationMs, {\n priority,\n status,\n })\n }\n\n if (this.taskCounter) {\n this.taskCounter.add(1, {\n priority,\n status,\n })\n }\n }\n\n /**\n * Record auto-scaling event\n *\n * @param action - 'scale-up' or 'scale-down'\n * @param reason - Reason for scaling\n */\n recordAutoScaling(action: 'scale-up' | 'scale-down', reason: string): void {\n if (this.autoScalingCounter) {\n this.autoScalingCounter.add(1, {\n action,\n reason,\n })\n }\n }\n\n /**\n * Get the current pool size (for provider callbacks)\n */\n getPoolSize(): number {\n return this.poolSizeCallback?.() ?? 0\n }\n\n /**\n * Get the current utilization (for provider callbacks)\n */\n getUtilization(): number {\n return this.utilizationCallback?.() ?? 0\n }\n\n /**\n * Get the current queue depth (for provider callbacks)\n */\n getQueueDepth(): number {\n return this.queueDepthCallback?.() ?? 0\n }\n}\n",
|
|
61
|
+
"/**\n * Worker Pool for concurrent event task processing.\n *\n * Manages a pool of workers for executing event tasks concurrently.\n * Supports auto-scaling, health checks, and OpenTelemetry metrics integration.\n *\n * @internal\n */\n\nimport type { Meter } from '@opentelemetry/api'\nimport { randomUUID } from '../compat/crypto'\nimport type { EventTask } from './types'\nimport type { TaskSource, WorkerPoolConfig, WorkerPoolStats, WorkerStats } from './WorkerPoolConfig'\nimport { WorkerPoolMetrics } from './WorkerPoolMetrics'\n\n/**\n * Internal Worker representation\n */\ninterface Worker {\n id: string\n state: 'idle' | 'busy' | 'terminated'\n tasksProcessed: number\n tasksSuccess: number\n tasksFailed: number\n totalDurationMs: number\n createdAt: number\n}\n\n/**\n * Worker Pool for managing concurrent task execution\n */\nexport class WorkerPool {\n private workers: Worker[] = []\n private taskQueue: EventTask[] = []\n private config: {\n concurrency: number\n workerThreads: number\n taskTimeout: number\n maxRetries: number\n enableAutoScaling: boolean\n minWorkers: number\n maxWorkers: number\n scaleUpThreshold: number\n scaleDownThreshold: number\n metricsInterval: number\n taskSource?: TaskSource\n }\n private metrics?: WorkerPoolMetrics\n private healthCheckTimer?: any\n private metricsTimer?: any\n private isRunning = false\n private scaleDownCounter = 0\n\n constructor(config: WorkerPoolConfig = {}, meter?: Meter) {\n // Set defaults\n let cpuCores = 1\n try {\n if (typeof process !== 'undefined' && !(process as any).browser) {\n // biome-ignore lint/security/noGlobalEval: hide from Vite\n cpuCores = Math.max(1, eval('require')('node:os').cpus().length)\n }\n } catch (_e) {\n // Fallback to 1\n }\n\n this.config = {\n concurrency: config.concurrency ?? 4,\n workerThreads: config.workerThreads ?? cpuCores,\n taskTimeout: config.taskTimeout ?? 30000,\n maxRetries: config.maxRetries ?? 3,\n enableAutoScaling: config.enableAutoScaling ?? true,\n minWorkers: config.minWorkers ?? 1,\n maxWorkers: config.maxWorkers ?? cpuCores * 2,\n scaleUpThreshold: config.scaleUpThreshold ?? 0.8,\n scaleDownThreshold: config.scaleDownThreshold ?? 0.2,\n metricsInterval: config.metricsInterval ?? 5000,\n taskSource: config.taskSource,\n }\n\n if (meter) {\n this.metrics = new WorkerPoolMetrics(meter)\n // Set up provider callbacks\n this.metrics.setPoolSizeProvider(() => this.getActiveWorkerCount())\n this.metrics.setUtilizationProvider(() => this.getUtilization())\n this.metrics.setQueueDepthProvider(() => this.taskQueue.length)\n }\n }\n\n /**\n * Start the worker pool\n */\n async start(): Promise<void> {\n if (this.isRunning) {\n return\n }\n\n this.isRunning = true\n\n // Initialize workers\n for (let i = 0; i < this.config.minWorkers; i++) {\n this.createWorker()\n }\n\n // Start health check\n this.healthCheckTimer = setInterval(\n () => this.performHealthCheck(),\n this.config.metricsInterval\n )\n\n // Start metrics collection\n this.metricsTimer = setInterval(() => this.collectMetrics(), this.config.metricsInterval)\n\n // Start auto-scaling if enabled\n if (this.config.enableAutoScaling) {\n setInterval(() => this.performAutoScaling(), this.config.metricsInterval)\n }\n }\n\n /**\n * Stop the worker pool\n */\n async stop(): Promise<void> {\n this.isRunning = false\n\n if (this.healthCheckTimer) {\n clearInterval(this.healthCheckTimer)\n }\n\n if (this.metricsTimer) {\n clearInterval(this.metricsTimer)\n }\n\n // Mark all workers as terminated\n for (const worker of this.workers) {\n worker.state = 'terminated'\n }\n\n // Wait for queue to drain\n const maxWaitTime = 5000\n const startTime = Date.now()\n while (this.taskQueue.length > 0 && Date.now() - startTime < maxWaitTime) {\n await new Promise((resolve) => setTimeout(resolve, 100))\n }\n\n this.workers = []\n this.taskQueue = []\n }\n\n /**\n * Submit a task to the worker pool\n */\n async submitTask(task: EventTask): Promise<void> {\n if (!this.isRunning) {\n throw new Error('Worker pool is not running')\n }\n\n this.taskQueue.push(task)\n this.processQueue()\n }\n\n /**\n * Process next task in queue\n */\n private processQueue(): void {\n if (!this.isRunning) {\n return\n }\n\n for (const worker of this.workers) {\n if (worker.state === 'idle' && this.taskQueue.length > 0) {\n const task = this.taskQueue.shift()\n if (task) {\n this.executeTask(worker, task)\n }\n }\n }\n }\n\n /**\n * Execute task on a worker\n */\n private async executeTask(worker: Worker, task: EventTask): Promise<void> {\n worker.state = 'busy'\n const startTime = performance.now()\n\n try {\n // Execute task callbacks\n await this.executeTaskCallbacks(task)\n\n worker.tasksSuccess++\n\n // Acknowledge completion to task source\n if (this.config.taskSource) {\n await this.config.taskSource.acknowledgeCompletion(task.id)\n }\n\n const duration = performance.now() - startTime\n this.metrics?.recordTaskExecution(duration, task.options.priority || 'normal', true)\n } catch (error) {\n worker.tasksFailed++\n\n // Acknowledge failure to task source\n if (this.config.taskSource && error instanceof Error) {\n await this.config.taskSource.acknowledgeFailure(task.id, error)\n }\n\n const duration = performance.now() - startTime\n this.metrics?.recordTaskExecution(duration, task.options.priority || 'normal', false)\n } finally {\n worker.tasksProcessed++\n worker.totalDurationMs += performance.now() - startTime\n\n worker.state = 'idle'\n\n // Process next task\n setImmediate(() => this.processQueue())\n }\n }\n\n /**\n * Execute task callbacks\n */\n private async executeTaskCallbacks(task: EventTask): Promise<void> {\n for (const callback of task.callbacks) {\n await Promise.race([callback(task.args), this.createTimeout(this.config.taskTimeout)])\n }\n }\n\n /**\n * Create a timeout promise\n */\n private createTimeout(ms: number): Promise<never> {\n return new Promise((_, reject) => {\n setTimeout(() => {\n reject(new Error(`Task timeout after ${ms}ms`))\n }, ms)\n })\n }\n\n /**\n * Create a new worker\n */\n private createWorker(): Worker {\n const worker: Worker = {\n id: randomUUID(),\n state: 'idle',\n tasksProcessed: 0,\n tasksSuccess: 0,\n tasksFailed: 0,\n totalDurationMs: 0,\n createdAt: Date.now(),\n }\n this.workers.push(worker)\n return worker\n }\n\n /**\n * Remove a worker\n */\n private removeWorker(workerId: string): void {\n const index = this.workers.findIndex((w) => w.id === workerId)\n if (index !== -1) {\n this.workers[index].state = 'terminated'\n this.workers.splice(index, 1)\n }\n }\n\n /**\n * Perform health check and cleanup\n */\n private performHealthCheck(): void {\n // Remove terminated workers\n this.workers = this.workers.filter((w) => w.state !== 'terminated')\n\n // If no workers, create minimum\n if (this.workers.length === 0 && this.isRunning) {\n for (let i = 0; i < this.config.minWorkers; i++) {\n this.createWorker()\n }\n }\n }\n\n /**\n * Perform auto-scaling based on load\n */\n private async performAutoScaling(): Promise<void> {\n if (!this.config.enableAutoScaling) {\n return\n }\n\n const utilization = this.getUtilization()\n const activeWorkers = this.getActiveWorkerCount()\n\n // Scale up if utilization is high\n if (utilization > this.config.scaleUpThreshold && activeWorkers < this.config.maxWorkers) {\n this.createWorker()\n this.metrics?.recordAutoScaling('scale-up', `Utilization: ${(utilization * 100).toFixed(1)}%`)\n this.scaleDownCounter = 0 // Reset scale down counter\n }\n // Scale down if utilization is low (with delay)\n else if (\n utilization < this.config.scaleDownThreshold &&\n activeWorkers > this.config.minWorkers\n ) {\n this.scaleDownCounter++\n if (this.scaleDownCounter >= 3) {\n const workerToRemove = this.workers.find((w) => w.state === 'idle')\n if (workerToRemove) {\n this.removeWorker(workerToRemove.id)\n this.metrics?.recordAutoScaling(\n 'scale-down',\n `Utilization: ${(utilization * 100).toFixed(1)}%`\n )\n }\n this.scaleDownCounter = 0\n }\n } else {\n this.scaleDownCounter = 0 // Reset if utilization is in range\n }\n }\n\n /**\n * Collect metrics\n */\n private collectMetrics(): void {\n // Metrics are collected through provider callbacks\n // This method can be extended for additional logic\n }\n\n /**\n * Get number of active workers (not terminated)\n */\n private getActiveWorkerCount(): number {\n return this.workers.filter((w) => w.state !== 'terminated').length\n }\n\n /**\n * Get worker utilization (0-1)\n */\n private getUtilization(): number {\n const busyWorkers = this.workers.filter((w) => w.state === 'busy').length\n const totalWorkers = this.getActiveWorkerCount()\n\n if (totalWorkers === 0) {\n return 0\n }\n\n return busyWorkers / totalWorkers\n }\n\n /**\n * Get queue depth\n */\n getQueueDepth(): number {\n return this.taskQueue.length\n }\n\n /**\n * Get worker utilization\n */\n getWorkerUtilization(): number {\n return this.getUtilization()\n }\n\n /**\n * Get worker statistics\n */\n getWorkerStats(): WorkerStats[] {\n return this.workers.map((w) => ({\n id: w.id,\n state: w.state,\n tasksProcessed: w.tasksProcessed,\n tasksSuccess: w.tasksSuccess,\n tasksFailed: w.tasksFailed,\n avgDurationMs: w.tasksProcessed > 0 ? w.totalDurationMs / w.tasksProcessed : 0,\n uptime: Date.now() - w.createdAt,\n }))\n }\n\n /**\n * Get pool statistics\n */\n getPoolStats(): WorkerPoolStats {\n const stats: WorkerPoolStats = {\n queueDepth: this.taskQueue.length,\n utilization: this.getUtilization(),\n activeWorkers: this.getActiveWorkerCount(),\n totalTasksProcessed: 0,\n totalSuccess: 0,\n totalFailures: 0,\n }\n\n for (const worker of this.workers) {\n stats.totalTasksProcessed += worker.tasksProcessed\n stats.totalSuccess += worker.tasksSuccess\n stats.totalFailures += worker.tasksFailed\n }\n\n return stats\n }\n}\n\nexport type { WorkerPoolConfig, WorkerStats, WorkerPoolStats }\n",
|
|
62
|
+
"/**\n * Event aggregation and deduplication types (FS-102)\n * @internal\n */\n\nimport { BackpressureState } from '../BackpressureManager'\n\n/**\n * Deduplication statistics.\n */\nexport interface DeduplicationStats {\n /** Total events submitted */\n totalEvents: number\n /** Events after deduplication */\n deduplicatedEvents: number\n /** Events removed by deduplication */\n removedCount: number\n /** Number of patterns aggregated */\n patternsAggregated: number\n /** Deduplication rate (%) */\n deduplicationRate: number\n}\n\n/**\n * Batch submission statistics.\n */\nexport interface BatchStats {\n /** Total batches submitted */\n totalBatches: number\n /** Total events submitted */\n totalEvents: number\n /** Average batch size */\n averageBatchSize: number\n /** Average batch latency (ms) */\n averageBatchLatency: number\n /** Last flush timestamp */\n lastFlushTime: number\n /** Pending events in queue */\n pendingEvents: number\n /** Auto-triggered flushes */\n autoFlushCount: number\n /** Manual flushes */\n manualFlushCount: number\n}\n\n/**\n * Aggregation window statistics.\n */\nexport interface WindowStats {\n /** Current window size (ms) */\n currentWindowMs: number\n /** Minimum window size (ms) */\n minWindowMs: number\n /** Maximum window size (ms) */\n maxWindowMs: number\n /** Last window adjustment reason */\n lastAdjustmentReason?: string\n /** Window adjustment count */\n adjustmentCount: number\n}\n\n/**\n * Complete aggregation statistics.\n */\nexport interface AggregationStats {\n deduplication: DeduplicationStats\n batching: BatchStats\n window: WindowStats\n timestamp: number\n}\n\n/**\n * Aggregation configuration options.\n */\nexport interface AggregationConfig {\n /** Enable aggregation (default: false) */\n enabled: boolean\n\n /** Window size in milliseconds (default: 200) */\n windowMs?: number\n\n /** Batch size threshold (default: 50) */\n batchSize?: number\n\n /** Deduplication strategy */\n deduplication?: 'pattern' | 'idempotencyKey' | 'off'\n\n /** Deduplication pattern (string or function) */\n pattern?: string | ((args: unknown) => string)\n\n /** Merge priority when deduplicating */\n mergePriority?: 'highest' | 'earliest' | 'latest'\n\n /** Enable background cleanup (default: true) */\n enableCleanup?: boolean\n\n /** Cleanup interval (ms, default: 5 minutes) */\n cleanupIntervalMs?: number\n\n /** TTL for deduplication entries (ms, default: 10 minutes) */\n ttlMs?: number\n}\n\n/**\n * Default aggregation configuration.\n */\nexport const DEFAULT_AGGREGATION_CONFIG: Required<AggregationConfig> = {\n enabled: false,\n windowMs: 200,\n batchSize: 50,\n deduplication: 'pattern',\n pattern: (args: unknown) => {\n if (!args || typeof args !== 'object') {\n return 'default'\n }\n const keys = Object.keys(args as Record<string, unknown>).sort()\n return keys.join(':')\n },\n mergePriority: 'highest',\n enableCleanup: true,\n cleanupIntervalMs: 5 * 60 * 1000, // 5 minutes\n ttlMs: 10 * 60 * 1000, // 10 minutes\n}\n\n/**\n * Priority order for merge decisions.\n */\nexport const PRIORITY_ORDER: Record<string, number> = {\n critical: 0,\n high: 1,\n normal: 2,\n low: 3,\n}\n\n/**\n * Backpressure-aware window configuration.\n */\nexport interface WindowAdjustmentConfig {\n /** Window for NORMAL backpressure state */\n normalMs: number\n /** Window for WARNING backpressure state */\n warningMs: number\n /** Window for CRITICAL backpressure state */\n criticalMs: number\n /** Window for OVERFLOW backpressure state */\n overflowMs: number\n}\n\n/**\n * Default window adjustment configuration based on backpressure.\n */\nexport const DEFAULT_WINDOW_ADJUSTMENT: WindowAdjustmentConfig = {\n normalMs: 200,\n warningMs: 150,\n criticalMs: 100,\n overflowMs: 50,\n}\n\n/**\n * Get suggested window size based on backpressure state.\n */\nexport function getSuggestedWindow(\n state: BackpressureState,\n config: WindowAdjustmentConfig = DEFAULT_WINDOW_ADJUSTMENT\n): number {\n switch (state) {\n case BackpressureState.NORMAL:\n return config.normalMs\n case BackpressureState.WARNING:\n return config.warningMs\n case BackpressureState.CRITICAL:\n return config.criticalMs\n case BackpressureState.OVERFLOW:\n return config.overflowMs\n default:\n return config.normalMs\n }\n}\n",
|
|
63
|
+
"/**\n * Aggregation window with backpressure awareness (FS-102)\n *\n * Adjusts batching window based on backpressure state:\n * - NORMAL: 200ms (optimal aggregation)\n * - WARNING: 150ms (accelerate processing)\n * - CRITICAL: 100ms (fast drain)\n * - OVERFLOW: 50ms (minimum latency)\n *\n * FS-103 增強:\n * - 與 BackpressureManager 雙向反饋\n * - 窗口調整通知機制\n */\n\nimport type { BackpressureManager } from '../BackpressureManager'\nimport { BackpressureState } from '../BackpressureManager'\nimport type { WindowStats } from './types'\nimport { DEFAULT_WINDOW_ADJUSTMENT } from './types'\n\n/**\n * Aggregation window manager with backpressure awareness.\n */\nexport class AggregationWindow {\n // Current window size (ms)\n private currentWindowMs: number\n\n // Minimum and maximum bounds\n private readonly minWindowMs: number = 50\n private readonly maxWindowMs: number = 500\n\n // Adjustment statistics\n private stats = {\n adjustmentCount: 0,\n lastAdjustmentReason: '' as string | undefined,\n }\n\n // FS-103:BackpressureManager 引用用於反饋\n private backpressureManager?: BackpressureManager\n\n /**\n * Create an aggregation window.\n *\n * @param initialWindowMs - Initial window size (default 200ms)\n */\n constructor(initialWindowMs = 200) {\n this.currentWindowMs = Math.max(this.minWindowMs, Math.min(this.maxWindowMs, initialWindowMs))\n }\n\n /**\n * Set the BackpressureManager for feedback loop (FS-103).\n *\n * @param manager - BackpressureManager instance\n */\n setBackpressureManager(manager: BackpressureManager): void {\n this.backpressureManager = manager\n }\n\n /**\n * Adjust window based on backpressure state.\n *\n * @param state - Current backpressure state\n */\n adjustWindow(state: BackpressureState): number {\n const oldWindow = this.currentWindowMs\n let newWindow = this.currentWindowMs\n let reason = 'unknown'\n\n switch (state) {\n case BackpressureState.NORMAL:\n newWindow = DEFAULT_WINDOW_ADJUSTMENT.normalMs\n reason = 'normal_backpressure'\n break\n case BackpressureState.WARNING:\n newWindow = DEFAULT_WINDOW_ADJUSTMENT.warningMs\n reason = 'warning_backpressure'\n break\n case BackpressureState.CRITICAL:\n newWindow = DEFAULT_WINDOW_ADJUSTMENT.criticalMs\n reason = 'critical_backpressure'\n break\n case BackpressureState.OVERFLOW:\n newWindow = DEFAULT_WINDOW_ADJUSTMENT.overflowMs\n reason = 'overflow_backpressure'\n break\n default:\n reason = `unknown_state_${String(state)}`\n }\n\n // Ensure bounds\n newWindow = Math.max(this.minWindowMs, Math.min(this.maxWindowMs, newWindow))\n\n // Update if changed\n if (newWindow !== oldWindow) {\n this.currentWindowMs = newWindow\n this.stats.adjustmentCount++\n this.stats.lastAdjustmentReason = `${reason} (${oldWindow}ms → ${newWindow}ms)`\n\n // === FS-103:Notify BackpressureManager of window adjustment ===\n this.notifyBackpressureManager(oldWindow, newWindow)\n }\n\n return this.currentWindowMs\n }\n\n /**\n * Notify BackpressureManager of window adjustment (FS-103).\n * Part of the backpressure feedback loop for automatic state recovery.\n *\n * @param oldWindowMs - Previous window size\n * @param newWindowMs - New window size\n * @private\n */\n private notifyBackpressureManager(oldWindowMs: number, newWindowMs: number): void {\n if (!this.backpressureManager) {\n return\n }\n\n this.backpressureManager.notifyWindowAdjustment(oldWindowMs, newWindowMs)\n }\n\n /**\n * Get current window size.\n */\n getCurrentWindow(): number {\n return this.currentWindowMs\n }\n\n /**\n * Get minimum window size.\n */\n getMinWindow(): number {\n return this.minWindowMs\n }\n\n /**\n * Get maximum window size.\n */\n getMaxWindow(): number {\n return this.maxWindowMs\n }\n\n /**\n * Get statistics.\n */\n getStats(): WindowStats {\n return {\n currentWindowMs: this.currentWindowMs,\n minWindowMs: this.minWindowMs,\n maxWindowMs: this.maxWindowMs,\n lastAdjustmentReason: this.stats.lastAdjustmentReason,\n adjustmentCount: this.stats.adjustmentCount,\n }\n }\n\n /**\n * Reset statistics.\n */\n resetStats(): void {\n this.stats.adjustmentCount = 0\n this.stats.lastAdjustmentReason = undefined\n }\n\n /**\n * Reset window to initial size.\n */\n reset(windowMs?: number): void {\n this.currentWindowMs = Math.max(\n this.minWindowMs,\n Math.min(this.maxWindowMs, windowMs ?? DEFAULT_WINDOW_ADJUSTMENT.normalMs)\n )\n this.resetStats()\n }\n}\n",
|
|
64
|
+
"/**\n * Event deduplication manager (FS-102)\n *\n * Merges events with the same pattern/idempotency key to reduce duplicates.\n * - Pattern-based deduplication\n * - Priority-aware merge strategy\n * - Wildcard pattern aggregation\n * - Deduplication rate tracking\n *\n * Ported from: examples/flash-sale-fullstack/src/cache/events/EventDeduplicator.ts\n */\n\nimport type { EventTask } from '../types'\nimport type { AggregationConfig, DeduplicationStats } from './types'\nimport { DEFAULT_AGGREGATION_CONFIG, PRIORITY_ORDER } from './types'\n\n/**\n * Event deduplication manager using pattern matching.\n */\nexport class DeduplicationManager {\n // Pattern -> EventTask mapping for deduplication\n private patternMap: Map<string, EventTask> = new Map()\n\n // Statistics\n private stats: DeduplicationStats = {\n totalEvents: 0,\n deduplicatedEvents: 0,\n removedCount: 0,\n patternsAggregated: 0,\n deduplicationRate: 0,\n }\n\n // Pattern regex cache\n private regexCache = new Map<string, RegExp>()\n\n // Cached deduplicated result\n private cachedDeduplicated: EventTask[] | null = null\n private deduplicatedDirty = true\n\n // Configuration\n private config: Required<AggregationConfig>\n\n // TTL tracking for cleanup\n private patternTimestamps: Map<string, number> = new Map()\n\n // Cleanup timer\n private cleanupTimer: NodeJS.Timeout | null = null\n\n /**\n * Create a deduplication manager.\n */\n constructor(config?: Partial<AggregationConfig>) {\n this.config = {\n ...DEFAULT_AGGREGATION_CONFIG,\n ...config,\n }\n\n if (this.config.enableCleanup) {\n this.startCleanup()\n }\n }\n\n /**\n * Add an event for deduplication.\n *\n * Rules:\n * 1. Events with same pattern are merged into one\n * 2. Higher priority events are kept, lower priority removed\n * 3. If new event has higher priority, it replaces old event\n */\n addEvent(event: EventTask): void {\n this.stats.totalEvents++\n this.deduplicatedDirty = true\n\n // Generate deduplication pattern\n const pattern = this.generatePattern(event)\n\n const existing = this.patternMap.get(pattern)\n\n if (!existing) {\n // New pattern, add directly\n this.patternMap.set(pattern, event)\n this.patternTimestamps.set(pattern, Date.now())\n } else {\n // Pattern exists, compare priorities\n const existingPriority = PRIORITY_ORDER[existing.options.priority ?? 'normal'] ?? 2\n const newPriority = PRIORITY_ORDER[event.options.priority ?? 'normal'] ?? 2\n\n if (newPriority < existingPriority) {\n // New event has higher priority, replace\n this.patternMap.set(pattern, event)\n this.patternTimestamps.set(pattern, Date.now())\n this.stats.removedCount++\n } else if (newPriority === existingPriority) {\n // Same priority, keep the earliest\n if (event.createdAt < existing.createdAt) {\n this.patternMap.set(pattern, event)\n this.patternTimestamps.set(pattern, Date.now())\n }\n this.stats.removedCount++\n } else {\n // Existing event has higher priority, discard new event\n this.stats.removedCount++\n }\n }\n\n // Update statistics\n this.stats.deduplicatedEvents = this.patternMap.size\n this.stats.patternsAggregated = this.patternMap.size\n this.updateDeduplicationRate()\n }\n\n /**\n * Add multiple events.\n */\n addEvents(events: EventTask[]): void {\n for (const event of events) {\n this.addEvent(event)\n }\n }\n\n /**\n * Get all deduplicated events.\n * Result is cached until next addEvent() call.\n */\n getDeduplicated(): EventTask[] {\n // Return cached result if valid\n if (!this.deduplicatedDirty && this.cachedDeduplicated !== null) {\n return this.cachedDeduplicated\n }\n\n // Build deduplication result\n const eventMap = new Map<string, EventTask>()\n\n for (const event of this.patternMap.values()) {\n const existing = eventMap.get(event.id)\n\n if (!existing) {\n // First occurrence of this event ID\n eventMap.set(event.id, { ...event })\n }\n }\n\n // Cache result\n this.cachedDeduplicated = Array.from(eventMap.values())\n this.deduplicatedDirty = false\n\n return this.cachedDeduplicated\n }\n\n /**\n * Optimize patterns by merging overlapping patterns.\n *\n * Example: ['product:1', 'product:2', 'product:3'] -> ['product:*']\n */\n optimizePatterns(patterns: string[]): string[] {\n if (patterns.length < 3) {\n return patterns\n }\n\n const optimized: string[] = []\n const byPrefix = new Map<string, string[]>()\n\n // Group by prefix\n for (const pattern of patterns) {\n const parts = pattern.split(':')\n const prefix = parts[0]\n\n if (!byPrefix.has(prefix)) {\n byPrefix.set(prefix, [])\n }\n byPrefix.get(prefix)?.push(pattern)\n }\n\n // If a prefix has >50% of patterns, use wildcard\n const threshold = Math.ceil(patterns.length * 0.5)\n\n for (const [prefix, patternList] of byPrefix.entries()) {\n if (patternList.length >= threshold) {\n // Use wildcard\n optimized.push(`${prefix}:*`)\n } else {\n // Keep original patterns\n optimized.push(...patternList)\n }\n }\n\n return optimized\n }\n\n /**\n * Clear the deduplication manager.\n */\n clear(): void {\n this.patternMap.clear()\n this.patternTimestamps.clear()\n this.cachedDeduplicated = null\n this.deduplicatedDirty = true\n this.regexCache.clear()\n this.stats.totalEvents = 0\n this.stats.deduplicatedEvents = 0\n this.stats.removedCount = 0\n this.stats.patternsAggregated = 0\n this.stats.deduplicationRate = 0\n }\n\n /**\n * Get statistics.\n */\n getStats(): DeduplicationStats {\n return { ...this.stats }\n }\n\n /**\n * Reset statistics (but keep deduplication data).\n */\n resetStats(): void {\n this.stats.totalEvents = 0\n this.stats.removedCount = 0\n this.stats.deduplicationRate = 0\n this.updateDeduplicationRate()\n }\n\n /**\n * Get deduplication rate percentage.\n */\n getDeduplicationRate(): number {\n return this.stats.deduplicationRate\n }\n\n /**\n * Get pattern count.\n */\n getPatternCount(): number {\n return this.patternMap.size\n }\n\n /**\n * Get event for specific pattern.\n */\n getEventForPattern(pattern: string): EventTask | undefined {\n return this.patternMap.get(pattern)\n }\n\n /**\n * Check if pattern exists.\n */\n hasPattern(pattern: string): boolean {\n return this.patternMap.has(pattern)\n }\n\n /**\n * Remove specific pattern.\n */\n removePattern(pattern: string): boolean {\n const removed = this.patternMap.delete(pattern)\n if (removed) {\n this.patternTimestamps.delete(pattern)\n this.deduplicatedDirty = true\n }\n return removed\n }\n\n /**\n * Remove patterns by prefix.\n */\n removePatternByPrefix(prefix: string): number {\n let removed = 0\n const toDelete: string[] = []\n\n for (const pattern of this.patternMap.keys()) {\n if (pattern.startsWith(prefix)) {\n toDelete.push(pattern)\n }\n }\n\n for (const pattern of toDelete) {\n this.patternMap.delete(pattern)\n this.patternTimestamps.delete(pattern)\n removed++\n }\n\n if (removed > 0) {\n this.deduplicatedDirty = true\n }\n\n return removed\n }\n\n /**\n * Get all patterns.\n */\n getPatterns(): string[] {\n return Array.from(this.patternMap.keys())\n }\n\n /**\n * Find events matching invalidation pattern.\n * Supports wildcard patterns.\n */\n findEventsByPattern(invalidationPattern: string): EventTask[] {\n const result: Set<EventTask> = new Set()\n\n // Exact match\n const exact = this.patternMap.get(invalidationPattern)\n if (exact) {\n result.add(exact)\n }\n\n // Wildcard match\n if (invalidationPattern.includes('*')) {\n const regex = this.getPatternRegex(invalidationPattern)\n for (const [pattern, event] of this.patternMap.entries()) {\n if (regex.test(pattern)) {\n result.add(event)\n }\n }\n }\n\n return Array.from(result)\n }\n\n /**\n * Shutdown the manager (cleanup resources).\n */\n shutdown(): void {\n this.stopCleanup()\n this.clear()\n }\n\n /**\n * Generate deduplication pattern from event.\n */\n private generatePattern(event: EventTask): string {\n if (this.config.deduplication === 'idempotencyKey' && event.options.idempotencyKey) {\n return `idempotency:${event.options.idempotencyKey}`\n }\n\n if (typeof this.config.pattern === 'function') {\n try {\n return this.config.pattern(event.args) ?? 'default'\n } catch (_e) {\n return `${event.hook}:default`\n }\n }\n\n if (typeof this.config.pattern === 'string') {\n // Use hook and pattern as base\n return `${event.hook}:${this.config.pattern}`\n }\n\n // Default: use hook as pattern\n return event.hook\n }\n\n /**\n * Get pattern regex (cached).\n */\n private getPatternRegex(pattern: string): RegExp {\n if (!this.regexCache.has(pattern)) {\n this.regexCache.set(pattern, this.patternToRegex(pattern))\n }\n return this.regexCache.get(pattern)!\n }\n\n /**\n * Convert pattern to regex.\n */\n private patternToRegex(pattern: string): RegExp {\n const escaped = pattern.replace(/[.+^${}()|[\\]\\\\]/g, '\\\\$&').replace(/\\*/g, '.*')\n return new RegExp(`^${escaped}$`)\n }\n\n /**\n * Update deduplication rate.\n */\n private updateDeduplicationRate(): void {\n if (this.stats.totalEvents === 0) {\n this.stats.deduplicationRate = 0\n } else {\n this.stats.deduplicationRate = Math.round(\n (this.stats.removedCount / this.stats.totalEvents) * 100\n )\n }\n }\n\n /**\n * Start automatic cleanup timer.\n */\n private startCleanup(): void {\n this.cleanupTimer = setInterval(() => this.performCleanup(), this.config.cleanupIntervalMs)\n }\n\n /**\n * Stop cleanup timer.\n */\n private stopCleanup(): void {\n if (this.cleanupTimer !== null) {\n clearInterval(this.cleanupTimer)\n this.cleanupTimer = null\n }\n }\n\n /**\n * Perform TTL-based cleanup.\n */\n private performCleanup(): void {\n const now = Date.now()\n const ttl = this.config.ttlMs\n\n const toDelete: string[] = []\n for (const [pattern, timestamp] of this.patternTimestamps.entries()) {\n if (now - timestamp > ttl) {\n toDelete.push(pattern)\n }\n }\n\n for (const pattern of toDelete) {\n this.patternMap.delete(pattern)\n this.patternTimestamps.delete(pattern)\n }\n\n if (toDelete.length > 0) {\n this.deduplicatedDirty = true\n this.stats.deduplicatedEvents = this.patternMap.size\n }\n }\n\n /**\n * Internal method for testing.\n */\n __getPatternMap(): Map<string, EventTask> {\n return new Map(this.patternMap)\n }\n}\n",
|
|
65
|
+
"/**\n * Event batcher for microbatching optimization (FS-102)\n *\n * Implements dual-trigger batching:\n * - Time window trigger (default 50ms)\n * - Batch size trigger (default 50 events)\n *\n * Expected improvements: 10-15% throughput increase\n *\n * Ported from: examples/flash-sale-fullstack/src/cache/events/BatchSubmitter.ts\n */\n\nimport type { EventTask } from '../types'\nimport type { BatchStats } from './types'\n\n/**\n * Event batcher for optimizing submission throughput.\n */\nexport class EventBatcher {\n // Event submission queue\n private queue: EventTask[] = []\n\n // Batch size threshold\n private readonly batchSize: number\n\n // Flush interval (ms)\n private readonly flushIntervalMs: number\n\n // Flush timer\n private flushTimer: NodeJS.Timeout | null = null\n\n // Statistics\n private stats = {\n totalBatches: 0,\n totalEvents: 0,\n totalLatency: 0,\n autoFlushCount: 0,\n manualFlushCount: 0,\n lastFlushTime: Date.now(),\n }\n\n // Submission function\n private submitFn: (tasks: EventTask[]) => Promise<void>\n\n /**\n * Create an event batcher.\n *\n * @param batchSize - Batch size threshold (default 50)\n * @param flushIntervalMs - Time window (ms, default 50)\n * @param submitFn - Function to submit batches\n */\n constructor(\n batchSize = 50,\n flushIntervalMs = 50,\n submitFn: (tasks: EventTask[]) => Promise<void>\n ) {\n this.batchSize = batchSize\n this.flushIntervalMs = flushIntervalMs\n this.submitFn = submitFn\n }\n\n /**\n * Enqueue an event.\n *\n * Automatically triggers flush if:\n * 1. Batch size is reached\n * 2. Time window expires\n */\n async enqueue(task: EventTask): Promise<void> {\n this.queue.push(task)\n\n // Trigger 1: Batch size reached\n if (this.queue.length >= this.batchSize) {\n await this.flush(true) // Mark as auto-triggered\n } else if (this.flushTimer === null && this.queue.length > 0) {\n // Trigger 2: Start time-window timer\n this.startTimer()\n }\n }\n\n /**\n * Enqueue multiple events.\n */\n async enqueueBatch(tasks: EventTask[]): Promise<void> {\n for (const task of tasks) {\n await this.enqueue(task)\n }\n }\n\n /**\n * Flush queued events.\n *\n * @param auto - Whether this is an auto-triggered flush\n * @returns Array of flushed events\n */\n async flush(auto = false): Promise<EventTask[]> {\n if (this.queue.length === 0) {\n return []\n }\n\n // Clear timer\n this.clearTimer()\n\n // Extract all events\n const tasks = this.queue.splice(0, this.queue.length)\n\n // Record statistics\n const latency = Date.now() - this.stats.lastFlushTime\n this.stats.totalBatches++\n this.stats.totalEvents += tasks.length\n this.stats.totalLatency += latency\n this.stats.lastFlushTime = Date.now()\n\n if (auto) {\n this.stats.autoFlushCount++\n } else {\n this.stats.manualFlushCount++\n }\n\n // Submit events\n try {\n await this.submitFn(tasks)\n } catch (error) {\n // Put events back if submission fails\n this.queue.unshift(...tasks)\n throw error\n }\n\n return tasks\n }\n\n /**\n * Get statistics.\n */\n getStats(): BatchStats {\n const avgBatchSize =\n this.stats.totalBatches > 0 ? this.stats.totalEvents / this.stats.totalBatches : 0\n const avgLatency =\n this.stats.totalBatches > 0 ? this.stats.totalLatency / this.stats.totalBatches : 0\n\n return {\n totalBatches: this.stats.totalBatches,\n totalEvents: this.stats.totalEvents,\n averageBatchSize: Math.round(avgBatchSize * 100) / 100,\n averageBatchLatency: Math.round(avgLatency * 100) / 100,\n lastFlushTime: this.stats.lastFlushTime,\n pendingEvents: this.queue.length,\n autoFlushCount: this.stats.autoFlushCount,\n manualFlushCount: this.stats.manualFlushCount,\n }\n }\n\n /**\n * Get pending event count.\n */\n getPendingCount(): number {\n return this.queue.length\n }\n\n /**\n * Check if there are pending events.\n */\n hasPending(): boolean {\n return this.queue.length > 0\n }\n\n /**\n * Clear the queue.\n */\n clear(): void {\n this.queue = []\n this.clearTimer()\n }\n\n /**\n * Reset statistics.\n */\n resetStats(): void {\n this.stats = {\n totalBatches: 0,\n totalEvents: 0,\n totalLatency: 0,\n autoFlushCount: 0,\n manualFlushCount: 0,\n lastFlushTime: Date.now(),\n }\n }\n\n /**\n * Stop the batcher and flush remaining events.\n */\n async stop(): Promise<EventTask[]> {\n this.clearTimer()\n if (this.queue.length > 0) {\n return this.flush(false)\n }\n return []\n }\n\n /**\n * Start time-window timer.\n */\n private startTimer(): void {\n if (this.flushTimer !== null) {\n return\n }\n\n this.flushTimer = setTimeout(() => {\n this.flushTimer = null\n if (this.queue.length > 0) {\n // Async flush, non-blocking\n this.flush(true).catch((error) => {\n console.error('[EventBatcher] Flush error:', error)\n })\n }\n }, this.flushIntervalMs)\n }\n\n /**\n * Clear timer.\n */\n private clearTimer(): void {\n if (this.flushTimer !== null) {\n clearTimeout(this.flushTimer)\n this.flushTimer = null\n }\n }\n\n /**\n * Adjust batch size.\n */\n setBatchSize(newSize: number): void {\n // Check if immediate flush is needed\n if (this.queue.length >= newSize && newSize < this.batchSize) {\n this.flush(true).catch((error) => {\n console.error('[EventBatcher] Flush error:', error)\n })\n }\n }\n\n /**\n * Adjust flush interval.\n */\n setFlushInterval(_newIntervalMs: number): void {\n // Restart timer with new interval\n this.clearTimer()\n if (this.queue.length > 0) {\n this.startTimer()\n }\n }\n\n /**\n * Get batch size.\n */\n getBatchSize(): number {\n return this.batchSize\n }\n\n /**\n * Get flush interval.\n */\n getFlushInterval(): number {\n return this.flushIntervalMs\n }\n}\n",
|
|
66
|
+
"/**\n * Event aggregation manager (FS-102)\n *\n * Coordinates deduplication and batching for optimal event processing:\n * - Event deduplication (pattern-based or idempotency-key based)\n * - Micro-batching (time and size dual-trigger)\n * - Backpressure-aware window adjustment\n * - Complete statistics tracking\n *\n * Ported from: examples/flash-sale-fullstack/src/cache/events/EventAggregator.ts\n */\n\nimport type { BackpressureManager } from '../BackpressureManager'\nimport { BackpressureState } from '../BackpressureManager'\nimport type { EventTask } from '../types'\nimport { AggregationWindow } from './AggregationWindow'\nimport { DeduplicationManager } from './DeduplicationManager'\nimport { EventBatcher } from './EventBatcher'\nimport type { AggregationConfig, AggregationStats } from './types'\nimport { DEFAULT_AGGREGATION_CONFIG, DEFAULT_WINDOW_ADJUSTMENT, getSuggestedWindow } from './types'\n\n/**\n * Event aggregation manager.\n */\nexport class EventAggregationManager {\n // Core components\n private deduplicator: DeduplicationManager\n private batcher: EventBatcher\n private window: AggregationWindow\n\n // Configuration\n private config: Required<AggregationConfig>\n\n // Backpressure manager reference (optional)\n private backpressureManager?: BackpressureManager\n\n // Submission function to queue\n private submitToQueueFn?: (tasks: EventTask[]) => Promise<void>\n\n // Disabled flag\n private disabled = false\n\n /**\n * Create an event aggregation manager.\n */\n constructor(config?: Partial<AggregationConfig>) {\n this.config = {\n ...DEFAULT_AGGREGATION_CONFIG,\n ...config,\n }\n\n // Initialize components\n this.deduplicator = new DeduplicationManager(this.config)\n\n // Initialize batcher with default submit function (override later)\n this.batcher = new EventBatcher(\n this.config.batchSize ?? 50,\n this.config.windowMs ?? 200,\n (tasks) => this.submitToQueue(tasks)\n )\n\n this.window = new AggregationWindow(this.config.windowMs ?? 200)\n }\n\n /**\n * Set backpressure manager for window adjustment.\n * FS-103:Also sets BackpressureManager on AggregationWindow for feedback loop.\n */\n setBackpressureManager(backpressure: BackpressureManager): void {\n this.backpressureManager = backpressure\n // === FS-103:Enable bidirectional feedback between AggregationWindow and BackpressureManager ===\n this.window.setBackpressureManager(backpressure)\n }\n\n /**\n * Set the actual queue submission function.\n */\n setSubmitToQueueFn(fn: (tasks: EventTask[]) => Promise<void>): void {\n this.submitToQueueFn = fn\n }\n\n /**\n * Submit an event for aggregation.\n *\n * Returns true if accepted, false if rejected due to backpressure.\n */\n async submit(task: EventTask): Promise<boolean> {\n if (this.disabled) {\n throw new Error('[EventAggregationManager] Manager is disabled')\n }\n\n // Check backpressure state\n if (this.backpressureManager) {\n const state = this.backpressureManager.getState()\n\n // Adjust window based on backpressure\n const suggestedWindow = getSuggestedWindow(state, DEFAULT_WINDOW_ADJUSTMENT)\n if (suggestedWindow !== this.window.getCurrentWindow()) {\n this.window.adjustWindow(state)\n this.batcher.setFlushInterval(this.window.getCurrentWindow())\n }\n\n // Reject if overflow\n if (state === BackpressureState.OVERFLOW) {\n return false\n }\n }\n\n // Add to deduplicator\n this.deduplicator.addEvent(task)\n\n // Get deduplicated events\n const deduplicated = this.deduplicator.getDeduplicated()\n\n // Submit to batcher\n for (const event of deduplicated) {\n await this.batcher.enqueue(event)\n }\n\n return true\n }\n\n /**\n * Submit multiple events.\n */\n async submitBatch(tasks: EventTask[]): Promise<number> {\n let accepted = 0\n\n for (const task of tasks) {\n const result = await this.submit(task)\n if (result) {\n accepted++\n }\n }\n\n return accepted\n }\n\n /**\n * Flush current batch to queue.\n */\n async flush(): Promise<EventTask[]> {\n return this.batcher.flush(false)\n }\n\n /**\n * Get aggregation statistics.\n */\n getStats(): AggregationStats {\n return {\n deduplication: this.deduplicator.getStats(),\n batching: this.batcher.getStats(),\n window: this.window.getStats(),\n timestamp: Date.now(),\n }\n }\n\n /**\n * Reset statistics.\n */\n resetStats(): void {\n this.deduplicator.resetStats()\n this.batcher.resetStats()\n }\n\n /**\n * Clear aggregation state.\n */\n clear(): void {\n this.deduplicator.clear()\n this.batcher.clear()\n }\n\n /**\n * Check if there are pending events.\n */\n hasPending(): boolean {\n return this.batcher.hasPending()\n }\n\n /**\n * Get pending event count.\n */\n getPendingCount(): number {\n return this.batcher.getPendingCount()\n }\n\n /**\n * Disable the aggregation manager.\n */\n disable(): void {\n this.disabled = true\n }\n\n /**\n * Enable the aggregation manager.\n */\n enable(): void {\n this.disabled = false\n }\n\n /**\n * Check if aggregation manager is enabled.\n */\n isEnabled(): boolean {\n return !this.disabled\n }\n\n /**\n * Shutdown the aggregation manager.\n */\n async shutdown(): Promise<EventTask[]> {\n const pending = await this.batcher.stop()\n this.deduplicator.shutdown()\n return pending\n }\n\n /**\n * Submit deduplicated events to queue.\n */\n private async submitToQueue(tasks: EventTask[]): Promise<void> {\n if (!this.submitToQueueFn) {\n throw new Error('[EventAggregationManager] Queue submission function not configured')\n }\n\n await this.submitToQueueFn(tasks)\n }\n\n /**\n * Get deduplication manager (for testing).\n */\n __getDeduplicator(): DeduplicationManager {\n return this.deduplicator\n }\n\n /**\n * Get event batcher (for testing).\n */\n __getBatcher(): EventBatcher {\n return this.batcher\n }\n\n /**\n * Get aggregation window (for testing).\n */\n __getWindow(): AggregationWindow {\n return this.window\n }\n}\n",
|
|
67
|
+
"import { EventAggregationManager } from '../events/aggregation/EventAggregationManager'\nimport { CircuitBreaker } from '../events/CircuitBreaker'\nimport type { EventBackend } from '../events/EventBackend'\nimport type { EventOptions } from '../events/EventOptions'\nimport { DEFAULT_EVENT_OPTIONS } from '../events/EventOptions'\nimport { EventPriorityQueue } from '../events/EventPriorityQueue'\nimport { IdempotencyCache } from '../events/IdempotencyCache'\nimport type { EventTask } from '../events/types'\nimport type { AsyncDetector } from './AsyncDetector'\nimport type { MigrationWarner } from './MigrationWarner'\nimport type { ActionCallback, HookManagerConfig, ListenerInfo, ListenerOptions } from './types'\n\n/**\n * 管理 action hook 的登記與執行。\n *\n * Action hook 用於觸發副作用(例如發送 email、記錄日誌)。\n * 支援同步和非同步執行模式,以及透過 EventPriorityQueue 的優先級佇列處理。\n *\n * @internal\n */\nexport class ActionManager {\n /**\n * 儲存所有已登記的 action callbacks。\n */\n private actions: Map<string, ActionCallback[]> = new Map()\n\n private backend: EventBackend\n private idempotencyCache: IdempotencyCache\n private config: HookManagerConfig\n private asyncDetector: AsyncDetector\n private migrationWarner: MigrationWarner\n private aggregationManager?: EventAggregationManager\n\n constructor(\n backend: EventBackend,\n config: HookManagerConfig,\n asyncDetector: AsyncDetector,\n migrationWarner: MigrationWarner\n ) {\n this.backend = backend\n this.config = config\n this.asyncDetector = asyncDetector\n this.migrationWarner = migrationWarner\n this.idempotencyCache = new IdempotencyCache()\n\n // 初始化事件聚合管理器(FS-102)\n if (config.aggregation?.enabled) {\n this.aggregationManager = new EventAggregationManager(config.aggregation)\n this.aggregationManager.setSubmitToQueueFn((tasks) => {\n for (const task of tasks) {\n this.backend.enqueue(task)\n }\n return Promise.resolve()\n })\n // 連接 backpressure manager 以調整窗口\n if (this.backend instanceof EventPriorityQueue) {\n const backpressure = this.backend.getBackpressureManager()\n if (backpressure) {\n this.aggregationManager.setBackpressureManager(backpressure)\n }\n }\n }\n }\n\n /**\n * 更新設定。\n */\n updateConfig(config: HookManagerConfig): void {\n this.config = config\n }\n\n /**\n * 更新 backend。\n */\n setBackend(backend: EventBackend): void {\n this.backend = backend\n }\n\n /**\n * Register an action hook.\n *\n * Actions are used to trigger side effects (e.g., logging, sending emails)\n * at specific points in the application lifecycle.\n *\n * @template TArgs - The type of arguments passed to the action.\n * @param hook - The unique name of the hook.\n * @param callback - The callback function to execute.\n * @param options - Optional listener options (type override, circuit breaker).\n *\n * @example\n * ```typescript\n * actionManager.addAction('user_registered', async (user: User) => {\n * await sendWelcomeEmail(user)\n * })\n * ```\n */\n addAction<TArgs = unknown>(\n hook: string,\n callback: ActionCallback<TArgs>,\n options?: ListenerOptions\n ): void {\n if (!this.actions.has(hook)) {\n this.actions.set(hook, [])\n }\n\n let finalCallback = callback\n\n if (options?.circuitBreaker) {\n const breaker = new CircuitBreaker(options.circuitBreaker)\n\n finalCallback = async (args: TArgs) => {\n return breaker.execute(async () => {\n const result = callback(args)\n if (result instanceof Promise) {\n await result\n }\n })\n }\n }\n\n // 如有 type override,儲存至 AsyncDetector\n if (options?.type && options.type !== 'auto') {\n this.asyncDetector.setTypeOverride(finalCallback as unknown as ActionCallback, options.type)\n }\n\n // Generic type erasure for storage\n this.actions.get(hook)?.push(finalCallback as unknown as ActionCallback)\n }\n\n /**\n * 判斷是否需要使用非同步 dispatch,並在需要時發出遷移警告。\n *\n * 此方法僅判斷 dispatch 模式和發出警告,不執行實際的 dispatch。\n * 實際執行由 HookManager.doAction 負責,以確保 ObservableHookManager 等子類別\n * 的多型覆寫(override)能正確攔截 doActionSync / doActionAsync 呼叫。\n *\n * @param hook - Hook name\n * @param args - Event args (unused here, kept for API consistency)\n * @param options - Event options\n * @returns 'async' if async dispatch should be used, 'sync' otherwise\n */\n resolveDispatchMode<TArgs = unknown>(\n hook: string,\n _args: TArgs,\n options?: EventOptions\n ): 'async' | 'sync' {\n const callbacks = this.actions.get(hook) || []\n const shouldUseAsync = this.shouldUseAsyncDispatch(callbacks, options)\n\n if (!shouldUseAsync) {\n // 同步 dispatch(legacy mode)—發出遷移警告\n if (this.config.showDeprecationWarnings || this.config.migrationMode === 'hybrid') {\n if (this.config.migrationMode === 'hybrid' || this.config.showDeprecationWarnings) {\n this.migrationWarner.warn(\n hook,\n 'Consider migrating to async mode for better performance and reliability.'\n )\n }\n }\n }\n\n return shouldUseAsync ? 'async' : 'sync'\n }\n\n /**\n * Run all registered actions synchronously (legacy mode).\n *\n * @template TArgs - The type of arguments passed to the action.\n * @param hook - The name of the hook.\n * @param args - The arguments to pass to the callbacks.\n */\n async doActionSync<TArgs = unknown>(hook: string, args: TArgs): Promise<void> {\n const callbacks = this.actions.get(hook) || []\n\n for (const callback of callbacks) {\n try {\n await callback(args)\n } catch (error) {\n console.error(`[HookManager] Error in action '${hook}':`, error)\n }\n }\n }\n\n /**\n * Run all registered actions asynchronously via priority queue.\n *\n * 透過 EventPriorityQueue 進行非同步 dispatch,支援:\n * - 優先級處理(high > normal > low)\n * - 超時處理\n * - 順序保證(strict、partition、none)\n * - 冪等性\n *\n * @template TArgs - The type of arguments passed to the action.\n * @param hook - The name of the hook.\n * @param args - The arguments to pass to the callbacks.\n * @param options - Event options for async dispatch.\n */\n async doActionAsync<TArgs = unknown>(\n hook: string,\n args: TArgs,\n options: EventOptions = {}\n ): Promise<void> {\n const callbacks = this.actions.get(hook) || []\n\n if (callbacks.length === 0) {\n return\n }\n\n // 與預設選項合併\n const mergedOptions: EventOptions = {\n ...DEFAULT_EVENT_OPTIONS,\n ...options,\n async: true,\n }\n\n // 冪等性檢查\n if (mergedOptions.idempotencyKey) {\n const ttl = mergedOptions.ttl || DEFAULT_EVENT_OPTIONS.ttl\n const isDuplicate = this.idempotencyCache.isDuplicate(mergedOptions.idempotencyKey, ttl)\n\n if (isDuplicate) {\n console.warn(\n `[HookManager] Event '${hook}' with idempotency key '${mergedOptions.idempotencyKey}' was skipped (duplicate within TTL window)`\n )\n return\n }\n }\n\n // 使用 backend 加入佇列\n const nowMs = Date.now()\n const task: EventTask = {\n id: `task-${nowMs}-${Math.random().toString(36).substr(2, 9)}`,\n hook,\n args,\n callbacks: callbacks as ActionCallback[],\n options: mergedOptions,\n createdAt: nowMs,\n enqueuedAt: nowMs,\n partitionKey: mergedOptions.partitionKey,\n retryCount: 0,\n }\n\n // === FS-102: Aggregation layer interception ===\n if (this.aggregationManager && mergedOptions.aggregation?.enabled) {\n try {\n const accepted = await this.aggregationManager.submit(task)\n if (!accepted) {\n console.warn(\n `[HookManager] Event '${hook}' was rejected due to backpressure (aggregation overflow)`\n )\n }\n } catch (error) {\n console.error(`[HookManager] Aggregation error for event '${hook}':`, error)\n // 錯誤時降級至直接加入佇列\n this.backend.enqueue(task)\n }\n } else {\n // 直接加入佇列,不使用聚合\n this.backend.enqueue(task)\n }\n\n // 注意:此處不 await 佇列處理\n // 事件在背景非同步處理\n }\n\n /**\n * Determine if async dispatch should be used.\n *\n * @param callbacks - Callbacks to check\n * @param options - Event options\n * @returns True if async dispatch should be used\n */\n shouldUseAsyncDispatch(callbacks: ActionCallback[], options?: EventOptions): boolean {\n // 明確指定 async 選項\n if (options?.async === true) {\n return true\n }\n\n // 明確指定 sync 選項\n if (options?.async === false) {\n return false\n }\n\n // Migration mode: async\n if (this.config.migrationMode === 'async') {\n return true\n }\n\n // Migration mode: sync\n if (this.config.migrationMode === 'sync') {\n return false\n }\n\n // Migration mode: hybrid(自動偵測)\n if (this.config.migrationMode === 'hybrid') {\n const hasAsyncListeners = callbacks.some((cb) => this.asyncDetector.isEffectivelyAsync(cb))\n return hasAsyncListeners || this.config.asyncByDefault === true\n }\n\n return false\n }\n\n /**\n * Determine the dispatch mode for an event.\n *\n * @param hook - Hook name\n * @param options - Optional event options\n * @returns The dispatch mode: 'sync' or 'async'\n */\n detectMode(hook: string, options?: EventOptions): 'sync' | 'async' {\n const callbacks = this.getListeners(hook)\n const shouldUseAsync = this.shouldUseAsyncDispatch(callbacks, options)\n return shouldUseAsync ? 'async' : 'sync'\n }\n\n /**\n * Check if any listener for a hook is async (including type overrides).\n *\n * @param hook - Hook name\n * @returns True if any listener is async\n */\n hasAsyncListeners(hook: string): boolean {\n const callbacks = this.getListeners(hook)\n return callbacks.some((cb) => this.asyncDetector.isEffectivelyAsync(cb))\n }\n\n /**\n * Get detailed information about all listeners for a hook.\n *\n * @param hook - Hook name\n * @returns Array of listener info objects\n */\n getListenerInfo(hook: string): ListenerInfo[] {\n const callbacks = this.getListeners(hook)\n\n return callbacks.map((callback) => {\n const typeOverride = this.asyncDetector.getTypeOverride(callback)\n const isAsync = this.asyncDetector.isEffectivelyAsync(callback)\n\n return {\n callback,\n isAsync,\n typeOverride,\n }\n })\n }\n\n /**\n * Get all registered listeners for a hook.\n *\n * @param hook - Hook name\n * @returns Array of callbacks\n */\n getListeners(hook: string): ActionCallback[] {\n return this.actions.get(hook) || []\n }\n\n /**\n * Remove all listeners for a specific action hook.\n *\n * @param hook - Hook name\n */\n removeAction(hook: string): void {\n this.actions.delete(hook)\n }\n}\n",
|
|
68
|
+
"import type { ActionCallback } from './types'\n\n/**\n * 負責偵測 callback 函數是否為非同步(async)。\n *\n * 提供靜態偵測(透過函數簽名)和運行時偵測(透過執行函數)兩種方式,\n * 並包含快取機制以提升重複偵測的性能。\n *\n * @internal\n */\nexport class AsyncDetector {\n /**\n * 快取非同步偵測結果(WeakMap 以自動垃圾回收)。\n */\n private asyncDetectionCache: WeakMap<ActionCallback, boolean> = new WeakMap()\n\n /**\n * 快取中的項目計數(供測試/偵錯使用)。\n */\n private asyncDetectionCacheCount = 0\n\n /**\n * 儲存 listener type override(callback -> type)。\n */\n private listenerTypeOverrides: WeakMap<ActionCallback, 'sync' | 'async' | 'auto'> = new WeakMap()\n\n /**\n * 設定 listener 的 type override。\n *\n * @param callback - 目標 callback\n * @param type - Type override 值\n */\n setTypeOverride(callback: ActionCallback, type: 'sync' | 'async' | 'auto'): void {\n this.listenerTypeOverrides.set(callback, type)\n }\n\n /**\n * 取得 listener 的 type override。\n *\n * @param callback - 目標 callback\n * @returns Type override 或 undefined\n */\n getTypeOverride(callback: ActionCallback): 'sync' | 'async' | 'auto' | undefined {\n return this.listenerTypeOverrides.get(callback)\n }\n\n /**\n * Check if a callback is an async function (with caching).\n *\n * Detection methods:\n * 1. Check cache first\n * 2. Check type override\n * 3. Check constructor.name === 'AsyncFunction'\n * 4. Fallback: Check function string representation\n *\n * @param callback - The callback to check\n * @returns True if the callback is async\n * @public\n */\n isAsyncListener(callback: ActionCallback): boolean {\n // 先檢查快取\n const cachedResult = this.asyncDetectionCache.get(callback)\n if (cachedResult !== undefined) {\n return cachedResult\n }\n\n // 檢查 type override\n const typeOverride = this.listenerTypeOverrides.get(callback)\n if (typeOverride === 'async') {\n this.cacheResult(callback, true)\n return true\n }\n if (typeOverride === 'sync') {\n this.cacheResult(callback, false)\n return false\n }\n\n // 主要偵測:constructor name\n let isAsync = callback.constructor.name === 'AsyncFunction'\n\n // 備用偵測:函數字串表示\n // 處理邊緣情況,例如轉譯代碼或 bound functions\n if (!isAsync) {\n const fnStr = callback.toString()\n // 檢查起始的 async 關鍵字(處理 async arrow functions 和 async function expressions)\n isAsync = /^async\\s/.test(fnStr) || fnStr.startsWith('async ')\n }\n\n // 快取結果\n this.cacheResult(callback, isAsync)\n\n return isAsync\n }\n\n /**\n * Check if a listener is effectively async (considering type override).\n *\n * @param callback - The callback to check\n * @returns True if the listener should be treated as async\n */\n isEffectivelyAsync(callback: ActionCallback): boolean {\n const typeOverride = this.listenerTypeOverrides.get(callback)\n\n if (typeOverride === 'async') {\n return true\n }\n if (typeOverride === 'sync') {\n return false\n }\n\n // 自動偵測\n return this.isAsyncListener(callback)\n }\n\n /**\n * Runtime detection for functions that return Promises but aren't declared async.\n *\n * 此方法會實際執行 callback 來檢查是否回傳 Promise,請謹慎使用。\n *\n * @param callback - The callback to check\n * @param testArgs - Arguments to pass to the callback for testing\n * @returns True if the callback returns a Promise\n * @public\n */\n async isAsyncListenerRuntime<TArgs = unknown>(\n callback: ActionCallback<TArgs>,\n testArgs: TArgs\n ): Promise<boolean> {\n try {\n const result = callback(testArgs)\n const isPromise = result instanceof Promise\n\n // 用運行時結果更新快取\n if (isPromise) {\n this.cacheResult(callback as ActionCallback, true)\n // 等待 promise 結算(避免懸掛的 promise)\n await result.catch(() => {\n // 刻意吞掉錯誤 - 我們只關心 Promise 偵測\n })\n }\n\n return isPromise\n } catch {\n // 如果 callback 拋出例外,視為 sync(錯誤由正常流程處理)\n return false\n }\n }\n\n /**\n * Get the size of the async detection cache (for testing/debugging).\n *\n * @returns Number of cached detection results\n */\n getCacheSize(): number {\n return this.asyncDetectionCacheCount\n }\n\n /**\n * Clear the async detection cache.\n */\n clearCache(): void {\n this.asyncDetectionCache = new WeakMap()\n this.asyncDetectionCacheCount = 0\n }\n\n /**\n * 快取非同步偵測結果。\n * @internal\n */\n private cacheResult(callback: ActionCallback, isAsync: boolean): void {\n if (!this.asyncDetectionCache.has(callback)) {\n this.asyncDetectionCacheCount++\n }\n this.asyncDetectionCache.set(callback, isAsync)\n }\n}\n",
|
|
69
|
+
"import type { DeadLetterQueue } from '../events/DeadLetterQueue'\nimport type { EventOptions } from '../events/EventOptions'\nimport type { DeadLetterQueueManager } from '../reliability/DeadLetterQueueManager'\n\n/**\n * DLQ 操作輔助函式模組。\n *\n * 提取自 HookManager 的 DLQ 管理邏輯,以降低 HookManager 複雜度。\n * 這些函式接受所需依賴作為參數,便於測試與重用。\n *\n * @internal\n */\n\n/**\n * 重新加入單一 DLQ 項目到事件佇列。\n *\n * @param dlqEntryId - DLQ 項目 ID\n * @param dlq - 記憶體內 DLQ 實例\n * @param requeue - 重新加入佇列的回調函式\n * @returns 是否成功重新加入\n */\nexport async function requeueDLQEntry(\n dlqEntryId: string,\n dlq: DeadLetterQueue,\n requeue: (eventName: string, payload: unknown, options: EventOptions) => Promise<void>\n): Promise<boolean> {\n const entry = dlq.get(dlqEntryId)\n\n if (!entry) {\n return false\n }\n\n // 更新最後重試時間戳\n dlq.updateLastRetried(dlqEntryId)\n\n // 重新加入佇列\n await requeue(entry.eventName, entry.payload, entry.options)\n\n // 成功重新加入後從 DLQ 刪除\n dlq.delete(dlqEntryId)\n\n return true\n}\n\n/**\n * 批次重新加入指定事件名稱的所有 DLQ 項目。\n *\n * @param eventName - 事件名稱\n * @param dlq - 記憶體內 DLQ 實例\n * @param requeue - 重新加入佇列的回調函式\n * @returns 成功重新加入的項目數\n */\nexport async function requeueDLQBatch(\n eventName: string,\n dlq: DeadLetterQueue,\n requeue: (entryId: string) => Promise<boolean>\n): Promise<number> {\n const entries = dlq.list({ eventName })\n let requeuedCount = 0\n\n for (const entry of entries) {\n const success = await requeue(entry.id)\n if (success) {\n requeuedCount++\n }\n }\n\n return requeuedCount\n}\n\n/**\n * 建立 EventPriorityQueue 持久化 DLQ 處理器。\n *\n * @param persistentDlqManager - 持久化 DLQ 管理器\n * @returns 持久化 DLQ 處理器函式\n */\nexport function createPersistentDLQHandler(persistentDlqManager: DeadLetterQueueManager) {\n return async (\n hook: string,\n args: unknown,\n options: EventOptions,\n error: Error,\n retryCount: number,\n _firstFailedAt: number\n ): Promise<void> => {\n try {\n // 將 event options 轉換為 retry policy\n const retryPolicy = options.retry\n ? {\n maxRetries: options.retry.maxRetries || 3,\n backoff: options.retry.backoff || 'exponential',\n initialDelayMs: options.retry.initialDelayMs || 1000,\n maxDelayMs: options.retry.maxDelayMs || 30000,\n }\n : undefined\n\n // 移至持久化 DLQ\n await persistentDlqManager.moveToDlq(hook, args, options, error, retryCount, retryPolicy)\n } catch (dlqError) {\n console.error(`[HookManager] Error moving event to persistent DLQ:`, dlqError)\n }\n }\n}\n\n/**\n * 重新加入持久化 DLQ 單一項目到事件佇列。\n *\n * @param dlqId - 持久化 DLQ 項目 UUID\n * @param persistentDlqManager - 持久化 DLQ 管理器\n * @param doActionAsync - 重新加入佇列的回調函式\n * @returns 是否成功重新加入\n */\nexport async function requeuePersistentDLQEntry(\n dlqId: string,\n persistentDlqManager: DeadLetterQueueManager,\n doActionAsync: (event: string, args: unknown, options: EventOptions) => Promise<void>\n): Promise<boolean> {\n try {\n const event = await persistentDlqManager.getById(dlqId)\n if (!event) {\n return false\n }\n\n // 重新加入佇列\n await doActionAsync(event.event_name, event.event_payload, event.event_options as EventOptions)\n\n // 標記為已重新加入佇列\n await persistentDlqManager.requeue(dlqId)\n\n return true\n } catch (error) {\n console.error(`[HookManager] Error requeuing persistent DLQ entry:`, error)\n return false\n }\n}\n\n/**\n * 批次重新加入持久化 DLQ 項目。\n *\n * @param filter - 過濾條件\n * @param persistentDlqManager - 持久化 DLQ 管理器\n * @returns 批次操作結果統計\n */\nexport async function requeuePersistentDLQBatch(\n filter:\n | { eventName?: string; status?: 'pending' | 'requeued' | 'resolved' | 'abandoned' }\n | undefined,\n persistentDlqManager: DeadLetterQueueManager\n): Promise<{ total: number; succeeded: number; failed: number }> {\n try {\n return await persistentDlqManager.retryBatch(filter)\n } catch (error) {\n console.error(`[HookManager] Error in persistent DLQ batch retry:`, error)\n return { total: 0, succeeded: 0, failed: 0 }\n }\n}\n",
|
|
70
|
+
"import type { FilterCallback } from './types'\n\n/**\n * 管理 filter hook 的登記與執行。\n *\n * Filter hook 用於轉換數值:每個 callback 接收前一個 callback 的回傳值,\n * 並返回新的轉換後數值。所有 callback 按登記順序依次執行。\n *\n * @internal\n */\nexport class FilterManager {\n /**\n * 儲存所有已登記的 filter callbacks。\n * Map key 為 hook 名稱,value 為 callback 陣列。\n */\n private filters: Map<string, FilterCallback[]> = new Map()\n\n /**\n * Register a filter hook.\n *\n * Filters are used to transform a value (input/output) through a chain of\n * callbacks. Each callback must return the modified value.\n *\n * @template T - The type of value being filtered.\n * @param hook - The unique name of the hook.\n * @param callback - The callback function to execute.\n *\n * @example\n * ```typescript\n * filterManager.addFilter('content', async (content: string) => {\n * return content.toUpperCase()\n * })\n * ```\n */\n addFilter<T = unknown>(hook: string, callback: FilterCallback<T>): void {\n if (!this.filters.has(hook)) {\n this.filters.set(hook, [])\n }\n // Generic type erasure for storage\n this.filters.get(hook)?.push(callback as unknown as FilterCallback)\n }\n\n /**\n * Apply all registered filters sequentially.\n *\n * Each callback receives the previous callback's return value.\n *\n * @template T - The type of value being filtered.\n * @param hook - The name of the hook.\n * @param initialValue - The initial value to filter.\n * @param args - Additional arguments to pass to the callbacks.\n * @returns The final filtered value.\n *\n * @example\n * ```typescript\n * const content = await filterManager.applyFilters('content', 'hello world')\n * ```\n */\n async applyFilters<T = unknown>(hook: string, initialValue: T, ...args: unknown[]): Promise<T> {\n const callbacks = this.filters.get(hook) || []\n let value = initialValue\n\n for (const callback of callbacks) {\n try {\n value = (await callback(value, ...args)) as T\n } catch (error) {\n console.error(`[HookManager] Error in filter '${hook}':`, error)\n // 錯誤處理策略:記錄錯誤並繼續使用當前值\n }\n }\n\n return value\n }\n\n /**\n * Check if any filters are registered for a hook.\n *\n * @param hook - Hook name\n * @returns True if at least one filter is registered\n */\n hasFilters(hook: string): boolean {\n const callbacks = this.filters.get(hook)\n return callbacks !== undefined && callbacks.length > 0\n }\n\n /**\n * Get count of registered filters for a hook.\n *\n * @param hook - Hook name\n * @returns Number of registered filters\n */\n getFilterCount(hook: string): number {\n return this.filters.get(hook)?.length ?? 0\n }\n\n /**\n * Remove all filters for a specific hook.\n *\n * @param hook - Hook name\n */\n removeFilters(hook: string): void {\n this.filters.delete(hook)\n }\n}\n",
|
|
71
|
+
"/**\n * Migration warning manager for deprecation warnings.\n *\n * 管理遷移警告訊息,支援透過環境變數抑制特定事件的警告。\n *\n * @internal\n */\nexport class MigrationWarner {\n private suppressedWarnings: Set<string> = new Set()\n\n constructor() {\n // 從環境變數載入已抑制的警告\n const suppressed = process.env.GRAVITO_SUPPRESS_MIGRATION_WARNING\n if (suppressed) {\n suppressed.split(',').forEach((event) => {\n this.suppressedWarnings.add(event.trim())\n })\n }\n }\n\n /**\n * 發出遷移警告訊息。\n *\n * @param eventName - 事件名稱\n * @param message - 警告訊息\n */\n warn(eventName: string, message: string): void {\n if (this.suppressedWarnings.has(eventName)) {\n return\n }\n\n console.warn(`[Gravito Migration] Event \"${eventName}\" using synchronous dispatch`)\n console.warn(` ${message}`)\n console.warn(` Reference: https://gravito.dev/docs/events/async-migration`)\n console.warn(` To suppress this warning: GRAVITO_SUPPRESS_MIGRATION_WARNING=\"${eventName}\"`)\n }\n\n /**\n * 抑制特定事件的遷移警告。\n *\n * @param eventName - 要抑制警告的事件名稱\n */\n suppress(eventName: string): void {\n this.suppressedWarnings.add(eventName)\n }\n}\n",
|
|
72
|
+
"/**\n * @gravito/core - Retry Policy\n *\n * 重試策略和退避算法的實現\n * 支持指數和線性退避,包含 Jitter 防止雷鳴羊群\n */\n\n/**\n * 重試策略配置接口\n *\n * @example\n * ```typescript\n * const policy: RetryPolicy = {\n * maxRetries: 3,\n * backoff: 'exponential',\n * initialDelayMs: 1000,\n * maxDelayMs: 30000,\n * dlqAfterMaxRetries: true\n * }\n * ```\n */\nexport interface RetryPolicy {\n /** 最大重試次數 */\n maxRetries: number\n\n /** 退避策略:指數或線性 */\n backoff: 'exponential' | 'linear'\n\n /** 初始延遲時間(毫秒) */\n initialDelayMs: number\n\n /** 最大延遲時間(毫秒) */\n maxDelayMs: number\n\n /** 超過最大重試次數後是否發送到死信隊列 */\n dlqAfterMaxRetries?: boolean\n}\n\n/**\n * 重試引擎\n *\n * 負責計算重試延遲、判斷是否應該重試等邏輯\n *\n * @example\n * ```typescript\n * const engine = new RetryEngine()\n * const delay = engine.calculateDelay(1, policy)\n * const shouldRetry = engine.shouldRetry(3, policy)\n * ```\n */\nexport class RetryEngine {\n /**\n * 計算下次重試的延遲時間(毫秒)\n *\n * @param attemptCount - 當前嘗試次數(從 1 開始)\n * @param policy - 重試策略配置\n * @returns 延遲時間(毫秒)\n *\n * @description\n * - 指數退避:delay = initialDelay * 2^(attemptCount - 1)\n * - 線性退避:delay = initialDelay * attemptCount\n * - 添加隨機抖動(Jitter),防止雷鳴羊群問題\n * - 結果不超過 maxDelayMs\n *\n * @example\n * ```typescript\n * const policy: RetryPolicy = {\n * maxRetries: 3,\n * backoff: 'exponential',\n * initialDelayMs: 1000,\n * maxDelayMs: 30000\n * }\n *\n * // 第 1 次重試:1000ms * 2^0 = 1000ms (加抖動)\n * const delay1 = engine.calculateDelay(1, policy) // ~1100ms\n *\n * // 第 2 次重試:1000ms * 2^1 = 2000ms (加抖動)\n * const delay2 = engine.calculateDelay(2, policy) // ~2200ms\n *\n * // 第 3 次重試:1000ms * 2^2 = 4000ms (加抖動)\n * const delay3 = engine.calculateDelay(3, policy) // ~4400ms\n * ```\n */\n calculateDelay(attemptCount: number, policy: RetryPolicy): number {\n const { backoff, initialDelayMs, maxDelayMs } = policy\n\n // 計算基礎延遲\n let delay: number\n\n if (backoff === 'exponential') {\n // 指數退避:delay = initialDelay * 2^(attemptCount - 1)\n // 確保 attemptCount >= 1\n const exponent = Math.max(0, attemptCount - 1)\n delay = initialDelayMs * 2 ** exponent\n } else {\n // 線性退避:delay = initialDelay * attemptCount\n delay = initialDelayMs * attemptCount\n }\n\n // 添加隨機抖動(Jitter),避免雷鳴羊群問題\n // 抖動範圍:0 到 delay 的 10%\n const jitter = Math.random() * delay * 0.1\n delay = delay + jitter\n\n // 限制最大延遲\n return Math.min(Math.ceil(delay), maxDelayMs)\n }\n\n /**\n * 判斷是否應該重試\n *\n * @param attemptCount - 當前嘗試次數(從 1 開始)\n * @param policy - 重試策略配置\n * @returns 是否應該重試\n *\n * @description\n * 當 attemptCount < maxRetries 時返回 true\n *\n * @example\n * ```typescript\n * const policy = { maxRetries: 3, ... }\n *\n * engine.shouldRetry(1, policy) // true(1 < 3)\n * engine.shouldRetry(3, policy) // false(3 >= 3)\n * engine.shouldRetry(4, policy) // false(4 >= 3)\n * ```\n */\n shouldRetry(attemptCount: number, policy: RetryPolicy): boolean {\n return attemptCount < policy.maxRetries\n }\n\n /**\n * 獲取退避時間(包含延遲計算和 Jitter)\n *\n * @param retryCount - 重試次數(從 0 開始)\n * @param policy - 重試策略配置\n * @returns 退避時間(毫秒)\n *\n * @description\n * 這是 calculateDelay 的別名,但接收的是從 0 開始的重試計數\n *\n * @example\n * ```typescript\n * const policy = { maxRetries: 3, initialDelayMs: 1000, maxDelayMs: 30000, backoff: 'exponential' }\n *\n * // 第 0 次重試(第 1 次嘗試失敗)\n * const time1 = engine.getBackoffTime(0, policy) // ~1100ms\n *\n * // 第 1 次重試(第 2 次嘗試失敗)\n * const time2 = engine.getBackoffTime(1, policy) // ~2200ms\n * ```\n */\n getBackoffTime(retryCount: number, policy: RetryPolicy): number {\n // retryCount 從 0 開始,而 calculateDelay 使用 attemptCount 從 1 開始\n return this.calculateDelay(retryCount + 1, policy)\n }\n\n /**\n * 計算下次重試的絕對時間\n *\n * @param retryCount - 重試次數(從 0 開始)\n * @param policy - 重試策略配置\n * @param baseTime - 基礎時間戳(默認為當前時間)\n * @returns 下次重試的時間戳(毫秒)\n *\n * @example\n * ```typescript\n * const policy = { maxRetries: 3, initialDelayMs: 1000, maxDelayMs: 30000, backoff: 'exponential' }\n *\n * const now = Date.now()\n * const nextRetryTime = engine.getNextRetryTime(0, policy, now)\n * // 返回大約 now + 1000 + jitter\n * ```\n */\n getNextRetryTime(retryCount: number, policy: RetryPolicy, baseTime: number = Date.now()): number {\n const delay = this.getBackoffTime(retryCount, policy)\n return baseTime + delay\n }\n\n /**\n * 驗證重試策略配置\n *\n * @param policy - 重試策略配置\n * @returns 是否有效\n *\n * @description\n * 檢查配置是否合理:\n * - maxRetries >= 0\n * - initialDelayMs > 0\n * - maxDelayMs >= initialDelayMs\n *\n * @example\n * ```typescript\n * const validPolicy = { maxRetries: 3, initialDelayMs: 1000, maxDelayMs: 30000, backoff: 'exponential' }\n * engine.isValidPolicy(validPolicy) // true\n *\n * const invalidPolicy = { maxRetries: 3, initialDelayMs: 5000, maxDelayMs: 1000, backoff: 'exponential' }\n * engine.isValidPolicy(invalidPolicy) // false (maxDelayMs < initialDelayMs)\n * ```\n */\n isValidPolicy(policy: RetryPolicy): boolean {\n if (policy.maxRetries < 0) {\n return false\n }\n\n if (policy.initialDelayMs <= 0) {\n return false\n }\n\n if (policy.maxDelayMs < policy.initialDelayMs) {\n return false\n }\n\n if (!['exponential', 'linear'].includes(policy.backoff)) {\n return false\n }\n\n return true\n }\n\n /**\n * 獲取人類可讀的重試信息\n *\n * @param attemptCount - 當前嘗試次數(從 1 開始)\n * @param policy - 重試策略配置\n * @returns 描述性文本\n *\n * @example\n * ```typescript\n * const policy = { maxRetries: 3, initialDelayMs: 1000, maxDelayMs: 30000, backoff: 'exponential' }\n *\n * engine.getRetryInfo(1, policy) // \"Retry 1 of 3, will retry in ~1100ms\"\n * engine.getRetryInfo(3, policy) // \"Retry 3 of 3 (final attempt), will retry in ~4400ms\"\n * engine.getRetryInfo(4, policy) // \"Max retries exceeded (4 >= 3)\"\n * ```\n */\n getRetryInfo(attemptCount: number, policy: RetryPolicy): string {\n if (attemptCount > policy.maxRetries) {\n return `Max retries exceeded (${attemptCount} > ${policy.maxRetries})`\n }\n\n if (!this.shouldRetry(attemptCount, policy)) {\n return `Max retries reached (${attemptCount} of ${policy.maxRetries})`\n }\n\n const delay = this.calculateDelay(attemptCount, policy)\n const isFinal = attemptCount === policy.maxRetries - 1\n const finalNote = isFinal ? ' (final attempt)' : ''\n\n return `Retry ${attemptCount} of ${policy.maxRetries}${finalNote}, will retry in ~${delay}ms`\n }\n}\n\n/**\n * 創建默認的重試策略\n *\n * @returns 默認重試策略\n *\n * @description\n * 默認配置:\n * - 最多重試 3 次\n * - 指數退避\n * - 初始延遲 1 秒\n * - 最大延遲 30 秒\n * - 超過最大重試後發送 DLQ\n *\n * @example\n * ```typescript\n * const policy = getDefaultRetryPolicy()\n * // { maxRetries: 3, backoff: 'exponential', initialDelayMs: 1000, maxDelayMs: 30000, dlqAfterMaxRetries: true }\n * ```\n */\nexport function getDefaultRetryPolicy(): RetryPolicy {\n return {\n maxRetries: 3,\n backoff: 'exponential',\n initialDelayMs: 1000,\n maxDelayMs: 30000,\n dlqAfterMaxRetries: true,\n }\n}\n\n/**\n * 為不同類型的操作獲取預設的重試策略\n *\n * @param type - 操作類型\n * @returns 推薦的重試策略\n *\n * @example\n * ```typescript\n * // 外部 API 調用:更多重試,更長延遲\n * const apiPolicy = getPresetRetryPolicy('external-api')\n *\n * // 數據庫操作:較少重試,短延遲\n * const dbPolicy = getPresetRetryPolicy('database')\n *\n * // 消息隊列:適中重試\n * const mqPolicy = getPresetRetryPolicy('message-queue')\n * ```\n */\nexport function getPresetRetryPolicy(\n type: 'external-api' | 'database' | 'message-queue' | 'default'\n): RetryPolicy {\n switch (type) {\n case 'external-api':\n // 外部 API 調用:更多重試,更長延遲\n return {\n maxRetries: 5,\n backoff: 'exponential',\n initialDelayMs: 1000,\n maxDelayMs: 60000,\n dlqAfterMaxRetries: true,\n }\n\n case 'database':\n // 數據庫操作:較少重試,短延遲\n return {\n maxRetries: 2,\n backoff: 'linear',\n initialDelayMs: 100,\n maxDelayMs: 1000,\n dlqAfterMaxRetries: false,\n }\n\n case 'message-queue':\n // 消息隊列:適中重試,指數退避\n return {\n maxRetries: 3,\n backoff: 'exponential',\n initialDelayMs: 500,\n maxDelayMs: 15000,\n dlqAfterMaxRetries: true,\n }\n default:\n return getDefaultRetryPolicy()\n }\n}\n",
|
|
73
|
+
"/**\n * @gravito/core - Dead Letter Queue Manager\n *\n * 管理失敗事件的持久化存儲\n * 支持 CRUD、重新入隊、批量重試和統計功能\n */\n\n// import type { ConnectionContract } from '@gravito/atlas'\nimport { randomUUID } from '../compat/crypto'\nimport type { EventOptions } from '../events/EventOptions'\nimport type { RetryPolicy } from './RetryPolicy'\nimport { RetryEngine } from './RetryPolicy'\n\n/**\n * DLQ 事件的數據庫記錄\n */\nexport interface DLQRecord {\n id: number\n dlq_id: string\n event_name: string\n event_payload: unknown\n event_options: unknown\n attempt_count: number\n max_retries: number\n next_retry_at: string | null\n last_error: unknown\n status: 'pending' | 'requeued' | 'resolved' | 'abandoned'\n resolution_notes: string | null\n failed_at: string\n created_at: string\n updated_at: string\n}\n\n/**\n * DLQ 事件查詢過濾選項(for DeadLetterQueueManager)\n */\nexport interface DLQManagerFilter {\n /** 按事件名稱篩選 */\n eventName?: string\n\n /** 按狀態篩選 */\n status?: 'pending' | 'requeued' | 'resolved' | 'abandoned'\n\n /** 開始時間 */\n from?: Date\n\n /** 結束時間 */\n to?: Date\n\n /** 結果數量限制 */\n limit?: number\n\n /** 分頁偏移 */\n offset?: number\n}\n\n/**\n * DLQ 統計信息\n */\nexport interface DLQStats {\n /** 事件總數 */\n total: number\n\n /** 按事件名稱的統計 */\n byEvent: Record<string, number>\n\n /** 按狀態的統計 */\n byStatus: Record<string, number>\n}\n\n/**\n * Dead Letter Queue 管理器\n *\n * 負責管理失敗事件的持久化存儲,支持:\n * - 自動將失敗事件移至 DLQ\n * - 查詢和篩選 DLQ 事件\n * - 重新入隊單個或批量事件\n * - 標記事件為已解決或已放棄\n * - 查看統計信息\n *\n * @example\n * ```typescript\n * const db = container.make('db') as any\n * const dlqManager = new DeadLetterQueueManager(db)\n *\n * // 將失敗事件移至 DLQ\n * const dlqId = await dlqManager.moveToDlq(\n * 'order:created',\n * { orderId: '123' },\n * { retry: {...} },\n * error,\n * 3\n * )\n *\n * // 查詢 DLQ 事件\n * const events = await dlqManager.list({ event: 'order:created', status: 'pending' })\n *\n * // 重新入隊\n * await dlqManager.requeue(dlqId)\n *\n * // 批量重試\n * const result = await dlqManager.retryBatch({ eventName: 'order:created' })\n *\n * // 統計\n * const stats = await dlqManager.getStats()\n * ```\n */\nexport class DeadLetterQueueManager {\n private retryEngine: RetryEngine\n\n constructor(private db: any) {\n this.retryEngine = new RetryEngine()\n }\n\n /**\n * 將失敗事件移至死信隊列\n *\n * @param eventName - 事件名稱\n * @param payload - 事件負載\n * @param options - 事件選項\n * @param error - 導致失敗的錯誤\n * @param attemptCount - 當前嘗試次數\n * @param retryPolicy - 重試策略配置\n * @returns DLQ 記錄的 UUID\n *\n * @example\n * ```typescript\n * const dlqId = await dlqManager.moveToDlq(\n * 'order:created',\n * { orderId: 'ORD-123' },\n * { retry: { maxRetries: 3, backoff: 'exponential' } },\n * new Error('Service unavailable'),\n * 3,\n * { maxRetries: 3, backoff: 'exponential', initialDelayMs: 1000, maxDelayMs: 30000 }\n * )\n * ```\n */\n async moveToDlq(\n eventName: string,\n payload: unknown,\n options: EventOptions,\n error: Error,\n attemptCount: number,\n retryPolicy?: RetryPolicy\n ): Promise<string> {\n const dlqId = randomUUID()\n\n // 如果提供了重試策略,計算下次重試時間\n let nextRetryAt: string | null = null\n if (retryPolicy && this.retryEngine.shouldRetry(attemptCount + 1, retryPolicy)) {\n const delayMs = this.retryEngine.calculateDelay(attemptCount + 1, retryPolicy)\n nextRetryAt = new Date(Date.now() + delayMs).toISOString()\n }\n\n const record = {\n dlq_id: dlqId,\n event_name: eventName,\n event_payload: payload,\n event_options: options,\n attempt_count: attemptCount,\n max_retries: retryPolicy?.maxRetries || 3,\n next_retry_at: nextRetryAt,\n last_error: {\n message: error.message,\n code: (error as any).code,\n stack: error.stack,\n timestamp: new Date().toISOString(),\n },\n status: 'pending',\n resolution_notes: null,\n failed_at: new Date(),\n }\n\n await this.db.table('event_dlq').insert(record)\n\n return dlqId\n }\n\n /**\n * 按 DLQ ID 獲取單個事件\n *\n * @param dlqId - DLQ 事件的 UUID\n * @returns DLQ 記錄,若不存在則返回 undefined\n *\n * @example\n * ```typescript\n * const event = await dlqManager.getById('550e8400-e29b-41d4-a716-446655440000')\n * if (event) {\n * console.log(event.event_name, event.status)\n * }\n * ```\n */\n async getById(dlqId: string): Promise<DLQRecord | undefined> {\n const result = await this.db.table('event_dlq').where('dlq_id', dlqId).first()\n return (result as DLQRecord | null) || undefined\n }\n\n /**\n * 查詢 DLQ 事件\n *\n * @param filter - 查詢過濾條件\n * @returns DLQ 記錄列表(按失敗時間倒序)\n *\n * @example\n * ```typescript\n * // 查詢特定事件的待處理事件\n * const events = await dlqManager.list({\n * eventName: 'order:created',\n * status: 'pending',\n * limit: 50\n * })\n *\n * // 查詢時間範圍內的所有失敗事件\n * const eventsInRange = await dlqManager.list({\n * from: new Date('2026-02-01'),\n * to: new Date('2026-02-03'),\n * limit: 100\n * })\n * ```\n */\n async list(filter: DLQManagerFilter = {}): Promise<DLQRecord[]> {\n let query = this.db.table('event_dlq')\n\n // 按事件名稱篩選\n if (filter.eventName) {\n query = query.where('event_name', filter.eventName)\n }\n\n // 按狀態篩選\n if (filter.status) {\n query = query.where('status', filter.status)\n }\n\n // 按時間範圍篩選\n if (filter.from) {\n query = query.where('failed_at', '>=', filter.from)\n }\n\n if (filter.to) {\n query = query.where('failed_at', '<=', filter.to)\n }\n\n // 排序並分頁\n const results = await query\n .orderBy('failed_at', 'desc')\n .limit(filter.limit || 100)\n .offset(filter.offset || 0)\n .get()\n\n return results as unknown as DLQRecord[]\n }\n\n /**\n * 重新入隊單個 DLQ 事件\n *\n * 重新入隊不會自動派發事件,而是標記狀態為 'requeued'\n * 實際的事件派發應由調用者負責處理\n *\n * @param dlqId - DLQ 事件的 UUID\n * @throws 如果事件不存在\n *\n * @example\n * ```typescript\n * const event = await dlqManager.getById(dlqId)\n * if (event) {\n * // 由調用者決定如何派發事件\n * await eventSystem.doActionAsync(\n * event.event_name,\n * event.event_payload,\n * event.event_options\n * )\n *\n * // 標記為已重新入隊\n * await dlqManager.requeue(dlqId)\n * }\n * ```\n */\n async requeue(dlqId: string): Promise<void> {\n const event = await this.getById(dlqId)\n\n if (!event) {\n throw new Error(`DLQ event not found: ${dlqId}`)\n }\n\n await this.updateStatus(dlqId, 'requeued', `Manual requeue at ${new Date().toISOString()}`)\n }\n\n /**\n * 批量重新入隊 DLQ 事件\n *\n * @param filter - 查詢過濾條件\n * @returns 包含處理結果的統計對象\n *\n * @example\n * ```typescript\n * const result = await dlqManager.retryBatch({\n * eventName: 'order:created',\n * status: 'pending'\n * })\n *\n * console.log(`Success: ${result.succeeded}, Failed: ${result.failed}`)\n * ```\n */\n async retryBatch(\n filter: DLQManagerFilter = {}\n ): Promise<{ total: number; succeeded: number; failed: number }> {\n const events = await this.list(filter)\n let succeeded = 0\n let failed = 0\n\n for (const event of events) {\n try {\n await this.requeue(event.dlq_id)\n succeeded++\n } catch (error) {\n console.error(`Failed to requeue ${event.dlq_id}:`, error)\n failed++\n }\n }\n\n return { total: events.length, succeeded, failed }\n }\n\n /**\n * 標記 DLQ 事件為已解決\n *\n * @param dlqId - DLQ 事件的 UUID\n * @param notes - 解決說明\n *\n * @example\n * ```typescript\n * await dlqManager.resolve(dlqId, 'Manual fix applied: Database issue resolved')\n * ```\n */\n async resolve(dlqId: string, notes?: string): Promise<void> {\n await this.updateStatus(dlqId, 'resolved', notes || `Resolved at ${new Date().toISOString()}`)\n }\n\n /**\n * 放棄 DLQ 事件(不再重試)\n *\n * @param dlqId - DLQ 事件的 UUID\n * @param reason - 放棄原因\n *\n * @example\n * ```typescript\n * await dlqManager.abandon(dlqId, 'Data corrupted, cannot recover')\n * ```\n */\n async abandon(dlqId: string, reason?: string): Promise<void> {\n await this.updateStatus(\n dlqId,\n 'abandoned',\n reason || `Abandoned at ${new Date().toISOString()}`\n )\n }\n\n /**\n * 更新 DLQ 事件狀態\n *\n * @param dlqId - DLQ 事件的 UUID\n * @param status - 新狀態\n * @param notes - 狀態變更說明\n * @throws 如果事件不存在\n *\n * @internal\n */\n async updateStatus(\n dlqId: string,\n status: 'pending' | 'requeued' | 'resolved' | 'abandoned',\n notes?: string\n ): Promise<void> {\n const event = await this.getById(dlqId)\n\n if (!event) {\n throw new Error(`DLQ event not found: ${dlqId}`)\n }\n\n await this.db\n .table('event_dlq')\n .where('dlq_id', dlqId)\n .update({\n status,\n resolution_notes: notes || event.resolution_notes,\n updated_at: new Date(),\n })\n }\n\n /**\n * 刪除單個 DLQ 事件\n *\n * @param dlqId - DLQ 事件的 UUID\n * @returns true 如果刪除成功,false 如果事件不存在\n *\n * @example\n * ```typescript\n * const deleted = await dlqManager.deleteEntry(dlqId)\n * if (deleted) {\n * console.log('Event deleted')\n * }\n * ```\n */\n async deleteEntry(dlqId: string): Promise<boolean> {\n const result = await this.db.table('event_dlq').where('dlq_id', dlqId).delete()\n\n return result > 0\n }\n\n /**\n * 刪除多個 DLQ 事件\n *\n * @param dlqIds - DLQ 事件 UUID 列表\n * @returns 刪除的事件數量\n *\n * @example\n * ```typescript\n * const deletedCount = await dlqManager.deleteEntries([id1, id2, id3])\n * ```\n */\n async deleteEntries(dlqIds: string[]): Promise<number> {\n if (dlqIds.length === 0) {\n return 0\n }\n\n return this.db.table('event_dlq').whereIn('dlq_id', dlqIds).delete()\n }\n\n /**\n * 獲取 DLQ 統計信息\n *\n * @returns 包含總數、按事件名稱和狀態的統計信息\n *\n * @example\n * ```typescript\n * const stats = await dlqManager.getStats()\n *\n * console.log(`Total events: ${stats.total}`)\n * console.log('By event:', stats.byEvent)\n * // Output: { 'order:created': 145, 'payment:succeeded': 23 }\n *\n * console.log('By status:', stats.byStatus)\n * // Output: { pending: 120, requeued: 15, resolved: 8, abandoned: 2 }\n * ```\n */\n async getStats(): Promise<DLQStats> {\n // 獲取總數\n const total = await this.db.table('event_dlq').count()\n\n // 按事件名稱分組統計(使用 raw SQL 以確保跨數據庫兼容)\n const byEventResult = await this.db.raw(\n 'SELECT event_name, COUNT(*) as count FROM event_dlq GROUP BY event_name'\n )\n\n const byEvent: Record<string, number> = {}\n for (const row of byEventResult.rows) {\n byEvent[row.event_name] = Number(row.count)\n }\n\n // 按狀態分組統計(使用 raw SQL 以確保跨數據庫兼容)\n const byStatusResult = await this.db.raw(\n 'SELECT status, COUNT(*) as count FROM event_dlq GROUP BY status'\n )\n\n const byStatus: Record<string, number> = {}\n for (const row of byStatusResult.rows) {\n byStatus[row.status] = Number(row.count)\n }\n\n return {\n total,\n byEvent,\n byStatus,\n }\n }\n\n /**\n * 獲取某個事件名稱的 DLQ 事件數\n *\n * @param eventName - 事件名稱\n * @returns 事件數量\n *\n * @example\n * ```typescript\n * const count = await dlqManager.getCountByEvent('order:created')\n * ```\n */\n async getCountByEvent(eventName: string): Promise<number> {\n return this.db.table('event_dlq').where('event_name', eventName).count()\n }\n\n /**\n * 清空所有 DLQ 事件(慎用!)\n *\n * @param includeResolved - 是否包含已解決的事件\n * @returns 清空的事件數量\n *\n * @example\n * ```typescript\n * // 只清空待處理的事件\n * const count = await dlqManager.clear(false)\n *\n * // 清空所有事件(包括已解決和已放棄)\n * const countAll = await dlqManager.clear(true)\n * ```\n */\n async clear(includeResolved = false): Promise<number> {\n let query = this.db.table('event_dlq')\n\n if (!includeResolved) {\n query = query.where('status', 'pending')\n }\n\n return query.delete()\n }\n\n /**\n * 通過 Bull Job ID 查找 DLQ 記錄\n *\n * 用於 MessageQueueBridge 集成:當 Bull Queue job 失敗時,\n * 通過 Job ID 查找相應的 DLQ 記錄。\n *\n * @param bullJobId - Bull Queue Job ID\n * @returns DLQ 記錄或 undefined\n *\n * @example\n * ```typescript\n * const record = await dlqManager.findByBullJobId('job-abc123')\n * if (record) {\n * console.log(`Event ${record.event_name} is in DLQ`)\n * }\n * ```\n */\n async findByBullJobId(bullJobId: string): Promise<DLQRecord | undefined> {\n try {\n const records = (await this.db\n .table('event_dlq')\n .whereRaw('event_options LIKE ?', [`%\"bullJobId\":\"${bullJobId}\"%`])\n .limit(1)\n .get()) as unknown as DLQRecord[]\n\n return records && records.length > 0 ? records[0] : undefined\n } catch (error) {\n console.error(`[DeadLetterQueueManager] Error finding record by Bull Job ID:`, error)\n return undefined\n }\n }\n\n /**\n * 調度延遲重試(整合 Bull Queue delayed jobs)\n *\n * 從 DLQ 中提取事件,通過 Bull Queue 的延遲 job 功能進行重試。\n *\n * @param dlqId - DLQ 記錄 ID\n * @param delayMs - 延遲時間(毫秒)\n * @throws 如果記錄不存在\n *\n * @example\n * ```typescript\n * // 延遲 1 小時後重試\n * await dlqManager.scheduleRetry('dlq-uuid-123', 3600000)\n * ```\n */\n async scheduleRetry(dlqId: string, delayMs: number): Promise<void> {\n // 1. 從 DLQ 獲取事件\n const record = await this.getById(dlqId)\n if (!record) {\n throw new Error(`[DeadLetterQueueManager] DLQ record not found: ${dlqId}`)\n }\n\n // 2. 使用 Bull Queue 創建延遲任務\n // TODO: Phase 4 Task 3 - 整合 MessageQueueBridge.dispatchDeferredQueued()\n // 當前作為 stub 實現,在完整實現中應該通過 HookManager 調用 dispatchDeferredQueued\n\n console.info(\n `[DeadLetterQueueManager] Scheduled retry for event ${record.event_name} (dlq_id: ${dlqId}) with delay ${delayMs}ms`\n )\n\n // 3. 更新 DLQ 狀態\n await this.requeue(dlqId)\n }\n}\n",
|
|
74
|
+
"import { DeadLetterQueue } from './events/DeadLetterQueue'\nimport type { EventBackend } from './events/EventBackend'\nimport type { EventOptions } from './events/EventOptions'\nimport { EventPriorityQueue } from './events/EventPriorityQueue'\nimport { ActionManager } from './hooks/ActionManager'\nimport { AsyncDetector } from './hooks/AsyncDetector'\nimport {\n createPersistentDLQHandler,\n requeueDLQBatch as dlqRequeueBatch,\n requeueDLQEntry as dlqRequeueEntry,\n requeuePersistentDLQBatch as persistentDlqRequeueBatch,\n requeuePersistentDLQEntry as persistentDlqRequeueEntry,\n} from './hooks/dlq-operations'\nimport { FilterManager } from './hooks/FilterManager'\nimport { MigrationWarner } from './hooks/MigrationWarner'\nimport { DeadLetterQueueManager } from './reliability/DeadLetterQueueManager'\n\n// 重新匯出所有 hook 相關型別,維持向後相容性\nexport type {\n ActionCallback,\n FilterCallback,\n HookManagerConfig,\n ListenerInfo,\n ListenerOptions,\n} from './hooks/types'\n\n// 導入類型供內部使用\nimport type {\n ActionCallback,\n FilterCallback,\n HookManagerConfig,\n ListenerInfo,\n ListenerOptions,\n} from './hooks/types'\n\n/**\n * Manager for WordPress-style hooks (actions and filters).\n *\n * 此為 facade 類別,將職責委派給專門的管理器:\n * - FilterManager:處理 filter hook 的登記與執行\n * - ActionManager:處理 action hook 的登記與執行\n * - AsyncDetector:非同步偵測快取\n * - MigrationWarner:遷移警告管理\n *\n * @public\n */\nexport class HookManager {\n private filterManager: FilterManager\n private actionManager: ActionManager\n private asyncDetector: AsyncDetector\n private migrationWarner: MigrationWarner\n\n private eventQueue: EventPriorityQueue\n private backend: EventBackend\n private dlq?: DeadLetterQueue\n private persistentDlqManager?: DeadLetterQueueManager\n private messageQueueBridge?: any\n private config: HookManagerConfig\n\n constructor(config: HookManagerConfig = {}) {\n this.config = {\n asyncByDefault: false,\n migrationMode: 'sync',\n showDeprecationWarnings: false,\n enableDLQ: true,\n enablePersistentDLQ: false,\n ...config,\n }\n this.eventQueue = new EventPriorityQueue(config.queue)\n this.backend = config.backend || this.eventQueue\n\n // 初始化專門管理器\n this.asyncDetector = new AsyncDetector()\n this.migrationWarner = new MigrationWarner()\n this.filterManager = new FilterManager()\n this.actionManager = new ActionManager(\n this.backend,\n this.config,\n this.asyncDetector,\n this.migrationWarner\n )\n\n // 初始化記憶體內 DLQ\n if (config.enableDLQ ?? true) {\n if (this.backend instanceof EventPriorityQueue) {\n this.dlq = new DeadLetterQueue()\n this.eventQueue.setDeadLetterQueue(this.dlq)\n }\n }\n\n // 若有啟用,初始化持久化 DLQ\n if (config.enablePersistentDLQ && config.db) {\n this.persistentDlqManager = new DeadLetterQueueManager(config.db)\n }\n\n // 若持久化 DLQ manager 存在,設定 EventPriorityQueue 的持久化 DLQ 處理器\n if (this.persistentDlqManager && this.backend instanceof EventPriorityQueue) {\n this.eventQueue.setPersistentDLQHandler(this.createPersistentDLQHandler())\n }\n\n // 初始化 Message Queue Bridge for distributed processing\n if (config.messageQueueBridge) {\n this.messageQueueBridge = config.messageQueueBridge\n }\n }\n\n // ========== Filter Methods(委派至 FilterManager)==========\n\n /**\n * Register a filter hook.\n *\n * Filters are used to transform a value (input/output) through a chain of\n * callbacks. Each callback must return the modified value.\n *\n * @template T - The type of value being filtered.\n * @param hook - The unique name of the hook.\n * @param callback - The callback function to execute.\n *\n * @example\n * ```typescript\n * core.hooks.addFilter('content', async (content: string) => {\n * return content.toUpperCase()\n * })\n * ```\n */\n addFilter<T = unknown>(hook: string, callback: FilterCallback<T>): void {\n this.filterManager.addFilter(hook, callback)\n }\n\n /**\n * Apply all registered filters sequentially.\n *\n * Each callback receives the previous callback's return value.\n *\n * @template T - The type of value being filtered.\n * @param hook - The name of the hook.\n * @param initialValue - The initial value to filter.\n * @param args - Additional arguments to pass to the callbacks.\n * @returns The final filtered value.\n *\n * @example\n * ```typescript\n * const content = await core.hooks.applyFilters('content', 'hello world')\n * ```\n */\n async applyFilters<T = unknown>(hook: string, initialValue: T, ...args: unknown[]): Promise<T> {\n return this.filterManager.applyFilters(hook, initialValue, ...args)\n }\n\n // ========== Action Methods(委派至 ActionManager)==========\n\n /**\n * Register an action hook.\n *\n * Actions are used to trigger side effects (e.g., logging, sending emails)\n * at specific points in the application lifecycle.\n *\n * @template TArgs - The type of arguments passed to the action.\n * @param hook - The unique name of the hook.\n * @param callback - The callback function to execute.\n * @param options - Optional listener options (type override, circuit breaker).\n *\n * @example\n * ```typescript\n * core.hooks.addAction('user_registered', async (user: User) => {\n * await sendWelcomeEmail(user)\n * })\n * ```\n */\n addAction<TArgs = unknown>(\n hook: string,\n callback: ActionCallback<TArgs>,\n options?: ListenerOptions\n ): void {\n this.actionManager.addAction(hook, callback, options)\n }\n\n /**\n * Run all registered actions.\n *\n * This method supports both synchronous and asynchronous dispatch based on configuration.\n * In hybrid mode, it auto-detects async listeners and uses async dispatch.\n *\n * 注意:dispatch 模式決策在此層級完成,以確保子類別(如 ObservableHookManager)\n * 可透過多型覆寫正確攔截 doActionSync / doActionAsync 的呼叫。\n *\n * @template TArgs - The type of arguments passed to the action.\n * @param hook - The name of the hook.\n * @param args - The arguments to pass to the callbacks.\n * @param options - Optional event options for async dispatch.\n *\n * @example\n * ```typescript\n * await core.hooks.doAction('user_registered', user)\n * ```\n */\n async doAction<TArgs = unknown>(\n hook: string,\n args: TArgs,\n options?: EventOptions\n ): Promise<void> {\n const mode = this.actionManager.resolveDispatchMode(hook, args, options)\n\n if (mode === 'async') {\n return this.doActionAsync(hook, args, options)\n }\n\n return this.doActionSync(hook, args)\n }\n\n /**\n * Run all registered actions synchronously (legacy mode).\n *\n * @template TArgs - The type of arguments passed to the action.\n * @param hook - The name of the hook.\n * @param args - The arguments to pass to the callbacks.\n */\n async doActionSync<TArgs = unknown>(hook: string, args: TArgs): Promise<void> {\n return this.actionManager.doActionSync(hook, args)\n }\n\n /**\n * Run all registered actions asynchronously via priority queue.\n *\n * @template TArgs - The type of arguments passed to the action.\n * @param hook - The name of the hook.\n * @param args - The arguments to pass to the callbacks.\n * @param options - Event options for async dispatch.\n *\n * @example\n * ```typescript\n * await core.hooks.doActionAsync('order:created', order, {\n * priority: 'high',\n * ordering: 'partition',\n * partitionKey: order.id,\n * timeout: 5000,\n * })\n * ```\n */\n async doActionAsync<TArgs = unknown>(\n hook: string,\n args: TArgs,\n options: EventOptions = {}\n ): Promise<void> {\n return this.actionManager.doActionAsync(hook, args, options)\n }\n\n // ========== Dispatch Mode Detection(委派至 ActionManager)==========\n\n /**\n * Determine the dispatch mode for an event.\n *\n * @param eventName - The name of the event\n * @param options - Optional event options\n * @returns The dispatch mode: 'sync' or 'async'\n * @public\n */\n detectMode(eventName: string, options?: EventOptions): 'sync' | 'async' {\n return this.actionManager.detectMode(eventName, options)\n }\n\n // ========== Async Detection(委派至 AsyncDetector)==========\n\n /**\n * Check if a callback is an async function (with caching).\n *\n * @param callback - The callback to check\n * @returns True if the callback is async\n * @public\n */\n isAsyncListener(callback: ActionCallback): boolean {\n return this.asyncDetector.isAsyncListener(callback)\n }\n\n /**\n * Runtime detection for functions that return Promises but aren't declared async.\n *\n * @param callback - The callback to check\n * @param testArgs - Arguments to pass to the callback for testing\n * @returns True if the callback returns a Promise\n * @public\n */\n async isAsyncListenerRuntime<TArgs = unknown>(\n callback: ActionCallback<TArgs>,\n testArgs: TArgs\n ): Promise<boolean> {\n return this.asyncDetector.isAsyncListenerRuntime(callback, testArgs)\n }\n\n /**\n * Get the size of the async detection cache (for testing/debugging).\n *\n * @returns Number of cached detection results\n * @public\n */\n getAsyncDetectionCacheSize(): number {\n return this.asyncDetector.getCacheSize()\n }\n\n /**\n * Clear the async detection cache.\n *\n * @public\n */\n clearAsyncDetectionCache(): void {\n this.asyncDetector.clearCache()\n }\n\n /**\n * Check if any listener for a hook is async (including type overrides).\n *\n * @param hook - Hook name\n * @returns True if any listener is async\n * @public\n */\n hasAsyncListeners(hook: string): boolean {\n return this.actionManager.hasAsyncListeners(hook)\n }\n\n /**\n * Get detailed information about all listeners for a hook.\n *\n * @param hook - Hook name\n * @returns Array of listener info objects\n * @public\n */\n getListenerInfo(hook: string): ListenerInfo[] {\n return this.actionManager.getListenerInfo(hook)\n }\n\n // ========== Queue Methods ==========\n\n /**\n * Get the current event queue depth.\n *\n * @returns Total number of events in the queue\n */\n getQueueDepth(): number {\n return this.eventQueue.getDepth()\n }\n\n /**\n * Get the event queue depth for a specific priority.\n *\n * @param priority - Priority level\n * @returns Number of events in the specified priority queue\n */\n getQueueDepthByPriority(priority: 'high' | 'normal' | 'low'): number {\n return this.eventQueue.getDepthByPriority(priority)\n }\n\n /**\n * Get the EventPriorityQueue instance.\n *\n * @returns EventPriorityQueue instance\n * @protected\n */\n protected getEventQueue(): EventPriorityQueue {\n return this.eventQueue\n }\n\n // ========== Listener Management ==========\n\n /**\n * Get all registered listeners for a hook.\n *\n * @param hook - Hook name\n * @returns Array of callbacks\n */\n getListeners(hook: string): ActionCallback[] {\n return this.actionManager.getListeners(hook)\n }\n\n /**\n * Remove all listeners for a specific action hook.\n *\n * @param hook - Hook name\n */\n removeAction(hook: string): void {\n this.actionManager.removeAction(hook)\n }\n\n // ========== Configuration ==========\n\n /**\n * Update HookManager configuration.\n *\n * @param config - New configuration\n */\n configure(config: Partial<HookManagerConfig>): void {\n this.config = {\n ...this.config,\n ...config,\n }\n this.actionManager.updateConfig(this.config)\n }\n\n /**\n * Get current configuration.\n *\n * @returns Current configuration\n */\n getConfig(): HookManagerConfig {\n return { ...this.config }\n }\n\n /**\n * Suppress migration warnings for a specific event.\n *\n * @param eventName - The name of the event to suppress warnings for\n * @public\n */\n suppressMigrationWarning(eventName: string): void {\n this.migrationWarner.suppress(eventName)\n }\n\n /**\n * Set the event backend for distributed processing.\n *\n * @param backend - Event backend instance\n */\n setBackend(backend: EventBackend): void {\n this.backend = backend\n this.actionManager.setBackend(backend)\n }\n\n // ========== Dead Letter Queue (In-Memory) ==========\n\n /**\n * Get the Dead Letter Queue instance.\n *\n * @returns Dead Letter Queue\n */\n getDLQ(): DeadLetterQueue | undefined {\n return this.dlq\n }\n\n /**\n * Requeue a failed event from the Dead Letter Queue.\n *\n * @param dlqEntryId - DLQ entry ID\n * @returns True if requeued successfully, false if entry not found\n */\n async requeueDLQEntry(dlqEntryId: string): Promise<boolean> {\n if (!this.dlq) {\n return false\n }\n return dlqRequeueEntry(dlqEntryId, this.dlq, (eventName, payload, options) =>\n this.doActionAsync(eventName, payload, options)\n )\n }\n\n /**\n * Requeue all failed events for a specific event name.\n *\n * @param eventName - Event name to requeue\n * @returns Number of events requeued\n */\n async requeueDLQBatch(eventName: string): Promise<number> {\n if (!this.dlq) {\n return 0\n }\n return dlqRequeueBatch(eventName, this.dlq, (entryId) => this.requeueDLQEntry(entryId))\n }\n\n /**\n * Get Dead Letter Queue entries with optional filtering.\n *\n * @param filter - Filter options\n * @returns Array of DLQ entries\n */\n getDLQEntries(filter: { eventName?: string; from?: number; to?: number; limit?: number } = {}) {\n if (!this.dlq) {\n return []\n }\n return this.dlq.list(filter)\n }\n\n /**\n * Get count of Dead Letter Queue entries for an event.\n *\n * @param eventName - Event name\n * @returns Count of entries\n */\n getDLQCount(eventName: string): number {\n if (!this.dlq) {\n return 0\n }\n return this.dlq.getCountByEvent(eventName)\n }\n\n /**\n * Delete a DLQ entry.\n *\n * @param entryId - DLQ entry ID\n * @returns True if deleted, false if not found\n */\n deleteDLQEntry(entryId: string): boolean {\n if (!this.dlq) {\n return false\n }\n return this.dlq.delete(entryId)\n }\n\n // ========== Circuit Breaker ==========\n\n /**\n * Get circuit breaker statistics for all events.\n *\n * @returns Array of circuit breaker statistics\n */\n getCircuitBreakerStats(): Array<{\n eventName: string\n state: string\n failureCount: number\n successCount: number\n lastFailureAt?: Date\n lastSuccessAt?: Date\n openedAt?: Date\n totalRequests: number\n totalFailures: number\n totalSuccesses: number\n }> {\n if (!(this.backend instanceof EventPriorityQueue)) {\n return []\n }\n\n const breakers = this.backend.getCircuitBreakers()\n const stats: Array<{\n eventName: string\n state: string\n failureCount: number\n successCount: number\n lastFailureAt?: Date\n lastSuccessAt?: Date\n openedAt?: Date\n totalRequests: number\n totalFailures: number\n totalSuccesses: number\n }> = []\n\n for (const [eventName, breaker] of breakers) {\n const metrics = breaker.getMetrics()\n stats.push({\n eventName,\n state: metrics.state,\n failureCount: metrics.failures,\n successCount: metrics.successes,\n lastFailureAt: metrics.lastFailureAt,\n lastSuccessAt: metrics.lastSuccessAt,\n openedAt: metrics.openedAt,\n totalRequests: metrics.totalRequests,\n totalFailures: metrics.totalFailures,\n totalSuccesses: metrics.totalSuccesses,\n })\n }\n\n return stats\n }\n\n /**\n * Reset a circuit breaker for an event.\n *\n * @param eventName - Event name\n * @returns True if reset, false if circuit breaker not found\n */\n resetCircuitBreaker(eventName: string): boolean {\n if (!(this.backend instanceof EventPriorityQueue)) {\n return false\n }\n\n return this.backend.resetCircuitBreaker(eventName)\n }\n\n // ========== Backpressure ==========\n\n /**\n * Get the current backpressure state.\n *\n * @returns Backpressure state ('NORMAL', 'WARNING', 'CRITICAL', 'OVERFLOW') or undefined if not enabled\n */\n getBackpressureState(): string | undefined {\n if (!(this.backend instanceof EventPriorityQueue)) {\n return undefined\n }\n\n const manager = this.backend.getBackpressureManager()\n return manager ? manager.getState() : undefined\n }\n\n /**\n * Get backpressure metrics snapshot.\n *\n * @returns Backpressure metrics snapshot or undefined if not enabled\n */\n getBackpressureMetrics(): object | undefined {\n if (!(this.backend instanceof EventPriorityQueue)) {\n return undefined\n }\n\n const manager = this.backend.getBackpressureManager()\n return manager ? manager.getMetrics() : undefined\n }\n\n // ========== Persistent DLQ ==========\n\n /**\n * Get the persistent DLQ manager instance.\n *\n * @returns DeadLetterQueueManager or undefined if not configured\n * @public\n */\n getPersistentDLQManager(): DeadLetterQueueManager | undefined {\n return this.persistentDlqManager\n }\n\n /**\n * Create a handler function for persistent DLQ.\n * This handler is used by EventPriorityQueue to persist failed events.\n *\n * @returns Handler function\n * @internal\n */\n private createPersistentDLQHandler() {\n if (!this.persistentDlqManager) {\n throw new Error('[HookManager] persistentDlqManager is not initialized')\n }\n return createPersistentDLQHandler(this.persistentDlqManager)\n }\n\n /**\n * Requeue a failed event from the persistent DLQ.\n *\n * @param dlqId - DLQ entry UUID\n * @returns True if requeued successfully\n * @public\n */\n async requeuePersistentDLQEntry(dlqId: string): Promise<boolean> {\n if (!this.persistentDlqManager) {\n return false\n }\n return persistentDlqRequeueEntry(dlqId, this.persistentDlqManager, (event, args, options) =>\n this.doActionAsync(event, args, options)\n )\n }\n\n /**\n * Requeue multiple events from persistent DLQ.\n *\n * @param filter - Filter criteria\n * @returns Result statistics\n * @public\n */\n async requeuePersistentDLQBatch(filter?: {\n eventName?: string\n status?: 'pending' | 'requeued' | 'resolved' | 'abandoned'\n }): Promise<{ total: number; succeeded: number; failed: number }> {\n if (!this.persistentDlqManager) {\n return { total: 0, succeeded: 0, failed: 0 }\n }\n return persistentDlqRequeueBatch(filter, this.persistentDlqManager)\n }\n\n // ========== Message Queue Bridge(分佈式處理)==========\n\n /**\n * Dispatch an event through Bull Queue(分佈式異步處理).\n *\n * 與 doActionAsync() 不同:\n * - doActionAsync() 使用 EventPriorityQueue (Memory-based)\n * - dispatchQueued() 使用 Bull Queue (Redis-backed, 分佈式)\n *\n * 適用於需要持久化和跨進程處理的事件。\n *\n * @template TArgs - 事件參數類型\n * @param event - 事件名稱\n * @param args - 事件參數\n * @param options - 事件選項(可選)\n * @returns Job ID\n * @throws 如果未配置 MessageQueueBridge 或沒有 listeners\n *\n * @example\n * ```typescript\n * const jobId = await hookManager.dispatchQueued('order:created', {\n * orderId: 'ORD-123',\n * amount: 999.99\n * })\n * ```\n *\n * @public\n */\n async dispatchQueued<TArgs = unknown>(\n event: string,\n args: TArgs,\n options?: any\n ): Promise<string> {\n if (!this.messageQueueBridge) {\n throw new Error(\n '[HookManager] MessageQueueBridge not configured. Set messageQueueBridge in HookManagerConfig to use dispatchQueued().'\n )\n }\n\n return this.messageQueueBridge.dispatchWithQueue(event, args, options)\n }\n\n /**\n * Dispatch an event through Bull Queue with delay(延遲隊列分發).\n *\n * @template TArgs - 事件參數類型\n * @param event - 事件名稱\n * @param args - 事件參數\n * @param delay - 延遲時間(毫秒)\n * @param options - 事件選項(可選)\n * @returns Job ID\n * @throws 如果未配置 MessageQueueBridge\n *\n * @example\n * ```typescript\n * // 延遲 5 秒後處理\n * const jobId = await hookManager.dispatchDeferredQueued(\n * 'reminder:send',\n * { userId: '123' },\n * 5000\n * )\n * ```\n *\n * @public\n */\n async dispatchDeferredQueued<TArgs = unknown>(\n event: string,\n args: TArgs,\n delay: number,\n options?: any\n ): Promise<string> {\n if (!this.messageQueueBridge) {\n throw new Error(\n '[HookManager] MessageQueueBridge not configured. Set messageQueueBridge in HookManagerConfig to use dispatchDeferredQueued().'\n )\n }\n\n // 將延遲時間添加到 options\n const mergedOptions = {\n ...options,\n delay,\n }\n\n return this.messageQueueBridge.dispatchWithQueue(event, args, mergedOptions)\n }\n\n /**\n * Get the execution status of an event.\n *\n * 查詢事件執行狀態,支持查詢 Bull Queue 和 DLQ 中的事件。\n *\n * @param eventId - 事件 ID (task.id 或 jobId)\n * @returns 事件狀態信息\n * @throws 如果未配置 MessageQueueBridge\n *\n * @public\n */\n async getEventStatus(eventId: string): Promise<any> {\n if (!this.messageQueueBridge) {\n throw new Error(\n '[HookManager] MessageQueueBridge not configured. Set messageQueueBridge in HookManagerConfig to use getEventStatus().'\n )\n }\n\n return this.messageQueueBridge.getEventStatus(eventId)\n }\n\n /**\n * Get persistent DLQ statistics.\n *\n * @returns Statistics object with total, byEvent, and byStatus counts\n * @public\n */\n async getPersistentDLQStats() {\n if (!this.persistentDlqManager) {\n return undefined\n }\n\n try {\n return await this.persistentDlqManager.getStats()\n } catch (error) {\n console.error(`[HookManager] Error getting persistent DLQ stats:`, error)\n return undefined\n }\n }\n}\n",
|
|
75
|
+
"import type { ConfigManager } from './ConfigManager'\nimport type { Container } from './Container'\nimport type { PlanetCore } from './PlanetCore'\n\n/**\n * ServiceProvider - The foundation for modular service registration.\n *\n * Service providers are the central place to configure your application.\n * They bind services to the container and bootstrap application features.\n *\n * Lifecycle:\n * 1. register() - Called during registration phase (sync or async)\n * 2. boot() - Called after ALL providers have registered\n *\n * @since 1.0.0\n * @example\n * ```typescript\n * class DatabaseServiceProvider extends ServiceProvider {\n * register(container: Container) {\n * container.singleton('db', () => new DatabaseManager());\n * }\n *\n * boot(core: PlanetCore) {\n * const db = core.container.make<DatabaseManager>('db');\n * db.setDefaultConnection(core.config.get('database.default'));\n * }\n * }\n * ```\n */\nexport abstract class ServiceProvider {\n /**\n * Reference to the application core instance.\n * Set during provider registration.\n */\n protected core?: PlanetCore\n\n /**\n * Whether this provider should be deferred.\n * Deferred providers are only registered when one of their\n * provided services is actually requested from the container.\n */\n public deferred = false\n\n /**\n * Get the services provided by this provider.\n * Used for deferred loading - provider is only loaded when\n * one of these services is requested.\n *\n * @returns Array of service keys this provider offers\n *\n * @example\n * ```typescript\n * provides(): string[] {\n * return ['db', 'db.connection'];\n * }\n * ```\n */\n provides(): string[] {\n return []\n }\n\n /**\n * Register bindings in the container.\n *\n * This method is called during the registration phase.\n * **Warning**: Do not resolve services from other providers here,\n * as they may not be registered yet.\n *\n * Supports both synchronous and asynchronous registration.\n *\n * @param container - The IoC container instance\n */\n abstract register(container: Container): void | Promise<void>\n\n /**\n * Bootstrap any application services.\n *\n * This method is called after ALL providers have registered.\n * You can safely resolve services from the container here.\n *\n * @param core - The PlanetCore application instance\n */\n boot?(core: PlanetCore): void | Promise<void>\n\n /**\n * Called when the application is ready to accept requests.\n *\n * This method is called after ALL providers have booted.\n * Use this for final initialization before the server starts accepting traffic.\n *\n * @param core - The PlanetCore application instance\n * @since 2.2.0\n */\n onReady?(core: PlanetCore): void | Promise<void>\n\n /**\n * Called when the application is shutting down.\n *\n * This method is called during graceful shutdown.\n * Use this to clean up resources, close connections, etc.\n *\n * Providers are called in reverse order (LIFO).\n *\n * @param core - The PlanetCore application instance\n * @since 2.2.0\n */\n onShutdown?(core: PlanetCore): void | Promise<void>\n\n /**\n * Set the core instance reference.\n * Called internally by the application during registration.\n *\n * @internal\n */\n setCore(core: PlanetCore): void {\n this.core = core\n }\n\n // ─────────────────────────────────────────────────────────────\n // Configuration Helpers\n // ─────────────────────────────────────────────────────────────\n\n /**\n * Merge configuration from a value into the application config.\n *\n * If the configuration key already exists and both the existing value and\n * the new value are objects, they will be shallow-merged. Otherwise, the\n * new value will overwrite the existing one.\n *\n * @param config - The ConfigManager instance.\n * @param key - The configuration key to set (supports dot notation).\n * @param value - The configuration value or object to merge.\n *\n * @example\n * ```typescript\n * this.mergeConfig(config, 'database', {\n * default: 'mysql',\n * connections: { ... }\n * });\n * ```\n */\n protected mergeConfig(config: ConfigManager, key: string, value: unknown): void {\n const existing = config.has(key) ? config.get(key) : {}\n\n if (\n typeof existing === 'object' &&\n existing !== null &&\n typeof value === 'object' &&\n value !== null\n ) {\n // Deep merge for objects\n config.set(key, { ...existing, ...value })\n } else {\n config.set(key, value)\n }\n }\n\n /**\n * Merge configuration from an async loader.\n * Useful for loading config from .ts files dynamically.\n *\n * @param config - The ConfigManager instance\n * @param key - The configuration key\n * @param loader - Async function that returns config value\n *\n * @example\n * ```typescript\n * await this.mergeConfigFrom(config, 'database', async () => {\n * return (await import('./config/database')).default;\n * });\n * ```\n */\n protected async mergeConfigFrom(\n config: ConfigManager,\n key: string,\n loader: () => Promise<unknown>\n ): Promise<void> {\n const value = await loader()\n this.mergeConfig(config, key, value)\n }\n\n // ─────────────────────────────────────────────────────────────\n // Publishing (for CLI support)\n // ─────────────────────────────────────────────────────────────\n\n /**\n * Paths that should be published by the CLI.\n * Maps source paths to destination paths.\n */\n private static publishables: Map<string, Map<string, string>> = new Map()\n\n /**\n * Register paths to be published by the CLI.\n *\n * Used by CLI commands like `gravito vendor:publish` to copy configuration,\n * views, or assets from the package to the application directory.\n *\n * @param paths - A record mapping source paths to destination paths.\n * @param group - Optional group name for selective publishing (e.g., 'config', 'views').\n *\n * @example\n * ```typescript\n * this.publishes({\n * './config/cache.ts': 'config/cache.ts',\n * './views/errors': 'resources/views/errors'\n * }, 'config');\n * ```\n */\n protected publishes(paths: Record<string, string>, group?: string): void {\n const groupKey = group ?? this.constructor.name\n\n if (!ServiceProvider.publishables.has(groupKey)) {\n ServiceProvider.publishables.set(groupKey, new Map())\n }\n\n const groupPaths = ServiceProvider.publishables.get(groupKey)!\n for (const [source, dest] of Object.entries(paths)) {\n groupPaths.set(source, dest)\n }\n }\n\n /**\n * Get all publishable paths for a group.\n *\n * @param group - The group name (defaults to provider class name)\n * @returns Map of source to destination paths\n */\n static getPublishables(group?: string): Map<string, string> {\n if (group) {\n return ServiceProvider.publishables.get(group) ?? new Map()\n }\n\n // Return all publishables merged\n const all = new Map<string, string>()\n for (const paths of ServiceProvider.publishables.values()) {\n for (const [source, dest] of paths) {\n all.set(source, dest)\n }\n }\n return all\n }\n\n /**\n * Get all publish groups.\n *\n * @returns Array of group names\n */\n static getPublishGroups(): string[] {\n return Array.from(ServiceProvider.publishables.keys())\n }\n}\n",
|
|
76
|
+
"/**\n * @fileoverview HealthProvider - Cloud-native health checks\n *\n * Provides liveness and readiness probes for Kubernetes and cloud deployments\n *\n * @module @gravito/core/health\n * @since 2.2.0\n */\n\nimport type { Container } from '../Container'\nimport type { PlanetCore } from '../PlanetCore'\nimport { ServiceProvider } from '../ServiceProvider'\n\n/**\n * Health check result\n * @public\n */\nexport interface HealthCheckResult {\n /** Health status */\n status: 'healthy' | 'unhealthy'\n /** Optional message */\n message?: string\n}\n\n/**\n * Health check function interface\n * @public\n */\nexport interface HealthCheck {\n /** Unique name for this check */\n name: string\n /** Check function that returns health status */\n check(): Promise<HealthCheckResult>\n}\n\n/**\n * HealthProvider - Provides /health/liveness and /health/readiness endpoints\n *\n * - Liveness: Always returns 200 OK (just checks if server is running)\n * - Readiness: Returns 200 if all checks are healthy, 503 otherwise\n *\n * @public\n */\nexport class HealthProvider extends ServiceProvider {\n private checks: HealthCheck[] = []\n\n register(container: Container) {\n container.singleton('health', () => this)\n }\n\n /**\n * Register a health check\n *\n * @param check - The health check to register\n *\n * @example\n * ```typescript\n * const health = core.container.make<HealthProvider>('health')\n * health.registerCheck({\n * name: 'database',\n * check: async () => {\n * try {\n * await db.ping()\n * return { status: 'healthy' }\n * } catch {\n * return { status: 'unhealthy', message: 'DB unreachable' }\n * }\n * }\n * })\n * ```\n */\n registerCheck(check: HealthCheck): void {\n this.checks.push(check)\n }\n\n boot(core: PlanetCore) {\n // Liveness probe - always 200\n core.router.get('/health/liveness', () => {\n return new Response('OK', { status: 200 })\n })\n\n // Readiness probe - depends on checks\n core.router.get('/health/readiness', async () => {\n const results = await Promise.all(this.checks.map((c) => c.check()))\n\n const checkResults: Record<string, HealthCheckResult> = {}\n for (let i = 0; i < this.checks.length; i++) {\n checkResults[this.checks[i].name] = results[i]\n }\n\n const allHealthy = results.every((r) => r.status === 'healthy')\n\n return Response.json(\n {\n status: allHealthy ? 'healthy' : 'unhealthy',\n timestamp: new Date().toISOString(),\n checks: checkResults,\n },\n {\n status: allHealthy ? 200 : 503,\n }\n )\n })\n }\n}\n\nexport default HealthProvider\n",
|
|
77
|
+
"/**\n * Path segment (key) in a data structure.\n * @public\n */\nexport type PathSegment = string | number\n\n/**\n * Path to a value (dot notation or array of segments).\n * @public\n */\nexport type DataPath = string | readonly PathSegment[]\n\nfunction parsePath(path: DataPath | null | undefined): PathSegment[] {\n if (path === null || path === undefined) {\n return []\n }\n if (typeof path !== 'string') {\n return [...path]\n }\n\n if (path === '') {\n return []\n }\n\n return path.split('.').map((segment) => {\n const n = Number(segment)\n if (Number.isInteger(n) && String(n) === segment) {\n return n\n }\n return segment\n })\n}\n\nfunction getChild(current: unknown, key: PathSegment): unknown {\n if (current === null || current === undefined) {\n return undefined\n }\n\n if (current instanceof Map) {\n return current.get(key)\n }\n\n if (typeof current === 'object' || typeof current === 'function') {\n const record = current as Record<PropertyKey, unknown>\n return record[key as PropertyKey]\n }\n\n return undefined\n}\n\nfunction hasChild(current: unknown, key: PathSegment): boolean {\n if (current === null || current === undefined) {\n return false\n }\n\n if (current instanceof Map) {\n return current.has(key)\n }\n\n if (typeof current === 'object' || typeof current === 'function') {\n const record = current as Record<PropertyKey, unknown>\n return key in record\n }\n\n return false\n}\n\nfunction setChild(current: unknown, key: PathSegment, next: unknown): void {\n if (current === null || current === undefined) {\n throw new TypeError('dataSet target cannot be null or undefined.')\n }\n\n // Prevent prototype pollution\n if (key === '__proto__' || key === 'constructor' || key === 'prototype') {\n return\n }\n\n if (current instanceof Map) {\n current.set(key, next)\n return\n }\n\n if (typeof current === 'object' || typeof current === 'function') {\n const record = current as Record<PropertyKey, unknown>\n record[key as PropertyKey] = next\n return\n }\n\n throw new TypeError('dataSet target must be object-like.')\n}\n\n/**\n * Retrieve a value from a deep object using dot notation.\n * @public\n */\nexport function dataGet<TDefault = undefined>(\n target: unknown,\n path: DataPath | null | undefined,\n defaultValue?: TDefault\n): unknown | TDefault {\n const segments = parsePath(path)\n if (segments.length === 0) {\n return target\n }\n\n let current: unknown = target\n for (const segment of segments) {\n current = getChild(current, segment)\n if (current === undefined) {\n return defaultValue as TDefault\n }\n }\n\n return current\n}\n\n/**\n * Check if a key exists in a deep object using dot notation.\n * @public\n */\nexport function dataHas(target: unknown, path: DataPath | null | undefined): boolean {\n const segments = parsePath(path)\n if (segments.length === 0) {\n return true\n }\n\n let current: unknown = target\n for (const segment of segments) {\n if (!hasChild(current, segment)) {\n return false\n }\n current = getChild(current, segment)\n }\n\n return true\n}\n\n/**\n * Set a value in a deep object using dot notation.\n * @public\n */\nexport function dataSet(\n target: unknown,\n path: DataPath,\n setValue: unknown,\n overwrite = true\n): unknown {\n const segments = parsePath(path)\n if (segments.length === 0) {\n return target\n }\n\n let current: unknown = target\n for (let i = 0; i < segments.length - 1; i++) {\n const segment = segments[i] as PathSegment\n const nextSegment = segments[i + 1] as PathSegment\n\n const existing = getChild(current, segment)\n if (\n existing !== undefined &&\n (typeof existing === 'object' || typeof existing === 'function')\n ) {\n current = existing\n continue\n }\n\n const created = typeof nextSegment === 'number' ? [] : {}\n setChild(current, segment, created)\n current = created\n }\n\n const last = segments[segments.length - 1] as PathSegment\n const existingLast = getChild(current, last)\n if (overwrite || existingLast === undefined) {\n setChild(current, last, setValue)\n }\n\n return target\n}\n",
|
|
78
|
+
"import { type DataPath, dataGet, dataHas, dataSet } from './data'\n\n/**\n * Array and Collection Helpers.\n * Provides utility methods for array manipulation and deep object access.\n * @public\n */\nexport const Arr = {\n get<TDefault = undefined>(\n target: unknown,\n path: DataPath | null | undefined,\n defaultValue?: TDefault\n ): unknown | TDefault {\n return dataGet(target, path, defaultValue)\n },\n\n has(target: unknown, path: DataPath | null | undefined): boolean {\n return dataHas(target, path)\n },\n\n set(target: unknown, path: DataPath, value: unknown, overwrite = true): unknown {\n return dataSet(target, path, value, overwrite)\n },\n\n wrap<T>(value: T | T[] | null | undefined): T[] {\n if (value === null || value === undefined) {\n return []\n }\n return Array.isArray(value) ? value : [value]\n },\n\n first<T>(items: readonly T[], callback?: (value: T, index: number) => boolean): T | undefined {\n if (!callback) {\n return items[0]\n }\n for (let i = 0; i < items.length; i++) {\n const value = items[i] as T\n if (callback(value, i)) {\n return value\n }\n }\n return undefined\n },\n\n last<T>(items: readonly T[], callback?: (value: T, index: number) => boolean): T | undefined {\n if (!callback) {\n return items.length ? items[items.length - 1] : undefined\n }\n for (let i = items.length - 1; i >= 0; i--) {\n const value = items[i] as T\n if (callback(value, i)) {\n return value\n }\n }\n return undefined\n },\n\n only<T extends Record<string, unknown>>(target: T, keys: readonly string[]): Partial<T> {\n const out: Partial<T> = {}\n for (const key of keys) {\n if (key in target) {\n out[key as keyof T] = target[key] as T[keyof T]\n }\n }\n return out\n },\n\n except<T extends Record<string, unknown>>(target: T, keys: readonly string[]): Partial<T> {\n const out: Partial<T> = {}\n const excluded = new Set(keys)\n for (const [key, value] of Object.entries(target)) {\n if (!excluded.has(key)) {\n out[key as keyof T] = value as T[keyof T]\n }\n }\n return out\n },\n\n flatten(items: unknown[], depth: number = Number.POSITIVE_INFINITY): unknown[] {\n const out: unknown[] = []\n const walk = (value: unknown, currentDepth: number) => {\n if (Array.isArray(value) && currentDepth > 0) {\n for (const v of value) {\n walk(v, currentDepth - 1)\n }\n return\n }\n out.push(value)\n }\n for (const item of items) {\n walk(item, depth)\n }\n return out\n },\n\n pluck<TItem extends Record<string, unknown>>(\n items: readonly TItem[],\n valuePath: DataPath,\n keyPath?: DataPath\n ): unknown[] | Record<string, unknown> {\n if (!keyPath) {\n return items.map((item) => dataGet(item, valuePath))\n }\n\n const out: Record<string, unknown> = {}\n for (const item of items) {\n const key = dataGet(item, keyPath)\n out[String(key)] = dataGet(item, valuePath)\n }\n return out\n },\n\n where<T>(items: readonly T[], callback: (value: T, index: number) => boolean): T[] {\n const out: T[] = []\n for (let i = 0; i < items.length; i++) {\n const value = items[i] as T\n if (callback(value, i)) {\n out.push(value)\n }\n }\n return out\n },\n} as const\n",
|
|
79
|
+
"import type { GravitoContext } from '../http/types'\n\n/**\n * Interface for displaying validation errors in views.\n * @public\n */\nexport interface ErrorBag {\n /** Check if a field has errors */\n has(field: string): boolean\n /** Get the first error message for a field (or any first error if no field specified) */\n first(field?: string): string | undefined\n /** Get all error messages for a field */\n get(field: string): string[]\n /** Get all errors for all fields */\n all(): Record<string, string[]>\n /** Check if there are any errors */\n any(): boolean\n /** Get total number of error messages */\n count(): number\n}\n\n/**\n * Create a new ErrorBag instance from raw errors.\n * @public\n */\nexport function createErrorBag(errors: Record<string, string[]>): ErrorBag {\n return {\n has: (field) => (errors[field]?.length ?? 0) > 0,\n first: (field) => {\n if (field) {\n return errors[field]?.[0]\n }\n for (const key of Object.keys(errors)) {\n if (errors[key]?.[0]) {\n return errors[key][0]\n }\n }\n return undefined\n },\n get: (field) => errors[field] ?? [],\n all: () => errors,\n any: () => Object.keys(errors).length > 0,\n count: () => Object.values(errors).flat().length,\n }\n}\n\n/**\n * Helper to retrieve the ErrorBag from session flash data.\n * @public\n */\nexport function errors(c: GravitoContext): ErrorBag {\n const session = c.get('session') as { getFlash?: (key: string) => unknown } | undefined\n const flashed = session?.getFlash?.('errors') ?? {}\n return createErrorBag(flashed as Record<string, string[]>)\n}\n\n/**\n * Helper to retrieve old input value from session flash.\n * @public\n */\nexport function old(c: GravitoContext, field: string, defaultValue?: unknown): unknown {\n const session = c.get('session') as { getFlash?: (key: string) => unknown } | undefined\n const oldInput = session?.getFlash?.('_old_input') ?? {}\n return (oldInput as Record<string, unknown>)[field] ?? defaultValue\n}\n",
|
|
80
|
+
"import { randomBytes, randomUUID } from '../compat/crypto'\nimport { getRuntimeKind } from '../runtime/detection'\n\ntype StartsEndsNeedle = string | readonly string[]\n\nfunction splitWords(input: string): string[] {\n const normalized = input\n .replace(/([a-z0-9])([A-Z])/g, '$1 $2')\n .replace(/[_-]+/g, ' ')\n .trim()\n\n return normalized ? normalized.split(/\\s+/) : []\n}\n\nfunction capitalize(word: string): string {\n if (!word) {\n return word\n }\n return word.charAt(0).toUpperCase() + word.slice(1)\n}\n\n/**\n * String Helper Utilities.\n * Provides methods for string manipulation, case conversion, and UUID generation.\n * @public\n */\nexport const Str = {\n lower(value: string): string {\n return value.toLowerCase()\n },\n\n upper(value: string): string {\n return value.toUpperCase()\n },\n\n startsWith(haystack: string, needles: StartsEndsNeedle): boolean {\n const list = Array.isArray(needles) ? needles : [needles]\n for (const needle of list) {\n if (needle !== '' && haystack.startsWith(needle)) {\n return true\n }\n }\n return false\n },\n\n endsWith(haystack: string, needles: StartsEndsNeedle): boolean {\n const list = Array.isArray(needles) ? needles : [needles]\n for (const needle of list) {\n if (needle !== '' && haystack.endsWith(needle)) {\n return true\n }\n }\n return false\n },\n\n contains(haystack: string, needles: StartsEndsNeedle): boolean {\n const list = Array.isArray(needles) ? needles : [needles]\n for (const needle of list) {\n if (needle !== '' && haystack.includes(needle)) {\n return true\n }\n }\n return false\n },\n\n snake(value: string): string {\n const words = splitWords(value).map((w) => w.toLowerCase())\n return words.join('_')\n },\n\n kebab(value: string): string {\n const words = splitWords(value).map((w) => w.toLowerCase())\n return words.join('-')\n },\n\n studly(value: string): string {\n return splitWords(value)\n .map((w) => capitalize(w.toLowerCase()))\n .join('')\n },\n\n camel(value: string): string {\n const words = splitWords(value).map((w) => w.toLowerCase())\n if (words.length === 0) {\n return ''\n }\n const first = words[0]\n if (first === undefined) {\n return ''\n }\n return first + words.slice(1).map(capitalize).join('')\n },\n\n title(value: string): string {\n return splitWords(value)\n .map((w) => capitalize(w.toLowerCase()))\n .join(' ')\n },\n\n limit(value: string, limit: number, end = '...'): string {\n if (limit < 0) {\n return ''\n }\n if (value.length <= limit) {\n return value\n }\n return value.slice(0, limit) + end\n },\n\n slug(value: string, separator = '-'): string {\n const normalized = value\n .normalize('NFKD')\n .replace(/\\p{Diacritic}/gu, '')\n .toLowerCase()\n\n const escaped = separator.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&')\n return normalized\n .replace(/[^a-z0-9]+/g, separator)\n .replace(new RegExp(`^${escaped}+|${escaped}+$`, 'g'), '')\n },\n\n uuid(): string {\n if (typeof globalThis.crypto?.randomUUID === 'function') {\n return globalThis.crypto.randomUUID()\n }\n return randomUUID()\n },\n\n /**\n * 生成 UUID v7(單調遞增,內含時間戳)。\n *\n * Bun 環境使用原生 Bun.randomUUIDv7() (C++ 實作)。\n * Node.js/Deno 環境使用 RFC 9562 的 JavaScript polyfill。\n *\n * UUID v7 的優勢:\n * - 資料庫主鍵天然有序 → B-tree 索引性能提升 2-10x\n * - 可從 UUID 提取毫秒級時間戳\n * - 仍保持全域唯一性\n *\n * @returns UUID v7 字串\n * @public\n */\n uuidv7(): string {\n const runtime = getRuntimeKind()\n if (runtime === 'bun' && typeof Bun !== 'undefined' && typeof Bun.randomUUIDv7 === 'function') {\n return Bun.randomUUIDv7()\n }\n return generateUUIDv7Fallback()\n },\n\n random(length = 16): string {\n if (length <= 0) {\n return ''\n }\n // base64url: [A-Za-z0-9_-]\n const bytes = randomBytes(Math.ceil((length * 3) / 4) + 2)\n const str = bytes.toString('base64').replace(/\\+/g, '-').replace(/\\//g, '_').replace(/=+$/g, '')\n return str.slice(0, length)\n },\n} as const\n\n/**\n * RFC 9562 UUID v7 的 JavaScript polyfill。\n *\n * UUID v7 格式 (128 bits):\n * - 時間戳:unix_ms (48 bits) → bytes[0-5]\n * - 亂數:12 bits → bytes[6-7] (upper 4 bits)\n * - 版本:0111 (v7) → bytes[6] 上 4 bits\n * - 亂數:62 bits → bytes[7-15]\n * - Variant:10xx → bytes[8] 上 2 bits (RFC 4122)\n *\n * @internal\n */\nfunction generateUUIDv7Fallback(): string {\n // 獲取當前毫秒級時間戳\n const timestamp = Date.now()\n\n // 建立 16 bytes 的 Uint8Array\n const bytes = new Uint8Array(16)\n\n // 填充前 6 bytes:時間戳 (big-endian)\n bytes[0] = (timestamp >>> 40) & 0xff\n bytes[1] = (timestamp >>> 32) & 0xff\n bytes[2] = (timestamp >>> 24) & 0xff\n bytes[3] = (timestamp >>> 16) & 0xff\n bytes[4] = (timestamp >>> 8) & 0xff\n bytes[5] = timestamp & 0xff\n\n // 填充後 10 bytes:隨機數\n const randomPart = randomBytes(10)\n for (let i = 0; i < 10; i += 1) {\n bytes[6 + i] = randomPart[i] ?? 0\n }\n\n // 設定 version bits:byte[6] 的上 4 bits = 0111 (v7)\n bytes[6] = ((bytes[6] ?? 0) & 0x0f) | 0x70\n\n // 設定 variant bits:byte[8] 的上 2 bits = 10 (RFC 4122)\n bytes[8] = ((bytes[8] ?? 0) & 0x3f) | 0x80\n\n // 格式化為標準 UUID 字串 (8-4-4-4-12)\n const hex = Array.from(bytes)\n .map((b) => b.toString(16).padStart(2, '0'))\n .join('')\n\n return [\n hex.slice(0, 8),\n hex.slice(8, 12),\n hex.slice(12, 16),\n hex.slice(16, 20),\n hex.slice(20, 32),\n ].join('-')\n}\n",
|
|
81
|
+
"import { HttpException } from './exceptions/HttpException'\nimport type { ContentfulStatusCode } from './http/types'\nimport type { PlanetCore } from './PlanetCore'\nimport type { Router } from './Router'\n\nexport { Arr } from './helpers/Arr'\nexport * from './helpers/data'\nexport * from './helpers/errors'\nexport * from './helpers/response'\nexport { Str } from './helpers/Str'\n\n/**\n * Error subclass used for dump and die functionality.\n * @internal\n */\nexport class DumpDieError extends Error {\n override name = 'DumpDieError'\n\n constructor(public readonly values: unknown[]) {\n super('Execution halted by dd()')\n }\n}\n\n/**\n * Options for dump output\n * @public\n */\nexport type DumpOptions = {\n depth?: number | null\n colors?: boolean\n}\n\nconst defaultDumpOptions: Required<DumpOptions> = {\n depth: null,\n colors: true,\n}\n\n/**\n * Dump data to console for debugging.\n *\n * Uses `console.dir` with configurable depth and colors to provide a\n * readable representation of any value.\n *\n * @param values - One or more values to dump to the console.\n *\n * @example\n * ```typescript\n * dump(user, { meta: 'data' });\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function dump(...values: unknown[]): void {\n for (const value of values) {\n console.dir(value, {\n depth: defaultDumpOptions.depth,\n colors: defaultDumpOptions.colors,\n })\n }\n}\n\n/**\n * Dump data to console and exit process (or throw in HTTP context).\n *\n * Short for \"Dump and Die\". In a CLI environment, it exits the process.\n * In an HTTP context (like a web request), it throws a `DumpDieError`\n * which is caught by the exception handler to display the debug output.\n *\n * @param values - One or more values to dump and then die.\n * @throws {DumpDieError} Always throws this error to halt execution.\n *\n * @example\n * ```typescript\n * dd(user.permissions);\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function dd(...values: unknown[]): never {\n dump(...values)\n throw new DumpDieError(values)\n}\n\n/**\n * Tap into a value, execute a callback, and return the value.\n *\n * This allows you to perform \"side effects\" on a value without breaking\n * the chain of operations.\n *\n * @param value - The value to tap into.\n * @param callback - A callback that receives the value.\n * @returns The original value.\n *\n * @example\n * ```typescript\n * const user = tap(new User(), (u) => {\n * u.name = 'Alice';\n * u.save();\n * });\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function tap<T>(value: T, callback: (value: T) => unknown): T {\n callback(value)\n return value\n}\n\n/**\n * Return the default value of the given value.\n *\n * If the value is a function, it will be executed with the provided arguments\n * and its result will be returned. Otherwise, the value itself is returned.\n * This is useful for handling optional lazy-loaded values.\n *\n * @param valueOrFactory - The value or a factory function.\n * @param args - Arguments to pass to the factory function if it is a function.\n * @returns The resolved value.\n *\n * @example\n * ```typescript\n * value(10); // 10\n * value(() => 10); // 10\n * value((name) => `Hello ${name}`, 'World'); // \"Hello World\"\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function value<TArgs extends readonly unknown[], TResult>(\n valueOrFactory: TResult | ((...args: TArgs) => TResult),\n ...args: TArgs\n): TResult {\n if (typeof valueOrFactory === 'function') {\n return (valueOrFactory as (...a: TArgs) => TResult)(...args)\n }\n return valueOrFactory\n}\n\nfunction isPlainObject(value: unknown): value is Record<string, unknown> {\n if (value === null || typeof value !== 'object') {\n return false\n }\n const proto = Object.getPrototypeOf(value)\n return proto === Object.prototype || proto === null\n}\n\n/**\n * Determine if the given value is \"blank\".\n *\n * A value is considered blank if it is:\n * - `null` or `undefined`\n * - An empty string or a string containing only whitespace\n * - An empty array\n * - An empty object\n * - An empty Map or Set\n *\n * @param value - The value to check.\n * @returns `true` if the value is blank, `false` otherwise.\n *\n * @example\n * ```typescript\n * blank(''); // true\n * blank(' '); // true\n * blank([]); // true\n * blank({}); // true\n * blank(0); // false\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function blank(value: unknown): boolean {\n if (value === null || value === undefined) {\n return true\n }\n\n if (typeof value === 'string') {\n return value.trim().length === 0\n }\n if (Array.isArray(value)) {\n return value.length === 0\n }\n if (value instanceof Map) {\n return value.size === 0\n }\n if (value instanceof Set) {\n return value.size === 0\n }\n if (isPlainObject(value)) {\n return Object.keys(value).length === 0\n }\n\n return false\n}\n\n/**\n * Determine if the given value is \"filled\" (not blank).\n *\n * This is the inverse of `blank()`.\n *\n * @param value - The value to check.\n * @returns `true` if the value is not blank, `false` otherwise.\n *\n * @example\n * ```typescript\n * filled('hello'); // true\n * filled([1, 2, 3]); // true\n * filled(''); // false\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function filled(value: unknown): boolean {\n return !blank(value)\n}\n\nfunction toError(error: Error | string | (() => Error)): Error {\n if (typeof error === 'string') {\n return new Error(error)\n }\n if (typeof error === 'function') {\n return error()\n }\n return error\n}\n\n/**\n * Throw an exception if the given condition is true.\n *\n * @param condition - The condition to evaluate.\n * @param error - The exception to throw, a factory function, or an error message string.\n * @throws {Error} If the condition evaluates to true.\n *\n * @example\n * ```typescript\n * throwIf(user.isBanned, 'User is banned from the system');\n * throwIf(count > 100, () => new ValidationError('Too many items'));\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function throwIf(\n condition: unknown,\n error: Error | string | (() => Error) = 'Error.'\n): void {\n if (condition) {\n throw toError(error)\n }\n}\n\n/**\n * Throw an exception unless the given condition is true.\n *\n * @param condition - The condition to evaluate.\n * @param error - The exception to throw, a factory function, or an error message string.\n * @throws {Error} If the condition evaluates to false.\n *\n * @example\n * ```typescript\n * throwUnless(user.isAdmin, 'Unauthorized access');\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function throwUnless(\n condition: unknown,\n error: Error | string | (() => Error) = 'Error.'\n): void {\n if (!condition) {\n throw toError(error)\n }\n}\n\ntype EnvShape = {\n Bun?: {\n env?: Record<string, string | undefined>\n }\n}\n\n/**\n * Get the value of an environment variable.\n *\n * Automatically detects the runtime environment (Bun or Node.js) to retrieve\n * the variable.\n *\n * @param key - The environment variable name.\n * @param defaultValue - An optional default value to return if the variable is not defined.\n * @returns The environment variable value or the default value.\n *\n * @example\n * ```typescript\n * const debug = env('DEBUG', 'false');\n * const apiKey = env('API_KEY');\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function env<TDefault = string | undefined>(key: string, defaultValue?: TDefault) {\n const bunEnv = (globalThis as EnvShape).Bun?.env\n const value = bunEnv?.[key] ?? process.env[key]\n return (value ?? defaultValue) as string | TDefault\n}\n\nlet currentApp: PlanetCore | undefined\n\n/**\n * Set the global application instance.\n *\n * This is used internally during the bootstrap process to provide global\n * access to the application instance via the `app()` helper.\n *\n * @param core - The PlanetCore instance to set as global.\n * @internal\n */\nexport function setApp(core: PlanetCore | null): void {\n currentApp = core ?? undefined\n}\n\n/**\n * Check if the global application instance has been initialized and set.\n *\n * @returns `true` if the application instance is set, `false` otherwise.\n *\n * @public\n * @since 3.0.0\n */\nexport function hasApp(): boolean {\n return currentApp !== undefined\n}\n\n/**\n * Get the global application instance.\n *\n * Provides access to the core application container, configuration, and services.\n *\n * @returns The initialized PlanetCore instance.\n * @throws {Error} If the application has not been initialized.\n *\n * @example\n * ```typescript\n * const core = app();\n * console.log(core.version);\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function app(): PlanetCore {\n if (!currentApp) {\n throw new Error('No app is bound. Call setApp(core) once during bootstrap.')\n }\n return currentApp\n}\n\n/**\n * Get a configuration value from the application.\n *\n * Supports dot notation for accessing nested configuration properties.\n *\n * @param key - The configuration key in dot notation (e.g., 'app.name').\n * @param defaultValue - An optional default value to return if the key is not found.\n * @returns The configuration value or the default value.\n *\n * @example\n * ```typescript\n * const appName = config('app.name');\n * const port = config('app.port', 3000);\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function config<T = unknown>(key: string, defaultValue?: T): T {\n if (defaultValue === undefined) {\n return app().config.get<T>(key)\n }\n return app().config.get<T>(key, defaultValue)\n}\n\n/**\n * Get the global logger instance.\n *\n * Shortcut for `app().logger`.\n *\n * @returns The application's logger instance.\n *\n * @example\n * ```typescript\n * logger().info('Operation completed successfully');\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function logger() {\n return app().logger\n}\n\n/**\n * Get the application's primary router instance.\n *\n * Shortcut for `app().router`.\n *\n * @returns The router instance.\n *\n * @public\n * @since 3.0.0\n */\nexport function router(): Router {\n return app().router\n}\n\n/**\n * Abort the current request with an HTTP exception.\n *\n * Throws an `HttpException` with the specified status code and optional message.\n *\n * @param status - The HTTP status code to return.\n * @param message - An optional custom error message.\n * @throws {HttpException} Always throws this exception.\n *\n * @example\n * ```typescript\n * abort(403, 'You do not have permission to access this resource');\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function abort(status: ContentfulStatusCode, message?: string): never {\n if (message === undefined) {\n throw new HttpException(status)\n }\n throw new HttpException(status, { message })\n}\n\n/**\n * Abort the request if the given condition is true.\n *\n * @param condition - The condition to evaluate.\n * @param status - The HTTP status code to return.\n * @param message - An optional custom error message.\n * @throws {HttpException} If the condition is true.\n *\n * @example\n * ```typescript\n * abortIf(!user.isActive, 403, 'Account is deactivated');\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function abortIf(condition: unknown, status: ContentfulStatusCode, message?: string): void {\n if (condition) {\n abort(status, message)\n }\n}\n\n/**\n * Abort the request unless the given condition is true.\n *\n * @param condition - The condition to evaluate.\n * @param status - The HTTP status code to return.\n * @param message - An optional custom error message.\n * @throws {HttpException} If the condition is false.\n *\n * @example\n * ```typescript\n * abortUnless(request.hasValidToken(), 401, 'Invalid authentication token');\n * ```\n *\n * @public\n * @since 3.0.0\n */\nexport function abortUnless(\n condition: unknown,\n status: ContentfulStatusCode,\n message?: string\n): void {\n if (!condition) {\n abort(status, message)\n }\n}\n",
|
|
82
|
+
"import type { Encrypter } from '../security/Encrypter'\nimport type { GravitoContext } from './types'\n\n/**\n * Options for setting cookies\n * @public\n */\nexport interface CookieOptions {\n path?: string\n domain?: string\n secure?: boolean\n httpOnly?: boolean\n sameSite?: 'Strict' | 'Lax' | 'None'\n maxAge?: number\n expires?: Date\n encrypt?: boolean\n}\n\n/**\n * Utility for managing cookies (request/response/encryption).\n * @public\n */\nexport class CookieJar {\n private queued: Map<string, { value: string; options: CookieOptions }> = new Map()\n\n constructor(private encrypter?: Encrypter) {}\n\n /**\n * Parse cookies from a Cookie header string using Bun's native CookieMap\n * @param header - The Cookie header value\n * @returns Parsed cookies as key-value pairs\n */\n static parseCookies(header: string): Record<string, string> {\n if (!header) {\n return {}\n }\n const cookieMap = new Bun.CookieMap(header)\n const out: Record<string, string> = {}\n for (const [key, value] of cookieMap.entries()) {\n out[key] = value\n }\n return out\n }\n\n /**\n * Queue a cookie to be sent with the response\n */\n queue(name: string, value: string, minutes = 60, options: CookieOptions = {}) {\n const resolved: CookieOptions = {\n path: options.path ?? '/',\n httpOnly: options.httpOnly ?? true,\n sameSite: options.sameSite ?? 'Lax',\n secure: options.secure ?? process.env.NODE_ENV === 'production',\n ...options,\n }\n // Convert minutes to maxAge (seconds)\n if (minutes && !resolved.maxAge) {\n resolved.maxAge = minutes * 60\n }\n\n let finalValue = value\n\n if (resolved.encrypt) {\n if (!this.encrypter) {\n throw new Error('Encryption is not available. Ensure APP_KEY is set.')\n }\n finalValue = this.encrypter.encrypt(value)\n }\n\n this.queued.set(name, { value: finalValue, options: resolved })\n }\n\n /**\n * Make a cookie that lasts \"forever\" (5 years)\n */\n forever(name: string, value: string, options: CookieOptions = {}) {\n this.queue(name, value, 2628000, options)\n }\n\n /**\n * Expire a cookie\n */\n forget(name: string, options: CookieOptions = {}) {\n this.queue(name, '', 0, { ...options, maxAge: 0, expires: new Date(0) })\n }\n\n /**\n * Serialize a cookie to a Set-Cookie header value using Bun's native Cookie API\n */\n private serializeCookie(name: string, value: string, opts: CookieOptions): string {\n return new Bun.Cookie(name, value, {\n path: opts.path,\n domain: opts.domain,\n expires: opts.expires,\n maxAge: opts.maxAge,\n secure: opts.secure,\n httpOnly: opts.httpOnly,\n sameSite: opts.sameSite?.toLowerCase() as 'strict' | 'lax' | 'none' | undefined,\n }).toString()\n }\n\n /**\n * Attach queued cookies to the context\n */\n attach(c: GravitoContext) {\n for (const [name, { value, options }] of this.queued) {\n c.header('Set-Cookie', this.serializeCookie(name, value, options), { append: true })\n }\n }\n}\n",
|
|
83
|
+
"import { CookieJar, type CookieOptions } from './CookieJar'\nimport type { GravitoContext } from './types'\n\n/**\n * Get a cookie value from the request\n * @param c - Context object\n * @param name - Cookie name\n * @returns Cookie value or undefined\n * @public\n */\nexport function getCookie(c: GravitoContext, name: string): string | undefined {\n const cookieHeader = c.req.header('Cookie') || ''\n const cookies = CookieJar.parseCookies(cookieHeader)\n return cookies[name]\n}\n\n/**\n * Set a cookie in the response\n * @param c - Context object\n * @param name - Cookie name\n * @param value - Cookie value\n * @param options - Cookie options\n * @public\n */\nexport function setCookie(\n c: GravitoContext,\n name: string,\n value: string,\n options: CookieOptions & { maxAge?: number } = {}\n): void {\n // Try to use CookieJar if available (when using PlanetCore)\n const cookieJar = c.get('cookieJar') as CookieJar | undefined\n if (cookieJar) {\n // Convert maxAge (seconds) to minutes for CookieJar\n const minutes = options.maxAge ? Math.floor(options.maxAge / 60) : 60\n cookieJar.queue(name, value, minutes, options)\n return\n }\n\n // Fallback: 直接設置 Set-Cookie header(使用 Bun 原生 Cookie API)\n const cookie = new Bun.Cookie(name, value, {\n path: options.path,\n domain: options.domain,\n expires: options.expires,\n maxAge: options.maxAge,\n secure: options.secure,\n httpOnly: options.httpOnly,\n sameSite: options.sameSite?.toLowerCase() as 'strict' | 'lax' | 'none' | undefined,\n })\n c.header('Set-Cookie', cookie.toString(), { append: true })\n}\n\n/**\n * Delete a cookie (expire it)\n * @param c - Context object\n * @param name - Cookie name\n * @param options - Cookie options (path, domain, etc.)\n * @public\n */\nexport function deleteCookie(c: GravitoContext, name: string, options: CookieOptions = {}): void {\n // Try to use CookieJar if available\n const cookieJar = c.get('cookieJar') as CookieJar | undefined\n if (cookieJar) {\n cookieJar.forget(name, options)\n return\n }\n\n // Fallback: Set cookie with expired date\n setCookie(c, name, '', {\n ...options,\n maxAge: 0,\n expires: new Date(0),\n })\n}\n",
|
|
84
|
+
"/**\n * Browser-safe runtime abstraction module.\n */\n\nimport { createUnknownAdapter } from './adapter-unknown'\nimport type { RuntimeAdapter, RuntimePasswordAdapter, RuntimeSqliteDatabase } from './types'\n\nexport { getRuntimeEnv, getRuntimeKind } from './detection'\nexport * from './types'\n\nlet runtimeAdapter: RuntimeAdapter | null = null\n\nexport function getRuntimeAdapter(): RuntimeAdapter {\n if (!runtimeAdapter) {\n runtimeAdapter = createUnknownAdapter()\n }\n return runtimeAdapter\n}\n\nlet passwordAdapter: RuntimePasswordAdapter | null = null\n\nexport function getPasswordAdapter(): RuntimePasswordAdapter {\n if (!passwordAdapter) {\n const message = '[RuntimeAdapter] Password hashing is not supported in the browser'\n passwordAdapter = {\n async hash() {\n throw new Error(message)\n },\n async verify() {\n throw new Error(message)\n },\n }\n }\n return passwordAdapter\n}\n\nexport async function createSqliteDatabase(_path: string): Promise<RuntimeSqliteDatabase> {\n throw new Error('[RuntimeAdapter] SQLite storage is not supported in the browser')\n}\n\nexport async function archiveFromDirectory() {\n throw new Error('Not supported in browser')\n}\nexport function getArchiveAdapter() {\n throw new Error('Not supported in browser')\n}\nexport function getCompressionAdapter() {\n throw new Error('Not supported in browser')\n}\nexport function createHtmlRenderCallbacks() {\n throw new Error('Not supported in browser')\n}\nexport function getMarkdownAdapter() {\n throw new Error('Not supported in browser')\n}\n\n// Deep Equals - Browser safe implementation\nexport function getDeepEquals() {\n return (a: any, b: any) => JSON.stringify(a) === JSON.stringify(b)\n}\n\n// Escape - Browser safe\nexport function getEscapeHtml() {\n return (str: string) =>\n str.replace(\n /[&<>\"']/g,\n (m) =>\n ({\n '&': '&',\n '<': '<',\n '>': '>',\n '\"': '"',\n \"'\": ''',\n })[m] || m\n )\n}\n\n/**\n * Convert various data types to Uint8Array.\n */\nexport async function toUint8Array(\n data: Blob | string | ArrayBuffer | Uint8Array\n): Promise<Uint8Array> {\n if (data instanceof Uint8Array) {\n return data\n }\n if (typeof data === 'string') {\n return new TextEncoder().encode(data)\n }\n if (data instanceof ArrayBuffer) {\n return new Uint8Array(data)\n }\n if (data instanceof Blob) {\n return new Uint8Array(await data.arrayBuffer())\n }\n return new Uint8Array()\n}\n",
|
|
85
|
+
"/// <reference types=\"bun-types\" />\n/**\n * @gravito/core\n *\n * Browser-safe entry point for the core micro-kernel.\n * Excludes Node.js specific modules like runtime adapters and FFI.\n *\n * @packageDocumentation\n */\n\n// Export version from package.json\nimport packageJson from '../package.json'\n\n/**\n * Current version of @gravito/core.\n * @public\n */\nexport const VERSION = packageJson.version\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Gravito HTTP Abstractions (Types are safe)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport { GravitoEngineAdapter } from './adapters/GravitoEngineAdapter'\nexport type { AdapterConfig, AdapterFactory, HttpAdapter, RouteDefinition } from './adapters/types'\nexport { isHttpAdapter } from './adapters/types'\nexport type {\n ContentfulStatusCode,\n GravitoContext,\n GravitoErrorHandler,\n GravitoHandler,\n GravitoMiddleware,\n GravitoNext,\n GravitoNotFoundHandler,\n GravitoRequest,\n GravitoVariables,\n HttpMethod,\n ProxyOptions,\n StatusCode,\n ValidationTarget,\n} from './http/types'\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Core Exports (Filtered for browser safety)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport { ConfigManager } from './ConfigManager'\nexport { Container, type Factory, type ServiceKey, type ServiceMap } from './Container'\nexport { registerQueueCommands } from './cli/queue-commands'\nexport {\n codeFromStatus,\n ErrorHandler,\n type ErrorHandlerDeps,\n messageFromStatus,\n} from './ErrorHandler'\nexport { EventManager } from './EventManager'\n\n// Event System (Types are safe)\nexport * from './events' // Most logic in events is browser-safe or has fallbacks\n\n// Hooks\nexport { HookManager, type HookManagerConfig } from './HookManager'\n\n// Health\nexport { HealthProvider } from './health/HealthProvider'\n\n// Helpers\nexport {\n Arr,\n abort,\n abortIf,\n abortUnless,\n app,\n blank,\n config,\n DumpDieError,\n dd,\n dump,\n env,\n filled,\n hasApp,\n logger,\n router,\n Str,\n setApp,\n tap,\n throwIf,\n throwUnless,\n value,\n} from './helpers'\nexport * from './helpers/data'\nexport * from './helpers/errors'\nexport * from './helpers/response'\n\n// HTTP utilities\nexport { CookieJar, type CookieOptions } from './http/CookieJar'\nexport { deleteCookie, getCookie, setCookie } from './http/cookie'\n\n// Service Provider\nexport { ServiceProvider } from './ServiceProvider'\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Browser-Safe Runtime\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport * from './runtime/index.browser'\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Configuration Helper\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport function defineConfig(config: any): any {\n return config\n}\n"
|
|
86
|
+
],
|
|
87
|
+
"mappings": "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",
|
|
88
|
+
"debugId": "004674489562A73F64756E2164756E21",
|
|
89
|
+
"names": []
|
|
90
|
+
}
|