@geekmidas/cli 0.7.0 → 0.8.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.
@@ -1 +0,0 @@
1
- {"version":3,"file":"openapi-Mwy2_R4W.mjs","names":["context: BuildContext","outputDir: string","generator: ConstructGenerator<T, R>","patterns?: Routes","options?: GeneratorOptions","constructs: GeneratedConstruct<T>[]","value: any","context: BuildContext","constructs: GeneratedConstruct<\n Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >\n >[]","outputDir: string","options?: GeneratorOptions","routes: RouteInfo[]","routeInfo: RouteInfo","sourceFile: string","exportName: string","provider: LegacyProvider","_endpoint: Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >","content: string","endpoints: GeneratedConstruct<\n Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >\n >[]","_context: BuildContext","importPath: string","envParserPath: string","envParserImportPattern: string","endpoints: Endpoint<any, any, any, any, any, any>[]","options: OpenApiTsOptions","endpointInfos: EndpointInfo[]","type: string","_name: string","authMap: Record<string, string | null>","interfaces: string[]","schema: StandardSchemaV1","defaultName: string","info: EndpointInfo","str: string","name: string","collectedDefs: Map<string, JsonSchema>","schema: JsonSchema","props: string[]","s: JsonSchema","e: string","pathEntries: string[]","methodEntries: string[]","parts: string[]","params: {\n title: string;\n version: string;\n description?: string;\n securitySchemes: SecuritySchemeInfo[];\n endpointAuth: Record<string, string | null>;\n schemaInterfaces: string;\n pathsInterface: string;\n }","config: GkmConfig","options: { silent?: boolean }","options: OpenAPIOptions"],"sources":["../src/generators/Generator.ts","../src/generators/EndpointGenerator.ts","../src/generators/OpenApiTsGenerator.ts","../src/openapi.ts"],"sourcesContent":["import { relative } from 'path';\nimport type { Construct } from '@geekmidas/constructs';\nimport fg from 'fast-glob';\nimport kebabCase from 'lodash.kebabcase';\nimport type { BuildContext } from '../build/types';\nimport type { LegacyProvider, Routes } from '../types';\n\nexport interface GeneratorOptions {\n provider?: LegacyProvider;\n [key: string]: any;\n}\n\nexport abstract class ConstructGenerator<T extends Construct, R = void> {\n abstract isConstruct(value: any): value is T;\n\n static async build<T extends Construct, R = void>(\n context: BuildContext,\n outputDir: string,\n generator: ConstructGenerator<T, R>,\n patterns?: Routes,\n options?: GeneratorOptions,\n ): Promise<R> {\n const constructs = await generator.load(patterns);\n return generator.build(context, constructs, outputDir, options);\n }\n\n abstract build(\n context: BuildContext,\n constructs: GeneratedConstruct<T>[],\n outputDir: string,\n options?: GeneratorOptions,\n ): Promise<R>;\n\n async load(\n patterns?: Routes,\n cwd = process.cwd(),\n ): Promise<GeneratedConstruct<T>[]> {\n const logger = console;\n\n // Normalize patterns to array\n const globPatterns = Array.isArray(patterns)\n ? patterns\n : patterns\n ? [patterns]\n : [];\n\n // Find all files\n const files = fg.stream(globPatterns, {\n cwd,\n absolute: true,\n });\n\n // Load constructs\n const constructs: GeneratedConstruct<T>[] = [];\n\n for await (const f of files) {\n try {\n const file = f.toString();\n const module = await import(file);\n\n // Check all exports for constructs\n for (const [key, construct] of Object.entries(module)) {\n if (this.isConstruct(construct)) {\n constructs.push({\n key,\n name: kebabCase(key),\n construct,\n path: {\n absolute: file,\n relative: relative(process.cwd(), file),\n },\n });\n }\n }\n } catch (error) {\n logger.warn(`Failed to load ${f}:`, (error as Error).message);\n throw new Error(\n 'Failed to load constructs. Please check the logs for details.',\n );\n }\n }\n\n return constructs;\n }\n}\n\nexport interface GeneratedConstruct<T extends Construct> {\n key: string;\n name: string;\n construct: T;\n path: {\n absolute: string;\n relative: string;\n };\n}\n","import { mkdir, writeFile } from 'node:fs/promises';\nimport { dirname, join, relative } from 'node:path';\nimport { Endpoint } from '@geekmidas/constructs/endpoints';\nimport type { BuildContext } from '../build/types';\nimport type { LegacyProvider, RouteInfo } from '../types';\nimport {\n ConstructGenerator,\n type GeneratedConstruct,\n type GeneratorOptions,\n} from './Generator';\n\nexport class EndpointGenerator extends ConstructGenerator<\n Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >,\n RouteInfo[]\n> {\n isConstruct(\n value: any,\n ): value is Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n > {\n return Endpoint.isEndpoint(value);\n }\n\n async build(\n context: BuildContext,\n constructs: GeneratedConstruct<\n Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >\n >[],\n outputDir: string,\n options?: GeneratorOptions,\n ): Promise<RouteInfo[]> {\n const provider = options?.provider || 'aws-apigatewayv2';\n const enableOpenApi = options?.enableOpenApi || false;\n const logger = console;\n const routes: RouteInfo[] = [];\n\n if (constructs.length === 0) {\n return routes;\n }\n\n if (provider === 'server') {\n // Generate endpoints.ts and app.ts\n await this.generateEndpointsFile(outputDir, constructs, context);\n const appFile = await this.generateAppFile(outputDir, context);\n\n routes.push({\n path: '*',\n method: 'ALL',\n handler: relative(process.cwd(), appFile),\n authorizer: 'none',\n });\n\n logger.log(\n `Generated server with ${constructs.length} endpoints${enableOpenApi ? ' (OpenAPI enabled)' : ''}`,\n );\n } else if (provider === 'aws-lambda') {\n // For aws-lambda, create routes subdirectory\n const routesDir = join(outputDir, 'routes');\n await mkdir(routesDir, { recursive: true });\n\n // Generate individual handlers for API Gateway routes\n for (const { key, construct, path } of constructs) {\n const handlerFile = await this.generateHandlerFile(\n routesDir,\n path.relative,\n key,\n 'aws-apigatewayv2',\n construct,\n context,\n );\n\n const routeInfo: RouteInfo = {\n path: construct._path,\n method: construct.method,\n handler: relative(process.cwd(), handlerFile).replace(\n /\\.ts$/,\n '.handler',\n ),\n timeout: construct.timeout,\n memorySize: construct.memorySize,\n environment: await construct.getEnvironment(),\n authorizer: construct.authorizer?.name ?? 'none',\n };\n\n routes.push(routeInfo);\n logger.log(\n `Generated handler for ${routeInfo.method} ${routeInfo.path}`,\n );\n }\n } else {\n // Generate individual handler files for AWS API Gateway providers\n for (const { key, construct, path } of constructs) {\n const handlerFile = await this.generateHandlerFile(\n outputDir,\n path.relative,\n key,\n provider,\n construct,\n context,\n );\n\n const routeInfo: RouteInfo = {\n path: construct._path,\n method: construct.method,\n handler: relative(process.cwd(), handlerFile).replace(\n /\\.ts$/,\n '.handler',\n ),\n timeout: construct.timeout,\n memorySize: construct.memorySize,\n environment: await construct.getEnvironment(),\n authorizer: construct.authorizer?.name ?? 'none',\n };\n\n routes.push(routeInfo);\n logger.log(\n `Generated handler for ${routeInfo.method} ${routeInfo.path}`,\n );\n }\n }\n\n return routes;\n }\n\n private async generateHandlerFile(\n outputDir: string,\n sourceFile: string,\n exportName: string,\n provider: LegacyProvider,\n _endpoint: Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >,\n context: BuildContext,\n ): Promise<string> {\n const handlerFileName = `${exportName}.ts`;\n const handlerPath = join(outputDir, handlerFileName);\n\n const relativePath = relative(dirname(handlerPath), sourceFile);\n const importPath = relativePath.replace(/\\.ts$/, '.js');\n\n const relativeEnvParserPath = relative(\n dirname(handlerPath),\n context.envParserPath,\n );\n\n let content: string;\n\n switch (provider) {\n case 'aws-apigatewayv1':\n content = this.generateAWSApiGatewayV1Handler(\n importPath,\n exportName,\n relativeEnvParserPath,\n context.envParserImportPattern,\n );\n break;\n case 'aws-apigatewayv2':\n content = this.generateAWSApiGatewayV2Handler(\n importPath,\n exportName,\n relativeEnvParserPath,\n context.envParserImportPattern,\n );\n break;\n case 'server':\n content = this.generateServerHandler(importPath, exportName);\n break;\n default:\n throw new Error(`Unsupported provider: ${provider}`);\n }\n\n await writeFile(handlerPath, content);\n return handlerPath;\n }\n\n private async generateEndpointsFile(\n outputDir: string,\n endpoints: GeneratedConstruct<\n Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >\n >[],\n _context: BuildContext,\n ): Promise<string> {\n const endpointsFileName = 'endpoints.ts';\n const endpointsPath = join(outputDir, endpointsFileName);\n\n // Group imports by file\n const importsByFile = new Map<string, string[]>();\n\n for (const { path, key } of endpoints) {\n const relativePath = relative(dirname(endpointsPath), path.relative);\n const importPath = relativePath.replace(/\\.ts$/, '.js');\n\n if (!importsByFile.has(importPath)) {\n importsByFile.set(importPath, []);\n }\n importsByFile.get(importPath)!.push(key);\n }\n\n // Generate import statements\n const imports = Array.from(importsByFile.entries())\n .map(\n ([importPath, exports]) =>\n `import { ${exports.join(', ')} } from '${importPath}';`,\n )\n .join('\\n');\n\n const allExportNames = endpoints.map(({ key }) => key);\n\n const content = `import type { EnvironmentParser } from '@geekmidas/envkit';\nimport type { Logger } from '@geekmidas/logger';\nimport { HonoEndpoint } from '@geekmidas/constructs/hono';\nimport { Endpoint } from '@geekmidas/constructs/endpoints';\nimport { ServiceDiscovery } from '@geekmidas/services';\nimport type { Hono } from 'hono';\n${imports}\n\nconst endpoints: Endpoint<any, any, any, any, any, any, any, any, any, any, any, any, any, any>[] = [\n ${allExportNames.join(',\\n ')}\n];\n\nexport async function setupEndpoints(\n app: Hono,\n envParser: EnvironmentParser<any>,\n logger: Logger,\n enableOpenApi: boolean = true,\n): Promise<void> {\n const serviceDiscovery = ServiceDiscovery.getInstance(\n logger,\n envParser\n );\n\n // Configure OpenAPI options based on enableOpenApi flag\n const openApiOptions: any = enableOpenApi ? {\n docsPath: '/__docs',\n openApiOptions: {\n title: 'API Documentation',\n version: '1.0.0',\n description: 'Generated API documentation'\n }\n } : { docsPath: false };\n\n HonoEndpoint.addRoutes(endpoints, serviceDiscovery, app, openApiOptions);\n\n // Add Swagger UI if OpenAPI is enabled\n if (enableOpenApi) {\n try {\n const { swaggerUI } = await import('@hono/swagger-ui');\n app.get('/__docs/ui', swaggerUI({ url: '/__docs' }));\n } catch {\n // @hono/swagger-ui not installed, skip Swagger UI\n }\n }\n}\n`;\n\n await writeFile(endpointsPath, content);\n\n return endpointsPath;\n }\n\n private async generateAppFile(\n outputDir: string,\n context: BuildContext,\n ): Promise<string> {\n const appFileName = 'app.ts';\n const appPath = join(outputDir, appFileName);\n\n const relativeLoggerPath = relative(dirname(appPath), context.loggerPath);\n\n const relativeEnvParserPath = relative(\n dirname(appPath),\n context.envParserPath,\n );\n\n // Generate telescope imports and setup if enabled\n const telescopeEnabled = context.telescope?.enabled;\n const telescopeWebSocketEnabled = context.telescope?.websocket;\n const usesExternalTelescope = !!context.telescope?.telescopePath;\n\n // Generate studio imports and setup if enabled\n const studioEnabled = context.studio?.enabled;\n const usesExternalStudio = !!context.studio?.studioPath;\n\n // Generate imports based on whether telescope is external or inline\n let telescopeImports = '';\n if (telescopeEnabled) {\n if (usesExternalTelescope) {\n const relativeTelescopePath = relative(\n dirname(appPath),\n context.telescope!.telescopePath!,\n );\n telescopeImports = `import ${context.telescope!.telescopeImportPattern} from '${relativeTelescopePath}';\nimport { createMiddleware, createUI } from '@geekmidas/telescope/hono';`;\n } else {\n telescopeImports = `import { Telescope, InMemoryStorage } from '@geekmidas/telescope';\nimport { createMiddleware, createUI } from '@geekmidas/telescope/hono';`;\n }\n }\n\n // Generate imports for studio\n let studioImports = '';\n if (studioEnabled) {\n if (usesExternalStudio) {\n const relativeStudioPath = relative(\n dirname(appPath),\n context.studio!.studioPath!,\n );\n studioImports = `import ${context.studio!.studioImportPattern} from '${relativeStudioPath}';\nimport { createStudioApp } from '@geekmidas/studio/server/hono';`;\n } else {\n studioImports = `// Studio requires a configured instance - use studio config path\n// import { createStudioApp } from '@geekmidas/studio/server/hono';`;\n }\n }\n\n const telescopeWebSocketSetupCode = telescopeWebSocketEnabled\n ? `\n // Setup WebSocket for real-time telescope updates\n try {\n const { createNodeWebSocket } = await import('@hono/node-ws');\n const { injectWebSocket, upgradeWebSocket } = createNodeWebSocket({ app: honoApp });\n // Add WebSocket route directly to main app (sub-app routes don't support WS upgrade)\n honoApp.get('${context.telescope!.path}/ws', upgradeWebSocket(() => ({\n onOpen: (_event: Event, ws: any) => {\n telescope.addWsClient(ws);\n },\n onClose: (_event: Event, ws: any) => {\n telescope.removeWsClient(ws);\n },\n onMessage: (event: MessageEvent, ws: any) => {\n try {\n const data = JSON.parse(event.data);\n if (data.type === 'ping') {\n ws.send(JSON.stringify({ type: 'pong' }));\n }\n } catch {\n // Ignore invalid messages\n }\n },\n })));\n // Store injectWebSocket for server entry to call after serve()\n (honoApp as any).__injectWebSocket = injectWebSocket;\n logger.info('Telescope WebSocket enabled');\n } catch (e) {\n logger.warn({ error: e }, 'WebSocket support not available - install @hono/node-ws for real-time updates');\n }\n`\n : '';\n\n // Generate telescope setup - either use external instance or create inline\n let telescopeSetup = '';\n if (telescopeEnabled) {\n if (usesExternalTelescope) {\n // Use external telescope instance - no need to create one\n telescopeSetup = `\n${telescopeWebSocketSetupCode}\n // Add telescope middleware (before endpoints to capture all requests)\n honoApp.use('*', createMiddleware(telescope));\n\n // Mount telescope UI\n const telescopeUI = createUI(telescope);\n honoApp.route('${context.telescope!.path}', telescopeUI);\n`;\n } else {\n // Create inline telescope instance\n telescopeSetup = `\n // Setup Telescope for debugging/monitoring\n const telescopeStorage = new InMemoryStorage({ maxEntries: ${context.telescope!.maxEntries} });\n const telescope = new Telescope({\n enabled: true,\n path: '${context.telescope!.path}',\n ignorePatterns: ${JSON.stringify(context.telescope!.ignore)},\n recordBody: ${context.telescope!.recordBody},\n storage: telescopeStorage,\n });\n${telescopeWebSocketSetupCode}\n // Add telescope middleware (before endpoints to capture all requests)\n honoApp.use('*', createMiddleware(telescope));\n\n // Mount telescope UI\n const telescopeUI = createUI(telescope);\n honoApp.route('${context.telescope!.path}', telescopeUI);\n`;\n }\n }\n\n // Generate studio setup - requires external instance\n let studioSetup = '';\n if (studioEnabled && usesExternalStudio) {\n studioSetup = `\n // Mount Studio data browser UI\n const studioApp = createStudioApp(studio);\n honoApp.route('${context.studio!.path}', studioApp);\n`;\n }\n\n const content = `/**\n * Generated server application\n *\n * ⚠️ WARNING: This is for LOCAL DEVELOPMENT ONLY\n * The subscriber polling mechanism is not production-ready.\n * For production, use AWS Lambda with SQS/SNS event sources.\n */\nimport { Hono } from 'hono';\nimport type { Hono as HonoType } from 'hono';\nimport { setupEndpoints } from './endpoints.js';\nimport { setupSubscribers } from './subscribers.js';\nimport ${context.envParserImportPattern} from '${relativeEnvParserPath}';\nimport ${context.loggerImportPattern} from '${relativeLoggerPath}';\n${telescopeImports}\n${studioImports}\n\nexport interface ServerApp {\n app: HonoType;\n start: (options?: {\n port?: number;\n serve: (app: HonoType, port: number) => void | Promise<void>;\n }) => Promise<void>;\n}\n\n/**\n * Create and configure the Hono application\n *\n * @param app - Optional Hono app instance to configure (creates new one if not provided)\n * @param enableOpenApi - Enable OpenAPI documentation (default: true)\n * @returns Server app with configured Hono app and start function\n *\n * @example\n * // With Bun\n * import { createApp } from './.gkm/server/app.js';\n *\n * const { app, start } = await createApp();\n *\n * await start({\n * port: 3000,\n * serve: (app, port) => {\n * Bun.serve({ port, fetch: app.fetch });\n * }\n * });\n *\n * @example\n * // With Node.js (using @hono/node-server)\n * import { serve } from '@hono/node-server';\n * import { createApp } from './.gkm/server/app.js';\n *\n * const { app, start } = await createApp();\n *\n * await start({\n * port: 3000,\n * serve: (app, port) => {\n * serve({ fetch: app.fetch, port });\n * }\n * });\n */\nexport async function createApp(app?: HonoType, enableOpenApi: boolean = true): Promise<ServerApp> {\n const honoApp = app || new Hono();\n${telescopeSetup}${studioSetup}\n // Setup HTTP endpoints\n await setupEndpoints(honoApp, envParser, logger, enableOpenApi);\n\n return {\n app: honoApp,\n async start(options) {\n if (!options?.serve) {\n throw new Error(\n 'serve function is required. Pass a serve function for your runtime:\\\\n' +\n ' - Bun: (app, port) => Bun.serve({ port, fetch: app.fetch })\\\\n' +\n ' - Node: (app, port) => serve({ fetch: app.fetch, port })'\n );\n }\n\n const port = options.port ?? 3000;\n\n // Start subscribers in background (non-blocking, local development only)\n await setupSubscribers(envParser, logger).catch((error) => {\n logger.error({ error }, 'Failed to start subscribers');\n });\n\n logger.info({ port }, 'Starting server');\n\n // Start HTTP server using provided serve function\n await options.serve(honoApp, port);\n\n logger.info({ port }, 'Server started');\n }\n };\n}\n\n// Default export for convenience\nexport default createApp;\n`;\n\n await writeFile(appPath, content);\n\n return appPath;\n }\n\n private generateAWSApiGatewayV1Handler(\n importPath: string,\n exportName: string,\n envParserPath: string,\n envParserImportPattern: string,\n ): string {\n return `import { AmazonApiGatewayV1Endpoint } from '@geekmidas/constructs/aws';\nimport { ${exportName} } from '${importPath}';\nimport ${envParserImportPattern} from '${envParserPath}';\n\nconst adapter = new AmazonApiGatewayV1Endpoint(envParser, ${exportName});\n\nexport const handler = adapter.handler;\n`;\n }\n\n private generateAWSApiGatewayV2Handler(\n importPath: string,\n exportName: string,\n envParserPath: string,\n envParserImportPattern: string,\n ): string {\n return `import { AmazonApiGatewayV2Endpoint } from '@geekmidas/constructs/aws';\nimport { ${exportName} } from '${importPath}';\nimport ${envParserImportPattern} from '${envParserPath}';\n\nconst adapter = new AmazonApiGatewayV2Endpoint(envParser, ${exportName});\n\nexport const handler = adapter.handler;\n`;\n }\n\n private generateServerHandler(\n importPath: string,\n exportName: string,\n ): string {\n return `import { ${exportName} } from '${importPath}';\n\n// Server handler - implement based on your server framework\nexport const handler = ${exportName};\n`;\n }\n}\n","import type { Endpoint } from '@geekmidas/constructs/endpoints';\nimport {\n StandardSchemaJsonSchema,\n getSchemaMetadata,\n} from '@geekmidas/schema/conversion';\nimport type { StandardSchemaV1 } from '@standard-schema/spec';\n\ninterface OpenApiTsOptions {\n title?: string;\n version?: string;\n description?: string;\n}\n\n// JSON Schema type definition\ninterface JsonSchema {\n type?: string;\n properties?: Record<string, JsonSchema>;\n items?: JsonSchema;\n required?: string[];\n enum?: string[];\n $ref?: string;\n anyOf?: JsonSchema[];\n oneOf?: JsonSchema[];\n allOf?: JsonSchema[];\n additionalProperties?: boolean | JsonSchema;\n [key: string]: unknown;\n}\n\n// Security scheme type (OpenAPI 3.1)\ninterface SecuritySchemeObject {\n type: 'apiKey' | 'http' | 'mutualTLS' | 'oauth2' | 'openIdConnect';\n description?: string;\n name?: string;\n in?: 'query' | 'header' | 'cookie';\n scheme?: string;\n bearerFormat?: string;\n flows?: Record<string, unknown>;\n openIdConnectUrl?: string;\n [key: string]: unknown;\n}\n\ninterface EndpointInfo {\n endpoint: string;\n route: string;\n method: string;\n authorizerName: string | null;\n /** @deprecated Use securityScheme instead */\n authorizerType: string | null;\n /** The OpenAPI security scheme definition for this endpoint's authorizer */\n securityScheme: SecuritySchemeObject | null;\n input?: {\n body?: StandardSchemaV1;\n query?: StandardSchemaV1;\n params?: StandardSchemaV1;\n };\n output?: StandardSchemaV1;\n description?: string;\n tags?: string[];\n operationId?: string;\n}\n\ninterface SecuritySchemeInfo {\n name: string;\n type: string;\n scheme: SecuritySchemeObject;\n}\n\n/**\n * Generates TypeScript OpenAPI module from endpoints.\n * Outputs:\n * - securitySchemes: typed security scheme definitions\n * - endpointAuth: runtime map of endpoints to auth requirements\n * - paths: TypeScript interface for type-safe fetcher\n * - schema interfaces: reusable TypeScript types from Zod/Valibot schemas\n */\nexport class OpenApiTsGenerator {\n async generate(\n endpoints: Endpoint<any, any, any, any, any, any>[],\n options: OpenApiTsOptions = {},\n ): Promise<string> {\n const { title = 'API', version = '1.0.0', description } = options;\n\n // Extract endpoint info\n const endpointInfos = await this.extractEndpointInfos(endpoints);\n\n // Collect unique security schemes\n const securitySchemes = this.collectSecuritySchemes(endpointInfos);\n\n // Build endpoint auth map\n const endpointAuth = this.buildEndpointAuthMap(endpointInfos);\n\n // Generate schema interfaces\n const schemaInterfaces = await this.generateSchemaInterfaces(endpointInfos);\n\n // Generate paths interface\n const pathsInterface = await this.generatePathsInterface(endpointInfos);\n\n // Build the final TypeScript module\n return this.buildModule({\n title,\n version,\n description,\n securitySchemes,\n endpointAuth,\n schemaInterfaces,\n pathsInterface,\n });\n }\n\n private async extractEndpointInfos(\n endpoints: Endpoint<any, any, any, any, any, any>[],\n ): Promise<EndpointInfo[]> {\n return endpoints.map((ep) => {\n const route = ep.route.replace(/:(\\w+)/g, '{$1}');\n const method = ep.method.toUpperCase();\n\n // Get security scheme from authorizer (if available)\n // This is the preferred way - the scheme is stored directly on the authorizer\n const securityScheme = ep.authorizer?.securityScheme as\n | SecuritySchemeObject\n | undefined;\n\n return {\n endpoint: `${method} ${route}`,\n route,\n method,\n authorizerName: ep.authorizer?.name ?? null,\n authorizerType: ep.authorizer?.type ?? null,\n securityScheme: securityScheme ?? null,\n input: ep.input,\n output: ep.outputSchema,\n description: ep.description,\n tags: ep.tags,\n operationId: ep.operationId,\n };\n });\n }\n\n private collectSecuritySchemes(\n endpointInfos: EndpointInfo[],\n ): SecuritySchemeInfo[] {\n const schemes = new Map<string, SecuritySchemeInfo>();\n\n for (const info of endpointInfos) {\n if (info.authorizerName && !schemes.has(info.authorizerName)) {\n // Prefer the stored security scheme (from .securitySchemes() or built-ins)\n // Fall back to inference from authorizerType for backward compatibility\n const scheme =\n info.securityScheme ??\n (info.authorizerType\n ? this.mapAuthorizerToSecurityScheme(\n info.authorizerType,\n info.authorizerName,\n )\n : null);\n\n if (scheme) {\n schemes.set(info.authorizerName, {\n name: info.authorizerName,\n type: scheme.type,\n scheme,\n });\n }\n }\n }\n\n return Array.from(schemes.values());\n }\n\n private mapAuthorizerToSecurityScheme(\n type: string,\n _name: string,\n ): SecuritySchemeObject {\n switch (type.toLowerCase()) {\n case 'jwt':\n case 'bearer':\n return {\n type: 'http',\n scheme: 'bearer',\n bearerFormat: 'JWT',\n };\n case 'iam':\n case 'aws-sigv4':\n case 'sigv4':\n return {\n type: 'apiKey',\n in: 'header',\n name: 'Authorization',\n 'x-amazon-apigateway-authtype': 'awsSigv4',\n };\n case 'apikey':\n case 'api-key':\n return {\n type: 'apiKey',\n in: 'header',\n name: 'X-API-Key',\n };\n case 'oauth2':\n return {\n type: 'oauth2',\n flows: {},\n };\n case 'oidc':\n case 'openidconnect':\n return {\n type: 'openIdConnect',\n openIdConnectUrl: '',\n };\n default:\n return {\n type: 'http',\n scheme: 'bearer',\n };\n }\n }\n\n private buildEndpointAuthMap(\n endpointInfos: EndpointInfo[],\n ): Record<string, string | null> {\n const authMap: Record<string, string | null> = {};\n\n for (const info of endpointInfos) {\n authMap[info.endpoint] = info.authorizerName;\n }\n\n return authMap;\n }\n\n private async generateSchemaInterfaces(\n endpointInfos: EndpointInfo[],\n ): Promise<string> {\n const interfaces: string[] = [];\n const generatedNames = new Set<string>();\n // Collect nested schemas with $defs (from .meta({ id: 'X' }))\n const collectedDefs = new Map<string, JsonSchema>();\n\n for (const info of endpointInfos) {\n const baseName = this.getSchemaBaseName(info);\n\n // Input body schema\n if (info.input?.body) {\n const name = await this.getSchemaName(\n info.input.body,\n `${baseName}Input`,\n );\n if (!generatedNames.has(name)) {\n const schema = await this.schemaToInterfaceWithDefs(\n info.input.body,\n name,\n collectedDefs,\n );\n if (schema) {\n interfaces.push(schema);\n generatedNames.add(name);\n }\n }\n }\n\n // Input params schema\n if (info.input?.params) {\n const name = await this.getSchemaName(\n info.input.params,\n `${baseName}Params`,\n );\n if (!generatedNames.has(name)) {\n const schema = await this.schemaToInterfaceWithDefs(\n info.input.params,\n name,\n collectedDefs,\n );\n if (schema) {\n interfaces.push(schema);\n generatedNames.add(name);\n }\n }\n }\n\n // Input query schema\n if (info.input?.query) {\n const name = await this.getSchemaName(\n info.input.query,\n `${baseName}Query`,\n );\n if (!generatedNames.has(name)) {\n const schema = await this.schemaToInterfaceWithDefs(\n info.input.query,\n name,\n collectedDefs,\n );\n if (schema) {\n interfaces.push(schema);\n generatedNames.add(name);\n }\n }\n }\n\n // Output schema\n if (info.output) {\n const name = await this.getSchemaName(info.output, `${baseName}Output`);\n if (!generatedNames.has(name)) {\n const schema = await this.schemaToInterfaceWithDefs(\n info.output,\n name,\n collectedDefs,\n );\n if (schema) {\n interfaces.push(schema);\n generatedNames.add(name);\n }\n }\n }\n }\n\n // Generate interfaces for collected $defs (nested schemas with .meta({ id: 'X' }))\n for (const [defName, defSchema] of collectedDefs) {\n if (!generatedNames.has(defName)) {\n const interfaceStr = this.jsonSchemaToInterface(defSchema, defName);\n interfaces.push(interfaceStr);\n generatedNames.add(defName);\n }\n }\n\n return interfaces.join('\\n\\n');\n }\n\n /**\n * Get the name for a schema, using metadata `id` if available,\n * otherwise falling back to the provided default name.\n */\n private async getSchemaName(\n schema: StandardSchemaV1,\n defaultName: string,\n ): Promise<string> {\n try {\n const metadata = await getSchemaMetadata(schema);\n if (metadata?.id) {\n return this.pascalCase(metadata.id);\n }\n } catch {\n // Ignore metadata extraction errors\n }\n return defaultName;\n }\n\n private getSchemaBaseName(info: EndpointInfo): string {\n if (info.operationId) {\n return this.pascalCase(info.operationId);\n }\n\n // Generate name from method + route\n const routeParts = info.route\n .replace(/[{}]/g, '')\n .split('/')\n .filter(Boolean)\n .map((part) => this.pascalCase(part));\n\n return `${this.pascalCase(info.method.toLowerCase())}${routeParts.join('')}`;\n }\n\n private pascalCase(str: string): string {\n return str\n .replace(/[-_](.)/g, (_, c) => c.toUpperCase())\n .replace(/^./, (c) => c.toUpperCase());\n }\n\n /**\n * Convert schema to interface while collecting $defs for nested schemas\n * with .meta({ id: 'X' }).\n */\n private async schemaToInterfaceWithDefs(\n schema: StandardSchemaV1,\n name: string,\n collectedDefs: Map<string, JsonSchema>,\n ): Promise<string | null> {\n try {\n // Get raw JSON schema with $defs intact (don't use convertStandardSchemaToJsonSchema\n // which strips $defs)\n const vendor = schema['~standard']?.vendor;\n if (!vendor || !(vendor in StandardSchemaJsonSchema)) {\n return null;\n }\n\n const toJsonSchema =\n StandardSchemaJsonSchema[\n vendor as keyof typeof StandardSchemaJsonSchema\n ];\n const jsonSchema = await toJsonSchema(schema);\n if (!jsonSchema) return null;\n\n // Extract $defs from the JSON schema (these come from .meta({ id: 'X' }))\n if (jsonSchema.$defs && typeof jsonSchema.$defs === 'object') {\n for (const [defName, defSchema] of Object.entries(jsonSchema.$defs)) {\n if (!collectedDefs.has(defName)) {\n // Remove the 'id' field from the schema as it's just metadata\n const { id, ...schemaWithoutId } = defSchema as JsonSchema & {\n id?: string;\n };\n collectedDefs.set(defName, schemaWithoutId as JsonSchema);\n }\n }\n }\n\n // Remove $defs from the schema before converting to interface\n const { $defs, ...schemaWithoutDefs } = jsonSchema;\n return this.jsonSchemaToInterface(schemaWithoutDefs, name);\n } catch {\n return null;\n }\n }\n\n private jsonSchemaToInterface(schema: JsonSchema, name: string): string {\n if (schema.type !== 'object' || !schema.properties) {\n // For non-object types, create a type alias\n const typeStr = this.jsonSchemaTypeToTs(schema);\n return `export type ${name} = ${typeStr};`;\n }\n\n const props: string[] = [];\n const required = new Set(schema.required || []);\n\n for (const [propName, propSchema] of Object.entries(schema.properties)) {\n const isRequired = required.has(propName);\n const typeStr = this.jsonSchemaTypeToTs(propSchema as JsonSchema);\n const optionalMark = isRequired ? '' : '?';\n props.push(` ${propName}${optionalMark}: ${typeStr};`);\n }\n\n return `export interface ${name} {\\n${props.join('\\n')}\\n}`;\n }\n\n private jsonSchemaTypeToTs(schema: JsonSchema): string {\n if (!schema) return 'unknown';\n\n if (schema.$ref) {\n // Extract name from $ref\n const refName = schema.$ref.split('/').pop() || 'unknown';\n return refName;\n }\n\n if (schema.anyOf) {\n return schema.anyOf\n .map((s: JsonSchema) => this.jsonSchemaTypeToTs(s))\n .join(' | ');\n }\n\n if (schema.oneOf) {\n return schema.oneOf\n .map((s: JsonSchema) => this.jsonSchemaTypeToTs(s))\n .join(' | ');\n }\n\n if (schema.allOf) {\n return schema.allOf\n .map((s: JsonSchema) => this.jsonSchemaTypeToTs(s))\n .join(' & ');\n }\n\n switch (schema.type) {\n case 'string':\n if (schema.enum) {\n return schema.enum.map((e: string) => `'${e}'`).join(' | ');\n }\n return 'string';\n case 'number':\n case 'integer':\n return 'number';\n case 'boolean':\n return 'boolean';\n case 'null':\n return 'null';\n case 'array':\n if (schema.items) {\n return `Array<${this.jsonSchemaTypeToTs(schema.items as JsonSchema)}>`;\n }\n return 'Array<unknown>';\n case 'object':\n if (schema.properties) {\n const props: string[] = [];\n const required = new Set(schema.required || []);\n for (const [propName, propSchema] of Object.entries(\n schema.properties,\n )) {\n const isRequired = required.has(propName);\n const typeStr = this.jsonSchemaTypeToTs(propSchema as JsonSchema);\n const optionalMark = isRequired ? '' : '?';\n props.push(`${propName}${optionalMark}: ${typeStr}`);\n }\n return `{ ${props.join('; ')} }`;\n }\n if (schema.additionalProperties) {\n const valueType = this.jsonSchemaTypeToTs(\n schema.additionalProperties as JsonSchema,\n );\n return `Record<string, ${valueType}>`;\n }\n return 'Record<string, unknown>';\n default:\n return 'unknown';\n }\n }\n\n private async generatePathsInterface(\n endpointInfos: EndpointInfo[],\n ): Promise<string> {\n const pathGroups = new Map<string, EndpointInfo[]>();\n\n // Group endpoints by route\n for (const info of endpointInfos) {\n const existing = pathGroups.get(info.route) || [];\n existing.push(info);\n pathGroups.set(info.route, existing);\n }\n\n const pathEntries: string[] = [];\n\n for (const [route, infos] of pathGroups) {\n const methodEntries: string[] = [];\n\n for (const info of infos) {\n const methodDef = await this.generateMethodDefinition(info);\n methodEntries.push(` ${info.method.toLowerCase()}: ${methodDef};`);\n }\n\n // Add path parameters if present\n const firstWithParams = infos.find((i) => i.input?.params);\n let paramsEntry = '';\n if (firstWithParams?.input?.params) {\n const paramsName = await this.getSchemaName(\n firstWithParams.input.params,\n `${this.getSchemaBaseName(firstWithParams)}Params`,\n );\n paramsEntry = `\\n parameters: {\\n path: ${paramsName};\\n };`;\n }\n\n pathEntries.push(\n ` '${route}': {${paramsEntry}\\n${methodEntries.join('\\n')}\\n };`,\n );\n }\n\n return `export interface paths {\\n${pathEntries.join('\\n')}\\n}`;\n }\n\n private async generateMethodDefinition(info: EndpointInfo): Promise<string> {\n const parts: string[] = [];\n const baseName = this.getSchemaBaseName(info);\n\n // Request body\n if (info.input?.body) {\n const bodyName = await this.getSchemaName(\n info.input.body,\n `${baseName}Input`,\n );\n parts.push(`requestBody: {\n content: {\n 'application/json': ${bodyName};\n };\n }`);\n }\n\n // Query parameters\n if (info.input?.query) {\n const queryName = await this.getSchemaName(\n info.input.query,\n `${baseName}Query`,\n );\n parts.push(`parameters: {\n query: ${queryName};\n }`);\n }\n\n // Responses\n const outputName = info.output\n ? await this.getSchemaName(info.output, `${baseName}Output`)\n : 'unknown';\n parts.push(`responses: {\n 200: {\n content: {\n 'application/json': ${outputName};\n };\n };\n }`);\n\n return `{\\n ${parts.join(';\\n ')};\\n }`;\n }\n\n private buildModule(params: {\n title: string;\n version: string;\n description?: string;\n securitySchemes: SecuritySchemeInfo[];\n endpointAuth: Record<string, string | null>;\n schemaInterfaces: string;\n pathsInterface: string;\n }): string {\n const {\n title,\n version,\n description,\n securitySchemes,\n endpointAuth,\n schemaInterfaces,\n pathsInterface,\n } = params;\n\n const securitySchemesObj = securitySchemes.reduce(\n (acc, s) => {\n acc[s.name] = s.scheme;\n return acc;\n },\n {} as Record<string, SecuritySchemeObject>,\n );\n\n const schemeNames = securitySchemes.map((s) => `'${s.name}'`).join(' | ');\n\n // Generate createApi only if there are security schemes\n const hasSecuritySchemes = schemeNames.length > 0;\n\n const createApiSection = hasSecuritySchemes\n ? `\n// ============================================================\n// API Client Factory\n// ============================================================\n\nimport {\n createAuthAwareFetcher,\n type AuthStrategy,\n} from '@geekmidas/client/auth-fetcher';\nimport { createEndpointHooks } from '@geekmidas/client/endpoint-hooks';\nimport type { QueryClient } from '@tanstack/react-query';\n\n/**\n * Options for creating the API client.\n */\nexport interface CreateApiOptions {\n /** Base URL for all API requests (required) */\n baseURL: string;\n /** Auth strategies for each security scheme used in this API */\n authStrategies: Record<SecuritySchemeId, AuthStrategy>;\n /** Optional React Query client instance */\n queryClient?: QueryClient;\n /** Optional request interceptor */\n onRequest?: (config: RequestInit) => RequestInit | Promise<RequestInit>;\n}\n\n/**\n * Create a type-safe API client with authentication and React Query hooks.\n *\n * @example\n * \\`\\`\\`typescript\n * const api = createApi({\n * baseURL: 'https://api.example.com',\n * authStrategies: {\n * jwt: { type: 'bearer', tokenProvider },\n * },\n * });\n *\n * // Imperative fetch\n * const user = await api('GET /users/{id}', { params: { id: '123' } });\n *\n * // React Query hooks\n * const { data } = api.useQuery('GET /users/{id}', { params: { id: '123' } });\n * const mutation = api.useMutation('POST /users');\n * \\`\\`\\`\n */\nexport function createApi(options: CreateApiOptions) {\n const fetcher = createAuthAwareFetcher<paths, typeof endpointAuth, typeof securitySchemes>({\n baseURL: options.baseURL,\n endpointAuth,\n securitySchemes,\n authStrategies: options.authStrategies,\n onRequest: options.onRequest,\n });\n\n const hooks = createEndpointHooks<paths>(fetcher, { queryClient: options.queryClient });\n\n return Object.assign(fetcher, hooks);\n}\n`\n : `\n// ============================================================\n// API Client Factory\n// ============================================================\n\nimport { TypedFetcher, type FetcherOptions } from '@geekmidas/client/fetcher';\nimport { createEndpointHooks } from '@geekmidas/client/endpoint-hooks';\nimport type { QueryClient } from '@tanstack/react-query';\n\n/**\n * Options for creating the API client.\n */\nexport interface CreateApiOptions extends Omit<FetcherOptions, 'baseURL'> {\n /** Base URL for all API requests (required) */\n baseURL: string;\n /** Optional React Query client instance */\n queryClient?: QueryClient;\n}\n\n/**\n * Create a type-safe API client with React Query hooks.\n *\n * @example\n * \\`\\`\\`typescript\n * const api = createApi({\n * baseURL: 'https://api.example.com',\n * });\n *\n * // Imperative fetch\n * const data = await api('GET /health');\n *\n * // React Query hooks\n * const { data } = api.useQuery('GET /health');\n * \\`\\`\\`\n */\nexport function createApi(options: CreateApiOptions) {\n const { queryClient, ...fetcherOptions } = options;\n const fetcher = new TypedFetcher<paths>(fetcherOptions);\n\n const hooks = createEndpointHooks<paths>(fetcher.request.bind(fetcher), { queryClient });\n\n return Object.assign(fetcher.request.bind(fetcher), hooks);\n}\n`;\n\n return `// Auto-generated by @geekmidas/cli - DO NOT EDIT\n// Generated: ${new Date().toISOString()}\n\n// ============================================================\n// Security Scheme Type\n// ============================================================\n\ninterface SecuritySchemeObject {\n type: 'apiKey' | 'http' | 'mutualTLS' | 'oauth2' | 'openIdConnect';\n description?: string;\n name?: string;\n in?: 'query' | 'header' | 'cookie';\n scheme?: string;\n bearerFormat?: string;\n flows?: Record<string, unknown>;\n openIdConnectUrl?: string;\n [key: string]: unknown;\n}\n\n// ============================================================\n// API Info\n// ============================================================\n\nexport const apiInfo = {\n title: '${title}',\n version: '${version}',${description ? `\\n description: '${description.replace(/'/g, \"\\\\'\")}',` : ''}\n} as const;\n\n// ============================================================\n// Security Schemes\n// ============================================================\n\n/**\n * Available security schemes for this API.\n * Maps authorizer names to OpenAPI security scheme definitions.\n */\nexport const securitySchemes = ${JSON.stringify(securitySchemesObj, null, 2).replace(/\"([a-zA-Z_$][a-zA-Z0-9_$]*)\":/g, '$1:')} as const satisfies Record<string, SecuritySchemeObject>;\n\nexport type SecuritySchemeId = ${schemeNames || 'never'};\n\n// ============================================================\n// Endpoint Authentication Map\n// ============================================================\n\n/**\n * Runtime map of endpoints to their required authentication scheme.\n * \\`null\\` indicates a public endpoint (no auth required).\n */\nexport const endpointAuth = ${JSON.stringify(endpointAuth, null, 2).replace(/\"([^\"]+)\":/g, \"'$1':\")} as const satisfies Record<string, SecuritySchemeId | null>;\n\nexport type EndpointString = keyof typeof endpointAuth;\n\nexport type AuthenticatedEndpoint = {\n [K in EndpointString]: typeof endpointAuth[K] extends null ? never : K;\n}[EndpointString];\n\nexport type PublicEndpoint = {\n [K in EndpointString]: typeof endpointAuth[K] extends null ? K : never;\n}[EndpointString];\n\n// ============================================================\n// Schema Definitions\n// ============================================================\n\n${schemaInterfaces}\n\n// ============================================================\n// OpenAPI Paths\n// ============================================================\n\n${pathsInterface}\n${createApiSection}\n`;\n }\n}\n","#!/usr/bin/env -S npx tsx\n\nimport { mkdir, writeFile } from 'node:fs/promises';\nimport { dirname, join } from 'node:path';\nimport { loadConfig } from './config.js';\nimport { EndpointGenerator } from './generators/EndpointGenerator.js';\nimport { OpenApiTsGenerator } from './generators/OpenApiTsGenerator.js';\nimport type { GkmConfig, OpenApiConfig } from './types.js';\n\ninterface OpenAPIOptions {\n cwd?: string;\n}\n\n/**\n * Fixed output path for generated OpenAPI client (not configurable)\n */\nexport const OPENAPI_OUTPUT_PATH = './.gkm/openapi.ts';\n\n/**\n * Resolve OpenAPI config from GkmConfig\n */\nexport function resolveOpenApiConfig(\n config: GkmConfig,\n): OpenApiConfig & { enabled: boolean } {\n if (config.openapi === false) {\n return { enabled: false };\n }\n\n if (config.openapi === true || config.openapi === undefined) {\n return {\n enabled: config.openapi === true,\n title: 'API Documentation',\n version: '1.0.0',\n description: 'Auto-generated API documentation from endpoints',\n };\n }\n\n return {\n enabled: config.openapi.enabled !== false,\n title: config.openapi.title || 'API Documentation',\n version: config.openapi.version || '1.0.0',\n description:\n config.openapi.description ||\n 'Auto-generated API documentation from endpoints',\n };\n}\n\n/**\n * Generate OpenAPI spec from endpoints\n * @returns Object with output path and endpoint count, or null if disabled\n */\nexport async function generateOpenApi(\n config: GkmConfig,\n options: { silent?: boolean } = {},\n): Promise<{ outputPath: string; endpointCount: number } | null> {\n const logger = options.silent ? { log: () => {} } : console;\n const openApiConfig = resolveOpenApiConfig(config);\n\n if (!openApiConfig.enabled) {\n return null;\n }\n\n const endpointGenerator = new EndpointGenerator();\n const loadedEndpoints = await endpointGenerator.load(config.routes);\n\n if (loadedEndpoints.length === 0) {\n logger.log('No valid endpoints found for OpenAPI generation');\n return null;\n }\n\n const endpoints = loadedEndpoints.map(({ construct }) => construct);\n const outputPath = join(process.cwd(), OPENAPI_OUTPUT_PATH);\n\n await mkdir(dirname(outputPath), { recursive: true });\n\n const tsGenerator = new OpenApiTsGenerator();\n const tsContent = await tsGenerator.generate(endpoints, {\n title: openApiConfig.title!,\n version: openApiConfig.version!,\n description: openApiConfig.description!,\n });\n\n await writeFile(outputPath, tsContent);\n logger.log(`📄 OpenAPI client generated: ${OPENAPI_OUTPUT_PATH}`);\n\n return { outputPath, endpointCount: loadedEndpoints.length };\n}\n\nexport async function openapiCommand(\n options: OpenAPIOptions = {},\n): Promise<void> {\n const logger = console;\n\n try {\n const config = await loadConfig(options.cwd);\n\n // Enable openapi if not configured\n if (!config.openapi) {\n config.openapi = { enabled: true };\n }\n\n const result = await generateOpenApi(config);\n\n if (result) {\n logger.log(`Found ${result.endpointCount} endpoints`);\n }\n } catch (error) {\n throw new Error(`OpenAPI generation failed: ${(error as Error).message}`);\n }\n}\n"],"mappings":"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"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"openapi-tAIbJJU_.cjs","names":["context: BuildContext","outputDir: string","generator: ConstructGenerator<T, R>","patterns?: Routes","options?: GeneratorOptions","constructs: GeneratedConstruct<T>[]","module","value: any","context: BuildContext","constructs: GeneratedConstruct<\n Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >\n >[]","outputDir: string","options?: GeneratorOptions","routes: RouteInfo[]","path","routeInfo: RouteInfo","sourceFile: string","exportName: string","provider: LegacyProvider","_endpoint: Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >","content: string","endpoints: GeneratedConstruct<\n Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >\n >[]","_context: BuildContext","exports","importPath: string","envParserPath: string","envParserImportPattern: string","endpoints: Endpoint<any, any, any, any, any, any>[]","options: OpenApiTsOptions","endpointInfos: EndpointInfo[]","type: string","_name: string","authMap: Record<string, string | null>","interfaces: string[]","schema: StandardSchemaV1","defaultName: string","info: EndpointInfo","str: string","name: string","collectedDefs: Map<string, JsonSchema>","StandardSchemaJsonSchema","schema: JsonSchema","props: string[]","s: JsonSchema","e: string","pathEntries: string[]","methodEntries: string[]","parts: string[]","params: {\n title: string;\n version: string;\n description?: string;\n securitySchemes: SecuritySchemeInfo[];\n endpointAuth: Record<string, string | null>;\n schemaInterfaces: string;\n pathsInterface: string;\n }","config: GkmConfig","options: { silent?: boolean }","options: OpenAPIOptions"],"sources":["../src/generators/Generator.ts","../src/generators/EndpointGenerator.ts","../src/generators/OpenApiTsGenerator.ts","../src/openapi.ts"],"sourcesContent":["import { relative } from 'path';\nimport type { Construct } from '@geekmidas/constructs';\nimport fg from 'fast-glob';\nimport kebabCase from 'lodash.kebabcase';\nimport type { BuildContext } from '../build/types';\nimport type { LegacyProvider, Routes } from '../types';\n\nexport interface GeneratorOptions {\n provider?: LegacyProvider;\n [key: string]: any;\n}\n\nexport abstract class ConstructGenerator<T extends Construct, R = void> {\n abstract isConstruct(value: any): value is T;\n\n static async build<T extends Construct, R = void>(\n context: BuildContext,\n outputDir: string,\n generator: ConstructGenerator<T, R>,\n patterns?: Routes,\n options?: GeneratorOptions,\n ): Promise<R> {\n const constructs = await generator.load(patterns);\n return generator.build(context, constructs, outputDir, options);\n }\n\n abstract build(\n context: BuildContext,\n constructs: GeneratedConstruct<T>[],\n outputDir: string,\n options?: GeneratorOptions,\n ): Promise<R>;\n\n async load(\n patterns?: Routes,\n cwd = process.cwd(),\n ): Promise<GeneratedConstruct<T>[]> {\n const logger = console;\n\n // Normalize patterns to array\n const globPatterns = Array.isArray(patterns)\n ? patterns\n : patterns\n ? [patterns]\n : [];\n\n // Find all files\n const files = fg.stream(globPatterns, {\n cwd,\n absolute: true,\n });\n\n // Load constructs\n const constructs: GeneratedConstruct<T>[] = [];\n\n for await (const f of files) {\n try {\n const file = f.toString();\n const module = await import(file);\n\n // Check all exports for constructs\n for (const [key, construct] of Object.entries(module)) {\n if (this.isConstruct(construct)) {\n constructs.push({\n key,\n name: kebabCase(key),\n construct,\n path: {\n absolute: file,\n relative: relative(process.cwd(), file),\n },\n });\n }\n }\n } catch (error) {\n logger.warn(`Failed to load ${f}:`, (error as Error).message);\n throw new Error(\n 'Failed to load constructs. Please check the logs for details.',\n );\n }\n }\n\n return constructs;\n }\n}\n\nexport interface GeneratedConstruct<T extends Construct> {\n key: string;\n name: string;\n construct: T;\n path: {\n absolute: string;\n relative: string;\n };\n}\n","import { mkdir, writeFile } from 'node:fs/promises';\nimport { dirname, join, relative } from 'node:path';\nimport { Endpoint } from '@geekmidas/constructs/endpoints';\nimport type { BuildContext } from '../build/types';\nimport type { LegacyProvider, RouteInfo } from '../types';\nimport {\n ConstructGenerator,\n type GeneratedConstruct,\n type GeneratorOptions,\n} from './Generator';\n\nexport class EndpointGenerator extends ConstructGenerator<\n Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >,\n RouteInfo[]\n> {\n isConstruct(\n value: any,\n ): value is Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n > {\n return Endpoint.isEndpoint(value);\n }\n\n async build(\n context: BuildContext,\n constructs: GeneratedConstruct<\n Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >\n >[],\n outputDir: string,\n options?: GeneratorOptions,\n ): Promise<RouteInfo[]> {\n const provider = options?.provider || 'aws-apigatewayv2';\n const enableOpenApi = options?.enableOpenApi || false;\n const logger = console;\n const routes: RouteInfo[] = [];\n\n if (constructs.length === 0) {\n return routes;\n }\n\n if (provider === 'server') {\n // Generate endpoints.ts and app.ts\n await this.generateEndpointsFile(outputDir, constructs, context);\n const appFile = await this.generateAppFile(outputDir, context);\n\n routes.push({\n path: '*',\n method: 'ALL',\n handler: relative(process.cwd(), appFile),\n authorizer: 'none',\n });\n\n logger.log(\n `Generated server with ${constructs.length} endpoints${enableOpenApi ? ' (OpenAPI enabled)' : ''}`,\n );\n } else if (provider === 'aws-lambda') {\n // For aws-lambda, create routes subdirectory\n const routesDir = join(outputDir, 'routes');\n await mkdir(routesDir, { recursive: true });\n\n // Generate individual handlers for API Gateway routes\n for (const { key, construct, path } of constructs) {\n const handlerFile = await this.generateHandlerFile(\n routesDir,\n path.relative,\n key,\n 'aws-apigatewayv2',\n construct,\n context,\n );\n\n const routeInfo: RouteInfo = {\n path: construct._path,\n method: construct.method,\n handler: relative(process.cwd(), handlerFile).replace(\n /\\.ts$/,\n '.handler',\n ),\n timeout: construct.timeout,\n memorySize: construct.memorySize,\n environment: await construct.getEnvironment(),\n authorizer: construct.authorizer?.name ?? 'none',\n };\n\n routes.push(routeInfo);\n logger.log(\n `Generated handler for ${routeInfo.method} ${routeInfo.path}`,\n );\n }\n } else {\n // Generate individual handler files for AWS API Gateway providers\n for (const { key, construct, path } of constructs) {\n const handlerFile = await this.generateHandlerFile(\n outputDir,\n path.relative,\n key,\n provider,\n construct,\n context,\n );\n\n const routeInfo: RouteInfo = {\n path: construct._path,\n method: construct.method,\n handler: relative(process.cwd(), handlerFile).replace(\n /\\.ts$/,\n '.handler',\n ),\n timeout: construct.timeout,\n memorySize: construct.memorySize,\n environment: await construct.getEnvironment(),\n authorizer: construct.authorizer?.name ?? 'none',\n };\n\n routes.push(routeInfo);\n logger.log(\n `Generated handler for ${routeInfo.method} ${routeInfo.path}`,\n );\n }\n }\n\n return routes;\n }\n\n private async generateHandlerFile(\n outputDir: string,\n sourceFile: string,\n exportName: string,\n provider: LegacyProvider,\n _endpoint: Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >,\n context: BuildContext,\n ): Promise<string> {\n const handlerFileName = `${exportName}.ts`;\n const handlerPath = join(outputDir, handlerFileName);\n\n const relativePath = relative(dirname(handlerPath), sourceFile);\n const importPath = relativePath.replace(/\\.ts$/, '.js');\n\n const relativeEnvParserPath = relative(\n dirname(handlerPath),\n context.envParserPath,\n );\n\n let content: string;\n\n switch (provider) {\n case 'aws-apigatewayv1':\n content = this.generateAWSApiGatewayV1Handler(\n importPath,\n exportName,\n relativeEnvParserPath,\n context.envParserImportPattern,\n );\n break;\n case 'aws-apigatewayv2':\n content = this.generateAWSApiGatewayV2Handler(\n importPath,\n exportName,\n relativeEnvParserPath,\n context.envParserImportPattern,\n );\n break;\n case 'server':\n content = this.generateServerHandler(importPath, exportName);\n break;\n default:\n throw new Error(`Unsupported provider: ${provider}`);\n }\n\n await writeFile(handlerPath, content);\n return handlerPath;\n }\n\n private async generateEndpointsFile(\n outputDir: string,\n endpoints: GeneratedConstruct<\n Endpoint<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n >\n >[],\n _context: BuildContext,\n ): Promise<string> {\n const endpointsFileName = 'endpoints.ts';\n const endpointsPath = join(outputDir, endpointsFileName);\n\n // Group imports by file\n const importsByFile = new Map<string, string[]>();\n\n for (const { path, key } of endpoints) {\n const relativePath = relative(dirname(endpointsPath), path.relative);\n const importPath = relativePath.replace(/\\.ts$/, '.js');\n\n if (!importsByFile.has(importPath)) {\n importsByFile.set(importPath, []);\n }\n importsByFile.get(importPath)!.push(key);\n }\n\n // Generate import statements\n const imports = Array.from(importsByFile.entries())\n .map(\n ([importPath, exports]) =>\n `import { ${exports.join(', ')} } from '${importPath}';`,\n )\n .join('\\n');\n\n const allExportNames = endpoints.map(({ key }) => key);\n\n const content = `import type { EnvironmentParser } from '@geekmidas/envkit';\nimport type { Logger } from '@geekmidas/logger';\nimport { HonoEndpoint } from '@geekmidas/constructs/hono';\nimport { Endpoint } from '@geekmidas/constructs/endpoints';\nimport { ServiceDiscovery } from '@geekmidas/services';\nimport type { Hono } from 'hono';\n${imports}\n\nconst endpoints: Endpoint<any, any, any, any, any, any, any, any, any, any, any, any, any, any>[] = [\n ${allExportNames.join(',\\n ')}\n];\n\nexport async function setupEndpoints(\n app: Hono,\n envParser: EnvironmentParser<any>,\n logger: Logger,\n enableOpenApi: boolean = true,\n): Promise<void> {\n const serviceDiscovery = ServiceDiscovery.getInstance(\n logger,\n envParser\n );\n\n // Configure OpenAPI options based on enableOpenApi flag\n const openApiOptions: any = enableOpenApi ? {\n docsPath: '/__docs',\n openApiOptions: {\n title: 'API Documentation',\n version: '1.0.0',\n description: 'Generated API documentation'\n }\n } : { docsPath: false };\n\n HonoEndpoint.addRoutes(endpoints, serviceDiscovery, app, openApiOptions);\n\n // Add Swagger UI if OpenAPI is enabled\n if (enableOpenApi) {\n try {\n const { swaggerUI } = await import('@hono/swagger-ui');\n app.get('/__docs/ui', swaggerUI({ url: '/__docs' }));\n } catch {\n // @hono/swagger-ui not installed, skip Swagger UI\n }\n }\n}\n`;\n\n await writeFile(endpointsPath, content);\n\n return endpointsPath;\n }\n\n private async generateAppFile(\n outputDir: string,\n context: BuildContext,\n ): Promise<string> {\n const appFileName = 'app.ts';\n const appPath = join(outputDir, appFileName);\n\n const relativeLoggerPath = relative(dirname(appPath), context.loggerPath);\n\n const relativeEnvParserPath = relative(\n dirname(appPath),\n context.envParserPath,\n );\n\n // Generate telescope imports and setup if enabled\n const telescopeEnabled = context.telescope?.enabled;\n const telescopeWebSocketEnabled = context.telescope?.websocket;\n const usesExternalTelescope = !!context.telescope?.telescopePath;\n\n // Generate studio imports and setup if enabled\n const studioEnabled = context.studio?.enabled;\n const usesExternalStudio = !!context.studio?.studioPath;\n\n // Generate imports based on whether telescope is external or inline\n let telescopeImports = '';\n if (telescopeEnabled) {\n if (usesExternalTelescope) {\n const relativeTelescopePath = relative(\n dirname(appPath),\n context.telescope!.telescopePath!,\n );\n telescopeImports = `import ${context.telescope!.telescopeImportPattern} from '${relativeTelescopePath}';\nimport { createMiddleware, createUI } from '@geekmidas/telescope/hono';`;\n } else {\n telescopeImports = `import { Telescope, InMemoryStorage } from '@geekmidas/telescope';\nimport { createMiddleware, createUI } from '@geekmidas/telescope/hono';`;\n }\n }\n\n // Generate imports for studio\n let studioImports = '';\n if (studioEnabled) {\n if (usesExternalStudio) {\n const relativeStudioPath = relative(\n dirname(appPath),\n context.studio!.studioPath!,\n );\n studioImports = `import ${context.studio!.studioImportPattern} from '${relativeStudioPath}';\nimport { createStudioApp } from '@geekmidas/studio/server/hono';`;\n } else {\n studioImports = `// Studio requires a configured instance - use studio config path\n// import { createStudioApp } from '@geekmidas/studio/server/hono';`;\n }\n }\n\n const telescopeWebSocketSetupCode = telescopeWebSocketEnabled\n ? `\n // Setup WebSocket for real-time telescope updates\n try {\n const { createNodeWebSocket } = await import('@hono/node-ws');\n const { injectWebSocket, upgradeWebSocket } = createNodeWebSocket({ app: honoApp });\n // Add WebSocket route directly to main app (sub-app routes don't support WS upgrade)\n honoApp.get('${context.telescope!.path}/ws', upgradeWebSocket(() => ({\n onOpen: (_event: Event, ws: any) => {\n telescope.addWsClient(ws);\n },\n onClose: (_event: Event, ws: any) => {\n telescope.removeWsClient(ws);\n },\n onMessage: (event: MessageEvent, ws: any) => {\n try {\n const data = JSON.parse(event.data);\n if (data.type === 'ping') {\n ws.send(JSON.stringify({ type: 'pong' }));\n }\n } catch {\n // Ignore invalid messages\n }\n },\n })));\n // Store injectWebSocket for server entry to call after serve()\n (honoApp as any).__injectWebSocket = injectWebSocket;\n logger.info('Telescope WebSocket enabled');\n } catch (e) {\n logger.warn({ error: e }, 'WebSocket support not available - install @hono/node-ws for real-time updates');\n }\n`\n : '';\n\n // Generate telescope setup - either use external instance or create inline\n let telescopeSetup = '';\n if (telescopeEnabled) {\n if (usesExternalTelescope) {\n // Use external telescope instance - no need to create one\n telescopeSetup = `\n${telescopeWebSocketSetupCode}\n // Add telescope middleware (before endpoints to capture all requests)\n honoApp.use('*', createMiddleware(telescope));\n\n // Mount telescope UI\n const telescopeUI = createUI(telescope);\n honoApp.route('${context.telescope!.path}', telescopeUI);\n`;\n } else {\n // Create inline telescope instance\n telescopeSetup = `\n // Setup Telescope for debugging/monitoring\n const telescopeStorage = new InMemoryStorage({ maxEntries: ${context.telescope!.maxEntries} });\n const telescope = new Telescope({\n enabled: true,\n path: '${context.telescope!.path}',\n ignorePatterns: ${JSON.stringify(context.telescope!.ignore)},\n recordBody: ${context.telescope!.recordBody},\n storage: telescopeStorage,\n });\n${telescopeWebSocketSetupCode}\n // Add telescope middleware (before endpoints to capture all requests)\n honoApp.use('*', createMiddleware(telescope));\n\n // Mount telescope UI\n const telescopeUI = createUI(telescope);\n honoApp.route('${context.telescope!.path}', telescopeUI);\n`;\n }\n }\n\n // Generate studio setup - requires external instance\n let studioSetup = '';\n if (studioEnabled && usesExternalStudio) {\n studioSetup = `\n // Mount Studio data browser UI\n const studioApp = createStudioApp(studio);\n honoApp.route('${context.studio!.path}', studioApp);\n`;\n }\n\n const content = `/**\n * Generated server application\n *\n * ⚠️ WARNING: This is for LOCAL DEVELOPMENT ONLY\n * The subscriber polling mechanism is not production-ready.\n * For production, use AWS Lambda with SQS/SNS event sources.\n */\nimport { Hono } from 'hono';\nimport type { Hono as HonoType } from 'hono';\nimport { setupEndpoints } from './endpoints.js';\nimport { setupSubscribers } from './subscribers.js';\nimport ${context.envParserImportPattern} from '${relativeEnvParserPath}';\nimport ${context.loggerImportPattern} from '${relativeLoggerPath}';\n${telescopeImports}\n${studioImports}\n\nexport interface ServerApp {\n app: HonoType;\n start: (options?: {\n port?: number;\n serve: (app: HonoType, port: number) => void | Promise<void>;\n }) => Promise<void>;\n}\n\n/**\n * Create and configure the Hono application\n *\n * @param app - Optional Hono app instance to configure (creates new one if not provided)\n * @param enableOpenApi - Enable OpenAPI documentation (default: true)\n * @returns Server app with configured Hono app and start function\n *\n * @example\n * // With Bun\n * import { createApp } from './.gkm/server/app.js';\n *\n * const { app, start } = await createApp();\n *\n * await start({\n * port: 3000,\n * serve: (app, port) => {\n * Bun.serve({ port, fetch: app.fetch });\n * }\n * });\n *\n * @example\n * // With Node.js (using @hono/node-server)\n * import { serve } from '@hono/node-server';\n * import { createApp } from './.gkm/server/app.js';\n *\n * const { app, start } = await createApp();\n *\n * await start({\n * port: 3000,\n * serve: (app, port) => {\n * serve({ fetch: app.fetch, port });\n * }\n * });\n */\nexport async function createApp(app?: HonoType, enableOpenApi: boolean = true): Promise<ServerApp> {\n const honoApp = app || new Hono();\n${telescopeSetup}${studioSetup}\n // Setup HTTP endpoints\n await setupEndpoints(honoApp, envParser, logger, enableOpenApi);\n\n return {\n app: honoApp,\n async start(options) {\n if (!options?.serve) {\n throw new Error(\n 'serve function is required. Pass a serve function for your runtime:\\\\n' +\n ' - Bun: (app, port) => Bun.serve({ port, fetch: app.fetch })\\\\n' +\n ' - Node: (app, port) => serve({ fetch: app.fetch, port })'\n );\n }\n\n const port = options.port ?? 3000;\n\n // Start subscribers in background (non-blocking, local development only)\n await setupSubscribers(envParser, logger).catch((error) => {\n logger.error({ error }, 'Failed to start subscribers');\n });\n\n logger.info({ port }, 'Starting server');\n\n // Start HTTP server using provided serve function\n await options.serve(honoApp, port);\n\n logger.info({ port }, 'Server started');\n }\n };\n}\n\n// Default export for convenience\nexport default createApp;\n`;\n\n await writeFile(appPath, content);\n\n return appPath;\n }\n\n private generateAWSApiGatewayV1Handler(\n importPath: string,\n exportName: string,\n envParserPath: string,\n envParserImportPattern: string,\n ): string {\n return `import { AmazonApiGatewayV1Endpoint } from '@geekmidas/constructs/aws';\nimport { ${exportName} } from '${importPath}';\nimport ${envParserImportPattern} from '${envParserPath}';\n\nconst adapter = new AmazonApiGatewayV1Endpoint(envParser, ${exportName});\n\nexport const handler = adapter.handler;\n`;\n }\n\n private generateAWSApiGatewayV2Handler(\n importPath: string,\n exportName: string,\n envParserPath: string,\n envParserImportPattern: string,\n ): string {\n return `import { AmazonApiGatewayV2Endpoint } from '@geekmidas/constructs/aws';\nimport { ${exportName} } from '${importPath}';\nimport ${envParserImportPattern} from '${envParserPath}';\n\nconst adapter = new AmazonApiGatewayV2Endpoint(envParser, ${exportName});\n\nexport const handler = adapter.handler;\n`;\n }\n\n private generateServerHandler(\n importPath: string,\n exportName: string,\n ): string {\n return `import { ${exportName} } from '${importPath}';\n\n// Server handler - implement based on your server framework\nexport const handler = ${exportName};\n`;\n }\n}\n","import type { Endpoint } from '@geekmidas/constructs/endpoints';\nimport {\n StandardSchemaJsonSchema,\n getSchemaMetadata,\n} from '@geekmidas/schema/conversion';\nimport type { StandardSchemaV1 } from '@standard-schema/spec';\n\ninterface OpenApiTsOptions {\n title?: string;\n version?: string;\n description?: string;\n}\n\n// JSON Schema type definition\ninterface JsonSchema {\n type?: string;\n properties?: Record<string, JsonSchema>;\n items?: JsonSchema;\n required?: string[];\n enum?: string[];\n $ref?: string;\n anyOf?: JsonSchema[];\n oneOf?: JsonSchema[];\n allOf?: JsonSchema[];\n additionalProperties?: boolean | JsonSchema;\n [key: string]: unknown;\n}\n\n// Security scheme type (OpenAPI 3.1)\ninterface SecuritySchemeObject {\n type: 'apiKey' | 'http' | 'mutualTLS' | 'oauth2' | 'openIdConnect';\n description?: string;\n name?: string;\n in?: 'query' | 'header' | 'cookie';\n scheme?: string;\n bearerFormat?: string;\n flows?: Record<string, unknown>;\n openIdConnectUrl?: string;\n [key: string]: unknown;\n}\n\ninterface EndpointInfo {\n endpoint: string;\n route: string;\n method: string;\n authorizerName: string | null;\n /** @deprecated Use securityScheme instead */\n authorizerType: string | null;\n /** The OpenAPI security scheme definition for this endpoint's authorizer */\n securityScheme: SecuritySchemeObject | null;\n input?: {\n body?: StandardSchemaV1;\n query?: StandardSchemaV1;\n params?: StandardSchemaV1;\n };\n output?: StandardSchemaV1;\n description?: string;\n tags?: string[];\n operationId?: string;\n}\n\ninterface SecuritySchemeInfo {\n name: string;\n type: string;\n scheme: SecuritySchemeObject;\n}\n\n/**\n * Generates TypeScript OpenAPI module from endpoints.\n * Outputs:\n * - securitySchemes: typed security scheme definitions\n * - endpointAuth: runtime map of endpoints to auth requirements\n * - paths: TypeScript interface for type-safe fetcher\n * - schema interfaces: reusable TypeScript types from Zod/Valibot schemas\n */\nexport class OpenApiTsGenerator {\n async generate(\n endpoints: Endpoint<any, any, any, any, any, any>[],\n options: OpenApiTsOptions = {},\n ): Promise<string> {\n const { title = 'API', version = '1.0.0', description } = options;\n\n // Extract endpoint info\n const endpointInfos = await this.extractEndpointInfos(endpoints);\n\n // Collect unique security schemes\n const securitySchemes = this.collectSecuritySchemes(endpointInfos);\n\n // Build endpoint auth map\n const endpointAuth = this.buildEndpointAuthMap(endpointInfos);\n\n // Generate schema interfaces\n const schemaInterfaces = await this.generateSchemaInterfaces(endpointInfos);\n\n // Generate paths interface\n const pathsInterface = await this.generatePathsInterface(endpointInfos);\n\n // Build the final TypeScript module\n return this.buildModule({\n title,\n version,\n description,\n securitySchemes,\n endpointAuth,\n schemaInterfaces,\n pathsInterface,\n });\n }\n\n private async extractEndpointInfos(\n endpoints: Endpoint<any, any, any, any, any, any>[],\n ): Promise<EndpointInfo[]> {\n return endpoints.map((ep) => {\n const route = ep.route.replace(/:(\\w+)/g, '{$1}');\n const method = ep.method.toUpperCase();\n\n // Get security scheme from authorizer (if available)\n // This is the preferred way - the scheme is stored directly on the authorizer\n const securityScheme = ep.authorizer?.securityScheme as\n | SecuritySchemeObject\n | undefined;\n\n return {\n endpoint: `${method} ${route}`,\n route,\n method,\n authorizerName: ep.authorizer?.name ?? null,\n authorizerType: ep.authorizer?.type ?? null,\n securityScheme: securityScheme ?? null,\n input: ep.input,\n output: ep.outputSchema,\n description: ep.description,\n tags: ep.tags,\n operationId: ep.operationId,\n };\n });\n }\n\n private collectSecuritySchemes(\n endpointInfos: EndpointInfo[],\n ): SecuritySchemeInfo[] {\n const schemes = new Map<string, SecuritySchemeInfo>();\n\n for (const info of endpointInfos) {\n if (info.authorizerName && !schemes.has(info.authorizerName)) {\n // Prefer the stored security scheme (from .securitySchemes() or built-ins)\n // Fall back to inference from authorizerType for backward compatibility\n const scheme =\n info.securityScheme ??\n (info.authorizerType\n ? this.mapAuthorizerToSecurityScheme(\n info.authorizerType,\n info.authorizerName,\n )\n : null);\n\n if (scheme) {\n schemes.set(info.authorizerName, {\n name: info.authorizerName,\n type: scheme.type,\n scheme,\n });\n }\n }\n }\n\n return Array.from(schemes.values());\n }\n\n private mapAuthorizerToSecurityScheme(\n type: string,\n _name: string,\n ): SecuritySchemeObject {\n switch (type.toLowerCase()) {\n case 'jwt':\n case 'bearer':\n return {\n type: 'http',\n scheme: 'bearer',\n bearerFormat: 'JWT',\n };\n case 'iam':\n case 'aws-sigv4':\n case 'sigv4':\n return {\n type: 'apiKey',\n in: 'header',\n name: 'Authorization',\n 'x-amazon-apigateway-authtype': 'awsSigv4',\n };\n case 'apikey':\n case 'api-key':\n return {\n type: 'apiKey',\n in: 'header',\n name: 'X-API-Key',\n };\n case 'oauth2':\n return {\n type: 'oauth2',\n flows: {},\n };\n case 'oidc':\n case 'openidconnect':\n return {\n type: 'openIdConnect',\n openIdConnectUrl: '',\n };\n default:\n return {\n type: 'http',\n scheme: 'bearer',\n };\n }\n }\n\n private buildEndpointAuthMap(\n endpointInfos: EndpointInfo[],\n ): Record<string, string | null> {\n const authMap: Record<string, string | null> = {};\n\n for (const info of endpointInfos) {\n authMap[info.endpoint] = info.authorizerName;\n }\n\n return authMap;\n }\n\n private async generateSchemaInterfaces(\n endpointInfos: EndpointInfo[],\n ): Promise<string> {\n const interfaces: string[] = [];\n const generatedNames = new Set<string>();\n // Collect nested schemas with $defs (from .meta({ id: 'X' }))\n const collectedDefs = new Map<string, JsonSchema>();\n\n for (const info of endpointInfos) {\n const baseName = this.getSchemaBaseName(info);\n\n // Input body schema\n if (info.input?.body) {\n const name = await this.getSchemaName(\n info.input.body,\n `${baseName}Input`,\n );\n if (!generatedNames.has(name)) {\n const schema = await this.schemaToInterfaceWithDefs(\n info.input.body,\n name,\n collectedDefs,\n );\n if (schema) {\n interfaces.push(schema);\n generatedNames.add(name);\n }\n }\n }\n\n // Input params schema\n if (info.input?.params) {\n const name = await this.getSchemaName(\n info.input.params,\n `${baseName}Params`,\n );\n if (!generatedNames.has(name)) {\n const schema = await this.schemaToInterfaceWithDefs(\n info.input.params,\n name,\n collectedDefs,\n );\n if (schema) {\n interfaces.push(schema);\n generatedNames.add(name);\n }\n }\n }\n\n // Input query schema\n if (info.input?.query) {\n const name = await this.getSchemaName(\n info.input.query,\n `${baseName}Query`,\n );\n if (!generatedNames.has(name)) {\n const schema = await this.schemaToInterfaceWithDefs(\n info.input.query,\n name,\n collectedDefs,\n );\n if (schema) {\n interfaces.push(schema);\n generatedNames.add(name);\n }\n }\n }\n\n // Output schema\n if (info.output) {\n const name = await this.getSchemaName(info.output, `${baseName}Output`);\n if (!generatedNames.has(name)) {\n const schema = await this.schemaToInterfaceWithDefs(\n info.output,\n name,\n collectedDefs,\n );\n if (schema) {\n interfaces.push(schema);\n generatedNames.add(name);\n }\n }\n }\n }\n\n // Generate interfaces for collected $defs (nested schemas with .meta({ id: 'X' }))\n for (const [defName, defSchema] of collectedDefs) {\n if (!generatedNames.has(defName)) {\n const interfaceStr = this.jsonSchemaToInterface(defSchema, defName);\n interfaces.push(interfaceStr);\n generatedNames.add(defName);\n }\n }\n\n return interfaces.join('\\n\\n');\n }\n\n /**\n * Get the name for a schema, using metadata `id` if available,\n * otherwise falling back to the provided default name.\n */\n private async getSchemaName(\n schema: StandardSchemaV1,\n defaultName: string,\n ): Promise<string> {\n try {\n const metadata = await getSchemaMetadata(schema);\n if (metadata?.id) {\n return this.pascalCase(metadata.id);\n }\n } catch {\n // Ignore metadata extraction errors\n }\n return defaultName;\n }\n\n private getSchemaBaseName(info: EndpointInfo): string {\n if (info.operationId) {\n return this.pascalCase(info.operationId);\n }\n\n // Generate name from method + route\n const routeParts = info.route\n .replace(/[{}]/g, '')\n .split('/')\n .filter(Boolean)\n .map((part) => this.pascalCase(part));\n\n return `${this.pascalCase(info.method.toLowerCase())}${routeParts.join('')}`;\n }\n\n private pascalCase(str: string): string {\n return str\n .replace(/[-_](.)/g, (_, c) => c.toUpperCase())\n .replace(/^./, (c) => c.toUpperCase());\n }\n\n /**\n * Convert schema to interface while collecting $defs for nested schemas\n * with .meta({ id: 'X' }).\n */\n private async schemaToInterfaceWithDefs(\n schema: StandardSchemaV1,\n name: string,\n collectedDefs: Map<string, JsonSchema>,\n ): Promise<string | null> {\n try {\n // Get raw JSON schema with $defs intact (don't use convertStandardSchemaToJsonSchema\n // which strips $defs)\n const vendor = schema['~standard']?.vendor;\n if (!vendor || !(vendor in StandardSchemaJsonSchema)) {\n return null;\n }\n\n const toJsonSchema =\n StandardSchemaJsonSchema[\n vendor as keyof typeof StandardSchemaJsonSchema\n ];\n const jsonSchema = await toJsonSchema(schema);\n if (!jsonSchema) return null;\n\n // Extract $defs from the JSON schema (these come from .meta({ id: 'X' }))\n if (jsonSchema.$defs && typeof jsonSchema.$defs === 'object') {\n for (const [defName, defSchema] of Object.entries(jsonSchema.$defs)) {\n if (!collectedDefs.has(defName)) {\n // Remove the 'id' field from the schema as it's just metadata\n const { id, ...schemaWithoutId } = defSchema as JsonSchema & {\n id?: string;\n };\n collectedDefs.set(defName, schemaWithoutId as JsonSchema);\n }\n }\n }\n\n // Remove $defs from the schema before converting to interface\n const { $defs, ...schemaWithoutDefs } = jsonSchema;\n return this.jsonSchemaToInterface(schemaWithoutDefs, name);\n } catch {\n return null;\n }\n }\n\n private jsonSchemaToInterface(schema: JsonSchema, name: string): string {\n if (schema.type !== 'object' || !schema.properties) {\n // For non-object types, create a type alias\n const typeStr = this.jsonSchemaTypeToTs(schema);\n return `export type ${name} = ${typeStr};`;\n }\n\n const props: string[] = [];\n const required = new Set(schema.required || []);\n\n for (const [propName, propSchema] of Object.entries(schema.properties)) {\n const isRequired = required.has(propName);\n const typeStr = this.jsonSchemaTypeToTs(propSchema as JsonSchema);\n const optionalMark = isRequired ? '' : '?';\n props.push(` ${propName}${optionalMark}: ${typeStr};`);\n }\n\n return `export interface ${name} {\\n${props.join('\\n')}\\n}`;\n }\n\n private jsonSchemaTypeToTs(schema: JsonSchema): string {\n if (!schema) return 'unknown';\n\n if (schema.$ref) {\n // Extract name from $ref\n const refName = schema.$ref.split('/').pop() || 'unknown';\n return refName;\n }\n\n if (schema.anyOf) {\n return schema.anyOf\n .map((s: JsonSchema) => this.jsonSchemaTypeToTs(s))\n .join(' | ');\n }\n\n if (schema.oneOf) {\n return schema.oneOf\n .map((s: JsonSchema) => this.jsonSchemaTypeToTs(s))\n .join(' | ');\n }\n\n if (schema.allOf) {\n return schema.allOf\n .map((s: JsonSchema) => this.jsonSchemaTypeToTs(s))\n .join(' & ');\n }\n\n switch (schema.type) {\n case 'string':\n if (schema.enum) {\n return schema.enum.map((e: string) => `'${e}'`).join(' | ');\n }\n return 'string';\n case 'number':\n case 'integer':\n return 'number';\n case 'boolean':\n return 'boolean';\n case 'null':\n return 'null';\n case 'array':\n if (schema.items) {\n return `Array<${this.jsonSchemaTypeToTs(schema.items as JsonSchema)}>`;\n }\n return 'Array<unknown>';\n case 'object':\n if (schema.properties) {\n const props: string[] = [];\n const required = new Set(schema.required || []);\n for (const [propName, propSchema] of Object.entries(\n schema.properties,\n )) {\n const isRequired = required.has(propName);\n const typeStr = this.jsonSchemaTypeToTs(propSchema as JsonSchema);\n const optionalMark = isRequired ? '' : '?';\n props.push(`${propName}${optionalMark}: ${typeStr}`);\n }\n return `{ ${props.join('; ')} }`;\n }\n if (schema.additionalProperties) {\n const valueType = this.jsonSchemaTypeToTs(\n schema.additionalProperties as JsonSchema,\n );\n return `Record<string, ${valueType}>`;\n }\n return 'Record<string, unknown>';\n default:\n return 'unknown';\n }\n }\n\n private async generatePathsInterface(\n endpointInfos: EndpointInfo[],\n ): Promise<string> {\n const pathGroups = new Map<string, EndpointInfo[]>();\n\n // Group endpoints by route\n for (const info of endpointInfos) {\n const existing = pathGroups.get(info.route) || [];\n existing.push(info);\n pathGroups.set(info.route, existing);\n }\n\n const pathEntries: string[] = [];\n\n for (const [route, infos] of pathGroups) {\n const methodEntries: string[] = [];\n\n for (const info of infos) {\n const methodDef = await this.generateMethodDefinition(info);\n methodEntries.push(` ${info.method.toLowerCase()}: ${methodDef};`);\n }\n\n // Add path parameters if present\n const firstWithParams = infos.find((i) => i.input?.params);\n let paramsEntry = '';\n if (firstWithParams?.input?.params) {\n const paramsName = await this.getSchemaName(\n firstWithParams.input.params,\n `${this.getSchemaBaseName(firstWithParams)}Params`,\n );\n paramsEntry = `\\n parameters: {\\n path: ${paramsName};\\n };`;\n }\n\n pathEntries.push(\n ` '${route}': {${paramsEntry}\\n${methodEntries.join('\\n')}\\n };`,\n );\n }\n\n return `export interface paths {\\n${pathEntries.join('\\n')}\\n}`;\n }\n\n private async generateMethodDefinition(info: EndpointInfo): Promise<string> {\n const parts: string[] = [];\n const baseName = this.getSchemaBaseName(info);\n\n // Request body\n if (info.input?.body) {\n const bodyName = await this.getSchemaName(\n info.input.body,\n `${baseName}Input`,\n );\n parts.push(`requestBody: {\n content: {\n 'application/json': ${bodyName};\n };\n }`);\n }\n\n // Query parameters\n if (info.input?.query) {\n const queryName = await this.getSchemaName(\n info.input.query,\n `${baseName}Query`,\n );\n parts.push(`parameters: {\n query: ${queryName};\n }`);\n }\n\n // Responses\n const outputName = info.output\n ? await this.getSchemaName(info.output, `${baseName}Output`)\n : 'unknown';\n parts.push(`responses: {\n 200: {\n content: {\n 'application/json': ${outputName};\n };\n };\n }`);\n\n return `{\\n ${parts.join(';\\n ')};\\n }`;\n }\n\n private buildModule(params: {\n title: string;\n version: string;\n description?: string;\n securitySchemes: SecuritySchemeInfo[];\n endpointAuth: Record<string, string | null>;\n schemaInterfaces: string;\n pathsInterface: string;\n }): string {\n const {\n title,\n version,\n description,\n securitySchemes,\n endpointAuth,\n schemaInterfaces,\n pathsInterface,\n } = params;\n\n const securitySchemesObj = securitySchemes.reduce(\n (acc, s) => {\n acc[s.name] = s.scheme;\n return acc;\n },\n {} as Record<string, SecuritySchemeObject>,\n );\n\n const schemeNames = securitySchemes.map((s) => `'${s.name}'`).join(' | ');\n\n // Generate createApi only if there are security schemes\n const hasSecuritySchemes = schemeNames.length > 0;\n\n const createApiSection = hasSecuritySchemes\n ? `\n// ============================================================\n// API Client Factory\n// ============================================================\n\nimport {\n createAuthAwareFetcher,\n type AuthStrategy,\n} from '@geekmidas/client/auth-fetcher';\nimport { createEndpointHooks } from '@geekmidas/client/endpoint-hooks';\nimport type { QueryClient } from '@tanstack/react-query';\n\n/**\n * Options for creating the API client.\n */\nexport interface CreateApiOptions {\n /** Base URL for all API requests (required) */\n baseURL: string;\n /** Auth strategies for each security scheme used in this API */\n authStrategies: Record<SecuritySchemeId, AuthStrategy>;\n /** Optional React Query client instance */\n queryClient?: QueryClient;\n /** Optional request interceptor */\n onRequest?: (config: RequestInit) => RequestInit | Promise<RequestInit>;\n}\n\n/**\n * Create a type-safe API client with authentication and React Query hooks.\n *\n * @example\n * \\`\\`\\`typescript\n * const api = createApi({\n * baseURL: 'https://api.example.com',\n * authStrategies: {\n * jwt: { type: 'bearer', tokenProvider },\n * },\n * });\n *\n * // Imperative fetch\n * const user = await api('GET /users/{id}', { params: { id: '123' } });\n *\n * // React Query hooks\n * const { data } = api.useQuery('GET /users/{id}', { params: { id: '123' } });\n * const mutation = api.useMutation('POST /users');\n * \\`\\`\\`\n */\nexport function createApi(options: CreateApiOptions) {\n const fetcher = createAuthAwareFetcher<paths, typeof endpointAuth, typeof securitySchemes>({\n baseURL: options.baseURL,\n endpointAuth,\n securitySchemes,\n authStrategies: options.authStrategies,\n onRequest: options.onRequest,\n });\n\n const hooks = createEndpointHooks<paths>(fetcher, { queryClient: options.queryClient });\n\n return Object.assign(fetcher, hooks);\n}\n`\n : `\n// ============================================================\n// API Client Factory\n// ============================================================\n\nimport { TypedFetcher, type FetcherOptions } from '@geekmidas/client/fetcher';\nimport { createEndpointHooks } from '@geekmidas/client/endpoint-hooks';\nimport type { QueryClient } from '@tanstack/react-query';\n\n/**\n * Options for creating the API client.\n */\nexport interface CreateApiOptions extends Omit<FetcherOptions, 'baseURL'> {\n /** Base URL for all API requests (required) */\n baseURL: string;\n /** Optional React Query client instance */\n queryClient?: QueryClient;\n}\n\n/**\n * Create a type-safe API client with React Query hooks.\n *\n * @example\n * \\`\\`\\`typescript\n * const api = createApi({\n * baseURL: 'https://api.example.com',\n * });\n *\n * // Imperative fetch\n * const data = await api('GET /health');\n *\n * // React Query hooks\n * const { data } = api.useQuery('GET /health');\n * \\`\\`\\`\n */\nexport function createApi(options: CreateApiOptions) {\n const { queryClient, ...fetcherOptions } = options;\n const fetcher = new TypedFetcher<paths>(fetcherOptions);\n\n const hooks = createEndpointHooks<paths>(fetcher.request.bind(fetcher), { queryClient });\n\n return Object.assign(fetcher.request.bind(fetcher), hooks);\n}\n`;\n\n return `// Auto-generated by @geekmidas/cli - DO NOT EDIT\n// Generated: ${new Date().toISOString()}\n\n// ============================================================\n// Security Scheme Type\n// ============================================================\n\ninterface SecuritySchemeObject {\n type: 'apiKey' | 'http' | 'mutualTLS' | 'oauth2' | 'openIdConnect';\n description?: string;\n name?: string;\n in?: 'query' | 'header' | 'cookie';\n scheme?: string;\n bearerFormat?: string;\n flows?: Record<string, unknown>;\n openIdConnectUrl?: string;\n [key: string]: unknown;\n}\n\n// ============================================================\n// API Info\n// ============================================================\n\nexport const apiInfo = {\n title: '${title}',\n version: '${version}',${description ? `\\n description: '${description.replace(/'/g, \"\\\\'\")}',` : ''}\n} as const;\n\n// ============================================================\n// Security Schemes\n// ============================================================\n\n/**\n * Available security schemes for this API.\n * Maps authorizer names to OpenAPI security scheme definitions.\n */\nexport const securitySchemes = ${JSON.stringify(securitySchemesObj, null, 2).replace(/\"([a-zA-Z_$][a-zA-Z0-9_$]*)\":/g, '$1:')} as const satisfies Record<string, SecuritySchemeObject>;\n\nexport type SecuritySchemeId = ${schemeNames || 'never'};\n\n// ============================================================\n// Endpoint Authentication Map\n// ============================================================\n\n/**\n * Runtime map of endpoints to their required authentication scheme.\n * \\`null\\` indicates a public endpoint (no auth required).\n */\nexport const endpointAuth = ${JSON.stringify(endpointAuth, null, 2).replace(/\"([^\"]+)\":/g, \"'$1':\")} as const satisfies Record<string, SecuritySchemeId | null>;\n\nexport type EndpointString = keyof typeof endpointAuth;\n\nexport type AuthenticatedEndpoint = {\n [K in EndpointString]: typeof endpointAuth[K] extends null ? never : K;\n}[EndpointString];\n\nexport type PublicEndpoint = {\n [K in EndpointString]: typeof endpointAuth[K] extends null ? K : never;\n}[EndpointString];\n\n// ============================================================\n// Schema Definitions\n// ============================================================\n\n${schemaInterfaces}\n\n// ============================================================\n// OpenAPI Paths\n// ============================================================\n\n${pathsInterface}\n${createApiSection}\n`;\n }\n}\n","#!/usr/bin/env -S npx tsx\n\nimport { mkdir, writeFile } from 'node:fs/promises';\nimport { dirname, join } from 'node:path';\nimport { loadConfig } from './config.js';\nimport { EndpointGenerator } from './generators/EndpointGenerator.js';\nimport { OpenApiTsGenerator } from './generators/OpenApiTsGenerator.js';\nimport type { GkmConfig, OpenApiConfig } from './types.js';\n\ninterface OpenAPIOptions {\n cwd?: string;\n}\n\n/**\n * Fixed output path for generated OpenAPI client (not configurable)\n */\nexport const OPENAPI_OUTPUT_PATH = './.gkm/openapi.ts';\n\n/**\n * Resolve OpenAPI config from GkmConfig\n */\nexport function resolveOpenApiConfig(\n config: GkmConfig,\n): OpenApiConfig & { enabled: boolean } {\n if (config.openapi === false) {\n return { enabled: false };\n }\n\n if (config.openapi === true || config.openapi === undefined) {\n return {\n enabled: config.openapi === true,\n title: 'API Documentation',\n version: '1.0.0',\n description: 'Auto-generated API documentation from endpoints',\n };\n }\n\n return {\n enabled: config.openapi.enabled !== false,\n title: config.openapi.title || 'API Documentation',\n version: config.openapi.version || '1.0.0',\n description:\n config.openapi.description ||\n 'Auto-generated API documentation from endpoints',\n };\n}\n\n/**\n * Generate OpenAPI spec from endpoints\n * @returns Object with output path and endpoint count, or null if disabled\n */\nexport async function generateOpenApi(\n config: GkmConfig,\n options: { silent?: boolean } = {},\n): Promise<{ outputPath: string; endpointCount: number } | null> {\n const logger = options.silent ? { log: () => {} } : console;\n const openApiConfig = resolveOpenApiConfig(config);\n\n if (!openApiConfig.enabled) {\n return null;\n }\n\n const endpointGenerator = new EndpointGenerator();\n const loadedEndpoints = await endpointGenerator.load(config.routes);\n\n if (loadedEndpoints.length === 0) {\n logger.log('No valid endpoints found for OpenAPI generation');\n return null;\n }\n\n const endpoints = loadedEndpoints.map(({ construct }) => construct);\n const outputPath = join(process.cwd(), OPENAPI_OUTPUT_PATH);\n\n await mkdir(dirname(outputPath), { recursive: true });\n\n const tsGenerator = new OpenApiTsGenerator();\n const tsContent = await tsGenerator.generate(endpoints, {\n title: openApiConfig.title!,\n version: openApiConfig.version!,\n description: openApiConfig.description!,\n });\n\n await writeFile(outputPath, tsContent);\n logger.log(`📄 OpenAPI client generated: ${OPENAPI_OUTPUT_PATH}`);\n\n return { outputPath, endpointCount: loadedEndpoints.length };\n}\n\nexport async function openapiCommand(\n options: OpenAPIOptions = {},\n): Promise<void> {\n const logger = console;\n\n try {\n const config = await loadConfig(options.cwd);\n\n // Enable openapi if not configured\n if (!config.openapi) {\n config.openapi = { enabled: true };\n }\n\n const result = await generateOpenApi(config);\n\n if (result) {\n logger.log(`Found ${result.endpointCount} endpoints`);\n }\n } catch (error) {\n throw new Error(`OpenAPI generation failed: ${(error as Error).message}`);\n }\n}\n"],"mappings":";;;;;;;;;;;AAYA,IAAsB,qBAAtB,MAAwE;CAGtE,aAAa,MACXA,SACAC,WACAC,WACAC,UACAC,SACY;EACZ,MAAM,aAAa,MAAM,UAAU,KAAK,SAAS;AACjD,SAAO,UAAU,MAAM,SAAS,YAAY,WAAW,QAAQ;CAChE;CASD,MAAM,KACJD,UACA,MAAM,QAAQ,KAAK,EACe;EAClC,MAAM,SAAS;EAGf,MAAM,eAAe,MAAM,QAAQ,SAAS,GACxC,WACA,WACE,CAAC,QAAS,IACV,CAAE;EAGR,MAAM,QAAQ,kBAAG,OAAO,cAAc;GACpC;GACA,UAAU;EACX,EAAC;EAGF,MAAME,aAAsC,CAAE;AAE9C,aAAW,MAAM,KAAK,MACpB,KAAI;GACF,MAAM,OAAO,EAAE,UAAU;GACzB,MAAMC,WAAS,MAAM,OAAO;AAG5B,QAAK,MAAM,CAAC,KAAK,UAAU,IAAI,OAAO,QAAQA,SAAO,CACnD,KAAI,KAAK,YAAY,UAAU,CAC7B,YAAW,KAAK;IACd;IACA,MAAM,8BAAU,IAAI;IACpB;IACA,MAAM;KACJ,UAAU;KACV,UAAU,mBAAS,QAAQ,KAAK,EAAE,KAAK;IACxC;GACF,EAAC;EAGP,SAAQ,OAAO;AACd,UAAO,MAAM,iBAAiB,EAAE,IAAK,MAAgB,QAAQ;AAC7D,SAAM,IAAI,MACR;EAEH;AAGH,SAAO;CACR;AACF;;;;ACzED,IAAa,oBAAb,cAAuC,mBAkBrC;CACA,YACEC,OAgBA;AACA,SAAO,0CAAS,WAAW,MAAM;CAClC;CAED,MAAM,MACJC,SACAC,YAkBAC,WACAC,SACsB;EACtB,MAAM,WAAW,SAAS,YAAY;EACtC,MAAM,gBAAgB,SAAS,iBAAiB;EAChD,MAAM,SAAS;EACf,MAAMC,SAAsB,CAAE;AAE9B,MAAI,WAAW,WAAW,EACxB,QAAO;AAGT,MAAI,aAAa,UAAU;AAEzB,SAAM,KAAK,sBAAsB,WAAW,YAAY,QAAQ;GAChE,MAAM,UAAU,MAAM,KAAK,gBAAgB,WAAW,QAAQ;AAE9D,UAAO,KAAK;IACV,MAAM;IACN,QAAQ;IACR,SAAS,wBAAS,QAAQ,KAAK,EAAE,QAAQ;IACzC,YAAY;GACb,EAAC;AAEF,UAAO,KACJ,wBAAwB,WAAW,OAAO,YAAY,gBAAgB,uBAAuB,GAAG,EAClG;EACF,WAAU,aAAa,cAAc;GAEpC,MAAM,YAAY,oBAAK,WAAW,SAAS;AAC3C,SAAM,4BAAM,WAAW,EAAE,WAAW,KAAM,EAAC;AAG3C,QAAK,MAAM,EAAE,KAAK,WAAW,cAAM,IAAI,YAAY;IACjD,MAAM,cAAc,MAAM,KAAK,oBAC7B,WACAC,OAAK,UACL,KACA,oBACA,WACA,QACD;IAED,MAAMC,YAAuB;KAC3B,MAAM,UAAU;KAChB,QAAQ,UAAU;KAClB,SAAS,wBAAS,QAAQ,KAAK,EAAE,YAAY,CAAC,QAC5C,SACA,WACD;KACD,SAAS,UAAU;KACnB,YAAY,UAAU;KACtB,aAAa,MAAM,UAAU,gBAAgB;KAC7C,YAAY,UAAU,YAAY,QAAQ;IAC3C;AAED,WAAO,KAAK,UAAU;AACtB,WAAO,KACJ,wBAAwB,UAAU,OAAO,GAAG,UAAU,KAAK,EAC7D;GACF;EACF,MAEC,MAAK,MAAM,EAAE,KAAK,WAAW,cAAM,IAAI,YAAY;GACjD,MAAM,cAAc,MAAM,KAAK,oBAC7B,WACAD,OAAK,UACL,KACA,UACA,WACA,QACD;GAED,MAAMC,YAAuB;IAC3B,MAAM,UAAU;IAChB,QAAQ,UAAU;IAClB,SAAS,wBAAS,QAAQ,KAAK,EAAE,YAAY,CAAC,QAC5C,SACA,WACD;IACD,SAAS,UAAU;IACnB,YAAY,UAAU;IACtB,aAAa,MAAM,UAAU,gBAAgB;IAC7C,YAAY,UAAU,YAAY,QAAQ;GAC3C;AAED,UAAO,KAAK,UAAU;AACtB,UAAO,KACJ,wBAAwB,UAAU,OAAO,GAAG,UAAU,KAAK,EAC7D;EACF;AAGH,SAAO;CACR;CAED,MAAc,oBACZJ,WACAK,YACAC,YACAC,UACAC,WAgBAV,SACiB;EACjB,MAAM,mBAAmB,EAAE,WAAW;EACtC,MAAM,cAAc,oBAAK,WAAW,gBAAgB;EAEpD,MAAM,eAAe,wBAAS,uBAAQ,YAAY,EAAE,WAAW;EAC/D,MAAM,aAAa,aAAa,QAAQ,SAAS,MAAM;EAEvD,MAAM,wBAAwB,wBAC5B,uBAAQ,YAAY,EACpB,QAAQ,cACT;EAED,IAAIW;AAEJ,UAAQ,UAAR;GACE,KAAK;AACH,cAAU,KAAK,+BACb,YACA,YACA,uBACA,QAAQ,uBACT;AACD;GACF,KAAK;AACH,cAAU,KAAK,+BACb,YACA,YACA,uBACA,QAAQ,uBACT;AACD;GACF,KAAK;AACH,cAAU,KAAK,sBAAsB,YAAY,WAAW;AAC5D;GACF,QACE,OAAM,IAAI,OAAO,wBAAwB,SAAS;EACrD;AAED,QAAM,gCAAU,aAAa,QAAQ;AACrC,SAAO;CACR;CAED,MAAc,sBACZT,WACAU,WAkBAC,UACiB;EACjB,MAAM,oBAAoB;EAC1B,MAAM,gBAAgB,oBAAK,WAAW,kBAAkB;EAGxD,MAAM,gCAAgB,IAAI;AAE1B,OAAK,MAAM,EAAE,cAAM,KAAK,IAAI,WAAW;GACrC,MAAM,eAAe,wBAAS,uBAAQ,cAAc,EAAER,OAAK,SAAS;GACpE,MAAM,aAAa,aAAa,QAAQ,SAAS,MAAM;AAEvD,QAAK,cAAc,IAAI,WAAW,CAChC,eAAc,IAAI,YAAY,CAAE,EAAC;AAEnC,iBAAc,IAAI,WAAW,CAAE,KAAK,IAAI;EACzC;EAGD,MAAM,UAAU,MAAM,KAAK,cAAc,SAAS,CAAC,CAChD,IACC,CAAC,CAAC,YAAYS,UAAQ,MACnB,WAAW,UAAQ,KAAK,KAAK,CAAC,WAAW,WAAW,IACxD,CACA,KAAK,KAAK;EAEb,MAAM,iBAAiB,UAAU,IAAI,CAAC,EAAE,KAAK,KAAK,IAAI;EAEtD,MAAM,WAAW;;;;;;EAMnB,QAAQ;;;IAGN,eAAe,KAAK,QAAQ,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAsC7B,QAAM,gCAAU,eAAe,QAAQ;AAEvC,SAAO;CACR;CAED,MAAc,gBACZZ,WACAF,SACiB;EACjB,MAAM,cAAc;EACpB,MAAM,UAAU,oBAAK,WAAW,YAAY;EAE5C,MAAM,qBAAqB,wBAAS,uBAAQ,QAAQ,EAAE,QAAQ,WAAW;EAEzE,MAAM,wBAAwB,wBAC5B,uBAAQ,QAAQ,EAChB,QAAQ,cACT;EAGD,MAAM,mBAAmB,QAAQ,WAAW;EAC5C,MAAM,4BAA4B,QAAQ,WAAW;EACrD,MAAM,0BAA0B,QAAQ,WAAW;EAGnD,MAAM,gBAAgB,QAAQ,QAAQ;EACtC,MAAM,uBAAuB,QAAQ,QAAQ;EAG7C,IAAI,mBAAmB;AACvB,MAAI,iBACF,KAAI,uBAAuB;GACzB,MAAM,wBAAwB,wBAC5B,uBAAQ,QAAQ,EAChB,QAAQ,UAAW,cACpB;AACD,uBAAoB,SAAS,QAAQ,UAAW,uBAAuB,SAAS,sBAAsB;;EAEvG,MACC,qBAAoB;;EAMxB,IAAI,gBAAgB;AACpB,MAAI,cACF,KAAI,oBAAoB;GACtB,MAAM,qBAAqB,wBACzB,uBAAQ,QAAQ,EAChB,QAAQ,OAAQ,WACjB;AACD,oBAAiB,SAAS,QAAQ,OAAQ,oBAAoB,SAAS,mBAAmB;;EAE3F,MACC,kBAAiB;;EAKrB,MAAM,8BAA8B,6BAC/B;;;;;;mBAMU,QAAQ,UAAW,KAAK;;;;;;;;;;;;;;;;;;;;;;;;IAyBnC;EAGJ,IAAI,iBAAiB;AACrB,MAAI,iBACF,KAAI,sBAEF,mBAAkB;EACxB,4BAA4B;;;;;;mBAMX,QAAQ,UAAW,KAAK;;MAInC,mBAAkB;;+DAEqC,QAAQ,UAAW,WAAW;;;aAGhF,QAAQ,UAAW,KAAK;sBACf,KAAK,UAAU,QAAQ,UAAW,OAAO,CAAC;kBAC9C,QAAQ,UAAW,WAAW;;;EAG9C,4BAA4B;;;;;;mBAMX,QAAQ,UAAW,KAAK;;EAMvC,IAAI,cAAc;AAClB,MAAI,iBAAiB,mBACnB,gBAAe;;;mBAGF,QAAQ,OAAQ,KAAK;;EAIpC,MAAM,WAAW;;;;;;;;;;;SAWZ,QAAQ,uBAAuB,SAAS,sBAAsB;SAC9D,QAAQ,oBAAoB,SAAS,mBAAmB;EAC/D,iBAAiB;EACjB,cAAc;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EA8Cd,eAAe,EAAE,YAAY;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAoC3B,QAAM,gCAAU,SAAS,QAAQ;AAEjC,SAAO;CACR;CAED,AAAQ,+BACNe,YACAP,YACAQ,eACAC,wBACQ;AACR,UAAQ;WACD,WAAW,WAAW,WAAW;SACnC,uBAAuB,SAAS,cAAc;;4DAEK,WAAW;;;;CAIpE;CAED,AAAQ,+BACNF,YACAP,YACAQ,eACAC,wBACQ;AACR,UAAQ;WACD,WAAW,WAAW,WAAW;SACnC,uBAAuB,SAAS,cAAc;;4DAEK,WAAW;;;;CAIpE;CAED,AAAQ,sBACNF,YACAP,YACQ;AACR,UAAQ,WAAW,WAAW,WAAW,WAAW;;;yBAG/B,WAAW;;CAEjC;AACF;;;;;;;;;;;;ACvhBD,IAAa,qBAAb,MAAgC;CAC9B,MAAM,SACJU,WACAC,UAA4B,CAAE,GACb;EACjB,MAAM,EAAE,QAAQ,OAAO,UAAU,SAAS,aAAa,GAAG;EAG1D,MAAM,gBAAgB,MAAM,KAAK,qBAAqB,UAAU;EAGhE,MAAM,kBAAkB,KAAK,uBAAuB,cAAc;EAGlE,MAAM,eAAe,KAAK,qBAAqB,cAAc;EAG7D,MAAM,mBAAmB,MAAM,KAAK,yBAAyB,cAAc;EAG3E,MAAM,iBAAiB,MAAM,KAAK,uBAAuB,cAAc;AAGvE,SAAO,KAAK,YAAY;GACtB;GACA;GACA;GACA;GACA;GACA;GACA;EACD,EAAC;CACH;CAED,MAAc,qBACZD,WACyB;AACzB,SAAO,UAAU,IAAI,CAAC,OAAO;GAC3B,MAAM,QAAQ,GAAG,MAAM,QAAQ,WAAW,OAAO;GACjD,MAAM,SAAS,GAAG,OAAO,aAAa;GAItC,MAAM,iBAAiB,GAAG,YAAY;AAItC,UAAO;IACL,WAAW,EAAE,OAAO,GAAG,MAAM;IAC7B;IACA;IACA,gBAAgB,GAAG,YAAY,QAAQ;IACvC,gBAAgB,GAAG,YAAY,QAAQ;IACvC,gBAAgB,kBAAkB;IAClC,OAAO,GAAG;IACV,QAAQ,GAAG;IACX,aAAa,GAAG;IAChB,MAAM,GAAG;IACT,aAAa,GAAG;GACjB;EACF,EAAC;CACH;CAED,AAAQ,uBACNE,eACsB;EACtB,MAAM,0BAAU,IAAI;AAEpB,OAAK,MAAM,QAAQ,cACjB,KAAI,KAAK,mBAAmB,QAAQ,IAAI,KAAK,eAAe,EAAE;GAG5D,MAAM,SACJ,KAAK,mBACJ,KAAK,iBACF,KAAK,8BACH,KAAK,gBACL,KAAK,eACN,GACD;AAEN,OAAI,OACF,SAAQ,IAAI,KAAK,gBAAgB;IAC/B,MAAM,KAAK;IACX,MAAM,OAAO;IACb;GACD,EAAC;EAEL;AAGH,SAAO,MAAM,KAAK,QAAQ,QAAQ,CAAC;CACpC;CAED,AAAQ,8BACNC,MACAC,OACsB;AACtB,UAAQ,KAAK,aAAa,EAA1B;GACE,KAAK;GACL,KAAK,SACH,QAAO;IACL,MAAM;IACN,QAAQ;IACR,cAAc;GACf;GACH,KAAK;GACL,KAAK;GACL,KAAK,QACH,QAAO;IACL,MAAM;IACN,IAAI;IACJ,MAAM;IACN,gCAAgC;GACjC;GACH,KAAK;GACL,KAAK,UACH,QAAO;IACL,MAAM;IACN,IAAI;IACJ,MAAM;GACP;GACH,KAAK,SACH,QAAO;IACL,MAAM;IACN,OAAO,CAAE;GACV;GACH,KAAK;GACL,KAAK,gBACH,QAAO;IACL,MAAM;IACN,kBAAkB;GACnB;GACH,QACE,QAAO;IACL,MAAM;IACN,QAAQ;GACT;EACJ;CACF;CAED,AAAQ,qBACNF,eAC+B;EAC/B,MAAMG,UAAyC,CAAE;AAEjD,OAAK,MAAM,QAAQ,cACjB,SAAQ,KAAK,YAAY,KAAK;AAGhC,SAAO;CACR;CAED,MAAc,yBACZH,eACiB;EACjB,MAAMI,aAAuB,CAAE;EAC/B,MAAM,iCAAiB,IAAI;EAE3B,MAAM,gCAAgB,IAAI;AAE1B,OAAK,MAAM,QAAQ,eAAe;GAChC,MAAM,WAAW,KAAK,kBAAkB,KAAK;AAG7C,OAAI,KAAK,OAAO,MAAM;IACpB,MAAM,OAAO,MAAM,KAAK,cACtB,KAAK,MAAM,OACV,EAAE,SAAS,OACb;AACD,SAAK,eAAe,IAAI,KAAK,EAAE;KAC7B,MAAM,SAAS,MAAM,KAAK,0BACxB,KAAK,MAAM,MACX,MACA,cACD;AACD,SAAI,QAAQ;AACV,iBAAW,KAAK,OAAO;AACvB,qBAAe,IAAI,KAAK;KACzB;IACF;GACF;AAGD,OAAI,KAAK,OAAO,QAAQ;IACtB,MAAM,OAAO,MAAM,KAAK,cACtB,KAAK,MAAM,SACV,EAAE,SAAS,QACb;AACD,SAAK,eAAe,IAAI,KAAK,EAAE;KAC7B,MAAM,SAAS,MAAM,KAAK,0BACxB,KAAK,MAAM,QACX,MACA,cACD;AACD,SAAI,QAAQ;AACV,iBAAW,KAAK,OAAO;AACvB,qBAAe,IAAI,KAAK;KACzB;IACF;GACF;AAGD,OAAI,KAAK,OAAO,OAAO;IACrB,MAAM,OAAO,MAAM,KAAK,cACtB,KAAK,MAAM,QACV,EAAE,SAAS,OACb;AACD,SAAK,eAAe,IAAI,KAAK,EAAE;KAC7B,MAAM,SAAS,MAAM,KAAK,0BACxB,KAAK,MAAM,OACX,MACA,cACD;AACD,SAAI,QAAQ;AACV,iBAAW,KAAK,OAAO;AACvB,qBAAe,IAAI,KAAK;KACzB;IACF;GACF;AAGD,OAAI,KAAK,QAAQ;IACf,MAAM,OAAO,MAAM,KAAK,cAAc,KAAK,SAAS,EAAE,SAAS,QAAQ;AACvE,SAAK,eAAe,IAAI,KAAK,EAAE;KAC7B,MAAM,SAAS,MAAM,KAAK,0BACxB,KAAK,QACL,MACA,cACD;AACD,SAAI,QAAQ;AACV,iBAAW,KAAK,OAAO;AACvB,qBAAe,IAAI,KAAK;KACzB;IACF;GACF;EACF;AAGD,OAAK,MAAM,CAAC,SAAS,UAAU,IAAI,cACjC,MAAK,eAAe,IAAI,QAAQ,EAAE;GAChC,MAAM,eAAe,KAAK,sBAAsB,WAAW,QAAQ;AACnE,cAAW,KAAK,aAAa;AAC7B,kBAAe,IAAI,QAAQ;EAC5B;AAGH,SAAO,WAAW,KAAK,OAAO;CAC/B;;;;;CAMD,MAAc,cACZC,QACAC,aACiB;AACjB,MAAI;GACF,MAAM,WAAW,MAAM,qDAAkB,OAAO;AAChD,OAAI,UAAU,GACZ,QAAO,KAAK,WAAW,SAAS,GAAG;EAEtC,QAAO,CAEP;AACD,SAAO;CACR;CAED,AAAQ,kBAAkBC,MAA4B;AACpD,MAAI,KAAK,YACP,QAAO,KAAK,WAAW,KAAK,YAAY;EAI1C,MAAM,aAAa,KAAK,MACrB,QAAQ,SAAS,GAAG,CACpB,MAAM,IAAI,CACV,OAAO,QAAQ,CACf,IAAI,CAAC,SAAS,KAAK,WAAW,KAAK,CAAC;AAEvC,UAAQ,EAAE,KAAK,WAAW,KAAK,OAAO,aAAa,CAAC,CAAC,EAAE,WAAW,KAAK,GAAG,CAAC;CAC5E;CAED,AAAQ,WAAWC,KAAqB;AACtC,SAAO,IACJ,QAAQ,YAAY,CAAC,GAAG,MAAM,EAAE,aAAa,CAAC,CAC9C,QAAQ,MAAM,CAAC,MAAM,EAAE,aAAa,CAAC;CACzC;;;;;CAMD,MAAc,0BACZH,QACAI,MACAC,eACwB;AACxB,MAAI;GAGF,MAAM,SAAS,OAAO,cAAc;AACpC,QAAK,YAAY,UAAUC,wDACzB,QAAO;GAGT,MAAM,eACJA,uDACE;GAEJ,MAAM,aAAa,MAAM,aAAa,OAAO;AAC7C,QAAK,WAAY,QAAO;AAGxB,OAAI,WAAW,gBAAgB,WAAW,UAAU,UAClD;SAAK,MAAM,CAAC,SAAS,UAAU,IAAI,OAAO,QAAQ,WAAW,MAAM,CACjE,MAAK,cAAc,IAAI,QAAQ,EAAE;KAE/B,MAAM,EAAE,GAAI,GAAG,iBAAiB,GAAG;AAGnC,mBAAc,IAAI,SAAS,gBAA8B;IAC1D;GACF;GAIH,MAAM,EAAE,MAAO,GAAG,mBAAmB,GAAG;AACxC,UAAO,KAAK,sBAAsB,mBAAmB,KAAK;EAC3D,QAAO;AACN,UAAO;EACR;CACF;CAED,AAAQ,sBAAsBC,QAAoBH,MAAsB;AACtE,MAAI,OAAO,SAAS,aAAa,OAAO,YAAY;GAElD,MAAM,UAAU,KAAK,mBAAmB,OAAO;AAC/C,WAAQ,cAAc,KAAK,KAAK,QAAQ;EACzC;EAED,MAAMI,QAAkB,CAAE;EAC1B,MAAM,WAAW,IAAI,IAAI,OAAO,YAAY,CAAE;AAE9C,OAAK,MAAM,CAAC,UAAU,WAAW,IAAI,OAAO,QAAQ,OAAO,WAAW,EAAE;GACtE,MAAM,aAAa,SAAS,IAAI,SAAS;GACzC,MAAM,UAAU,KAAK,mBAAmB,WAAyB;GACjE,MAAM,eAAe,aAAa,KAAK;AACvC,SAAM,MAAM,IAAI,SAAS,EAAE,aAAa,IAAI,QAAQ,GAAG;EACxD;AAED,UAAQ,mBAAmB,KAAK,MAAM,MAAM,KAAK,KAAK,CAAC;CACxD;CAED,AAAQ,mBAAmBD,QAA4B;AACrD,OAAK,OAAQ,QAAO;AAEpB,MAAI,OAAO,MAAM;GAEf,MAAM,UAAU,OAAO,KAAK,MAAM,IAAI,CAAC,KAAK,IAAI;AAChD,UAAO;EACR;AAED,MAAI,OAAO,MACT,QAAO,OAAO,MACX,IAAI,CAACE,MAAkB,KAAK,mBAAmB,EAAE,CAAC,CAClD,KAAK,MAAM;AAGhB,MAAI,OAAO,MACT,QAAO,OAAO,MACX,IAAI,CAACA,MAAkB,KAAK,mBAAmB,EAAE,CAAC,CAClD,KAAK,MAAM;AAGhB,MAAI,OAAO,MACT,QAAO,OAAO,MACX,IAAI,CAACA,MAAkB,KAAK,mBAAmB,EAAE,CAAC,CAClD,KAAK,MAAM;AAGhB,UAAQ,OAAO,MAAf;GACE,KAAK;AACH,QAAI,OAAO,KACT,QAAO,OAAO,KAAK,IAAI,CAACC,OAAe,GAAG,EAAE,GAAG,CAAC,KAAK,MAAM;AAE7D,WAAO;GACT,KAAK;GACL,KAAK,UACH,QAAO;GACT,KAAK,UACH,QAAO;GACT,KAAK,OACH,QAAO;GACT,KAAK;AACH,QAAI,OAAO,MACT,SAAQ,QAAQ,KAAK,mBAAmB,OAAO,MAAoB,CAAC;AAEtE,WAAO;GACT,KAAK;AACH,QAAI,OAAO,YAAY;KACrB,MAAMF,QAAkB,CAAE;KAC1B,MAAM,WAAW,IAAI,IAAI,OAAO,YAAY,CAAE;AAC9C,UAAK,MAAM,CAAC,UAAU,WAAW,IAAI,OAAO,QAC1C,OAAO,WACR,EAAE;MACD,MAAM,aAAa,SAAS,IAAI,SAAS;MACzC,MAAM,UAAU,KAAK,mBAAmB,WAAyB;MACjE,MAAM,eAAe,aAAa,KAAK;AACvC,YAAM,MAAM,EAAE,SAAS,EAAE,aAAa,IAAI,QAAQ,EAAE;KACrD;AACD,aAAQ,IAAI,MAAM,KAAK,KAAK,CAAC;IAC9B;AACD,QAAI,OAAO,sBAAsB;KAC/B,MAAM,YAAY,KAAK,mBACrB,OAAO,qBACR;AACD,aAAQ,iBAAiB,UAAU;IACpC;AACD,WAAO;GACT,QACE,QAAO;EACV;CACF;CAED,MAAc,uBACZb,eACiB;EACjB,MAAM,6BAAa,IAAI;AAGvB,OAAK,MAAM,QAAQ,eAAe;GAChC,MAAM,WAAW,WAAW,IAAI,KAAK,MAAM,IAAI,CAAE;AACjD,YAAS,KAAK,KAAK;AACnB,cAAW,IAAI,KAAK,OAAO,SAAS;EACrC;EAED,MAAMgB,cAAwB,CAAE;AAEhC,OAAK,MAAM,CAAC,OAAO,MAAM,IAAI,YAAY;GACvC,MAAMC,gBAA0B,CAAE;AAElC,QAAK,MAAM,QAAQ,OAAO;IACxB,MAAM,YAAY,MAAM,KAAK,yBAAyB,KAAK;AAC3D,kBAAc,MAAM,MAAM,KAAK,OAAO,aAAa,CAAC,IAAI,UAAU,GAAG;GACtE;GAGD,MAAM,kBAAkB,MAAM,KAAK,CAAC,MAAM,EAAE,OAAO,OAAO;GAC1D,IAAI,cAAc;AAClB,OAAI,iBAAiB,OAAO,QAAQ;IAClC,MAAM,aAAa,MAAM,KAAK,cAC5B,gBAAgB,MAAM,SACrB,EAAE,KAAK,kBAAkB,gBAAgB,CAAC,QAC5C;AACD,mBAAe,mCAAmC,WAAW;GAC9D;AAED,eAAY,MACT,KAAK,MAAM,MAAM,YAAY,IAAI,cAAc,KAAK,KAAK,CAAC,QAC5D;EACF;AAED,UAAQ,4BAA4B,YAAY,KAAK,KAAK,CAAC;CAC5D;CAED,MAAc,yBAAyBV,MAAqC;EAC1E,MAAMW,QAAkB,CAAE;EAC1B,MAAM,WAAW,KAAK,kBAAkB,KAAK;AAG7C,MAAI,KAAK,OAAO,MAAM;GACpB,MAAM,WAAW,MAAM,KAAK,cAC1B,KAAK,MAAM,OACV,EAAE,SAAS,OACb;AACD,SAAM,MAAM;;8BAEY,SAAS;;OAEhC;EACF;AAGD,MAAI,KAAK,OAAO,OAAO;GACrB,MAAM,YAAY,MAAM,KAAK,cAC3B,KAAK,MAAM,QACV,EAAE,SAAS,OACb;AACD,SAAM,MAAM;eACH,UAAU;OAClB;EACF;EAGD,MAAM,aAAa,KAAK,SACpB,MAAM,KAAK,cAAc,KAAK,SAAS,EAAE,SAAS,QAAQ,GAC1D;AACJ,QAAM,MAAM;;;gCAGgB,WAAW;;;OAGpC;AAEH,UAAQ,WAAW,MAAM,KAAK,YAAY,CAAC;CAC5C;CAED,AAAQ,YAAYC,QAQT;EACT,MAAM,EACJ,OACA,SACA,aACA,iBACA,cACA,kBACA,gBACD,GAAG;EAEJ,MAAM,qBAAqB,gBAAgB,OACzC,CAAC,KAAK,MAAM;AACV,OAAI,EAAE,QAAQ,EAAE;AAChB,UAAO;EACR,GACD,CAAE,EACH;EAED,MAAM,cAAc,gBAAgB,IAAI,CAAC,OAAO,GAAG,EAAE,KAAK,GAAG,CAAC,KAAK,MAAM;EAGzE,MAAM,qBAAqB,YAAY,SAAS;EAEhD,MAAM,mBAAmB,sBACpB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;KA4DA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA6CL,UAAQ;gBACI,qBAAI,QAAO,aAAa,CAAC;;;;;;;;;;;;;;;;;;;;;;;YAuB7B,MAAM;cACJ,QAAQ,IAAI,eAAe,oBAAoB,YAAY,QAAQ,MAAM,MAAM,CAAC,MAAM,GAAG;;;;;;;;;;;iCAWtE,KAAK,UAAU,oBAAoB,MAAM,EAAE,CAAC,QAAQ,kCAAkC,MAAM,CAAC;;iCAE7F,eAAe,QAAQ;;;;;;;;;;8BAU1B,KAAK,UAAU,cAAc,MAAM,EAAE,CAAC,QAAQ,eAAe,QAAQ,CAAC;;;;;;;;;;;;;;;;EAgBlG,iBAAiB;;;;;;EAMjB,eAAe;EACf,iBAAiB;;CAEhB;AACF;;;;;;;AC7wBD,MAAa,sBAAsB;;;;AAKnC,SAAgB,qBACdC,QACsC;AACtC,KAAI,OAAO,YAAY,MACrB,QAAO,EAAE,SAAS,MAAO;AAG3B,KAAI,OAAO,YAAY,QAAQ,OAAO,mBACpC,QAAO;EACL,SAAS,OAAO,YAAY;EAC5B,OAAO;EACP,SAAS;EACT,aAAa;CACd;AAGH,QAAO;EACL,SAAS,OAAO,QAAQ,YAAY;EACpC,OAAO,OAAO,QAAQ,SAAS;EAC/B,SAAS,OAAO,QAAQ,WAAW;EACnC,aACE,OAAO,QAAQ,eACf;CACH;AACF;;;;;AAMD,eAAsB,gBACpBA,QACAC,UAAgC,CAAE,GAC6B;CAC/D,MAAM,SAAS,QAAQ,SAAS,EAAE,KAAK,MAAM,CAAE,EAAE,IAAG;CACpD,MAAM,gBAAgB,qBAAqB,OAAO;AAElD,MAAK,cAAc,QACjB,QAAO;CAGT,MAAM,oBAAoB,IAAI;CAC9B,MAAM,kBAAkB,MAAM,kBAAkB,KAAK,OAAO,OAAO;AAEnE,KAAI,gBAAgB,WAAW,GAAG;AAChC,SAAO,IAAI,kDAAkD;AAC7D,SAAO;CACR;CAED,MAAM,YAAY,gBAAgB,IAAI,CAAC,EAAE,WAAW,KAAK,UAAU;CACnE,MAAM,aAAa,oBAAK,QAAQ,KAAK,EAAE,oBAAoB;AAE3D,OAAM,4BAAM,uBAAQ,WAAW,EAAE,EAAE,WAAW,KAAM,EAAC;CAErD,MAAM,cAAc,IAAI;CACxB,MAAM,YAAY,MAAM,YAAY,SAAS,WAAW;EACtD,OAAO,cAAc;EACrB,SAAS,cAAc;EACvB,aAAa,cAAc;CAC5B,EAAC;AAEF,OAAM,gCAAU,YAAY,UAAU;AACtC,QAAO,KAAK,+BAA+B,oBAAoB,EAAE;AAEjE,QAAO;EAAE;EAAY,eAAe,gBAAgB;CAAQ;AAC7D;AAED,eAAsB,eACpBC,UAA0B,CAAE,GACb;CACf,MAAM,SAAS;AAEf,KAAI;EACF,MAAM,SAAS,MAAM,0BAAW,QAAQ,IAAI;AAG5C,OAAK,OAAO,QACV,QAAO,UAAU,EAAE,SAAS,KAAM;EAGpC,MAAM,SAAS,MAAM,gBAAgB,OAAO;AAE5C,MAAI,OACF,QAAO,KAAK,QAAQ,OAAO,cAAc,YAAY;CAExD,SAAQ,OAAO;AACd,QAAM,IAAI,OAAO,6BAA8B,MAAgB,QAAQ;CACxE;AACF"}