@t-req/core 0.1.0 → 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +158 -15
- package/dist/client.d.ts +0 -3
- package/dist/client.d.ts.map +1 -1
- package/dist/config/engine-options.d.ts +48 -0
- package/dist/config/engine-options.d.ts.map +1 -0
- package/dist/config/index.d.ts +7 -3
- package/dist/config/index.d.ts.map +1 -1
- package/dist/config/index.js +5 -2
- package/dist/config/index.js.map +10 -5
- package/dist/config/jsonc.d.ts +18 -0
- package/dist/config/jsonc.d.ts.map +1 -0
- package/dist/config/load.d.ts +13 -2
- package/dist/config/load.d.ts.map +1 -1
- package/dist/config/merge.d.ts +31 -5
- package/dist/config/merge.d.ts.map +1 -1
- package/dist/config/resolve.d.ts +59 -0
- package/dist/config/resolve.d.ts.map +1 -0
- package/dist/config/substitution.d.ts +25 -0
- package/dist/config/substitution.d.ts.map +1 -0
- package/dist/config/types.d.ts +60 -6
- package/dist/config/types.d.ts.map +1 -1
- package/dist/cookies/persistence.d.ts +48 -0
- package/dist/cookies/persistence.d.ts.map +1 -0
- package/dist/cookies/persistence.js +4 -0
- package/dist/cookies/persistence.js.map +10 -0
- package/dist/cookies.js +5 -5
- package/dist/cookies.js.map +2 -2
- package/dist/engine/engine.d.ts +0 -4
- package/dist/engine/engine.d.ts.map +1 -1
- package/dist/engine/index.js +4 -4
- package/dist/engine/index.js.map +4 -4
- package/dist/index.d.ts +1 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +4 -4
- package/dist/index.js.map +8 -6
- package/dist/interpolate.d.ts.map +1 -1
- package/dist/resolver/command.d.ts +19 -0
- package/dist/resolver/command.d.ts.map +1 -0
- package/dist/resolver/index.d.ts +2 -0
- package/dist/resolver/index.d.ts.map +1 -0
- package/dist/resolver/index.js +5 -0
- package/dist/resolver/index.js.map +10 -0
- package/dist/runtime/index.js +2 -2
- package/dist/runtime/index.js.map +2 -2
- package/dist/server-client.d.ts +50 -0
- package/dist/server-client.d.ts.map +1 -0
- package/dist/server-metadata.d.ts +13 -0
- package/dist/server-metadata.d.ts.map +1 -0
- package/dist/types.d.ts +41 -0
- package/dist/types.d.ts.map +1 -1
- package/package.json +13 -4
package/dist/index.js.map
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
|
-
"sources": ["../src/runtime/fetch-transport.ts", "../src/runtime/auto-transport.ts", "../src/runtime/node-io.ts", "../src/utils/optional.ts", "../src/execute.ts", "../src/file-loader.ts", "../src/form-data-builder.ts", "../src/interpolate.ts", "../src/parser.ts", "../src/engine/engine.ts", "../src/client.ts"],
|
|
3
|
+
"sources": ["../src/runtime/fetch-transport.ts", "../src/runtime/auto-transport.ts", "../src/runtime/node-io.ts", "../src/utils/optional.ts", "../src/execute.ts", "../src/file-loader.ts", "../src/form-data-builder.ts", "../src/interpolate.ts", "../src/parser.ts", "../src/engine/engine.ts", "../src/server-client.ts", "../src/client.ts", "../src/server-metadata.ts"],
|
|
4
4
|
"sourcesContent": [
|
|
5
5
|
"import type { Transport } from './types';\n\ntype FetchLike = (input: RequestInfo | URL, init?: RequestInit) => Promise<Response>;\n\nfunction unsupported(name: string): never {\n throw new Error(\n `${name} is not supported by this transport. Provide a transport that supports it.`\n );\n}\n\n/**\n * Create a minimal transport backed by a standard fetch implementation.\n * This transport is renderer-safe and works in Node 18+, Bun, and webviews.\n *\n * Note: proxy and validateSSL are not supported by baseline fetch.\n */\nexport function createFetchTransport(fetchImpl: FetchLike = fetch): Transport {\n return {\n capabilities: { proxy: false, validateSSL: false },\n async fetch(url, init, ctx) {\n if (ctx.proxy) unsupported('proxy');\n if (ctx.validateSSL === false) unsupported('validateSSL=false');\n return await fetchImpl(url, init);\n }\n };\n}\n",
|
|
6
6
|
"import { createFetchTransport } from './fetch-transport';\nimport type { Transport } from './types';\n\ntype BunFetchInit = RequestInit & { tls?: { rejectUnauthorized: boolean }; proxy?: string };\n\nfunction hasBun(): boolean {\n return typeof (globalThis as Record<string, unknown>)['Bun'] !== 'undefined';\n}\n\n/**\n * Auto-select a transport for the current runtime.\n *\n * - Bun: supports `proxy` and `validateSSL=false` via Bun fetch extensions.\n * - Others: falls back to baseline fetch transport (no proxy/validateSSL).\n */\nexport function createAutoTransport(): Transport {\n if (!hasBun()) {\n return createFetchTransport(fetch);\n }\n\n return {\n capabilities: { proxy: true, validateSSL: true },\n async fetch(url, init, ctx) {\n const bunInit: BunFetchInit = { ...(init ?? {}) };\n\n if (ctx.proxy !== undefined) {\n bunInit.proxy = ctx.proxy;\n }\n\n if (ctx.validateSSL === false) {\n bunInit.tls = { rejectUnauthorized: false };\n }\n\n return await fetch(url, bunInit);\n }\n };\n}\n",
|
|
@@ -9,12 +9,14 @@
|
|
|
9
9
|
"import { createAutoTransport } from './runtime/auto-transport';\nimport type { Transport, TransportContext } from './runtime/types';\nimport type { ExecuteOptions, ExecuteRequest } from './types';\nimport { setOptional } from './utils/optional';\n\n/**\n * Determines if body should be attached to the request.\n * GET and HEAD requests should not have a body per HTTP spec.\n */\nfunction shouldAttachBody(method: string, body: ExecuteRequest['body']): boolean {\n return body !== undefined && !['GET', 'HEAD'].includes(method.toUpperCase());\n}\n\n/**\n * Build standard fetch RequestInit from request and execution options.\n */\nfunction buildRequestInit(\n request: ExecuteRequest,\n opts: {\n followRedirects: boolean;\n signal: AbortSignal;\n }\n): RequestInit {\n const shouldIncludeBody =\n shouldAttachBody(request.method, request.body) && request.body !== undefined;\n\n return setOptional<RequestInit>({\n method: request.method,\n redirect: opts.followRedirects ? 'follow' : 'manual',\n signal: opts.signal\n })\n .ifDefined('headers', request.headers)\n .ifDefined('body', shouldIncludeBody ? request.body : undefined)\n .build();\n}\n\n/**\n * Create an AbortSignal for request execution.\n * Uses provided signal or creates an internal timeout-based signal.\n */\nfunction createExecutionSignal(opts: { provided?: AbortSignal; timeout: number }): {\n signal: AbortSignal;\n isInternalTimeout: boolean;\n cleanup: () => void;\n} {\n if (opts.provided) {\n return {\n signal: opts.provided,\n isInternalTimeout: false,\n cleanup: () => {}\n };\n }\n\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), opts.timeout);\n\n return {\n signal: controller.signal,\n isInternalTimeout: true,\n cleanup: () => clearTimeout(timeoutId)\n };\n}\n\n/**\n * Map execution errors to user-friendly messages.\n * Converts internal timeout AbortError to descriptive timeout error.\n */\nfunction mapExecuteError(\n error: unknown,\n ctx: { timeout: number; isInternalTimeout: boolean }\n): Error {\n if (error instanceof Error && error.name === 'AbortError') {\n if (ctx.isInternalTimeout) {\n return new Error(`Request timeout after ${ctx.timeout}ms`);\n }\n }\n return error instanceof Error ? error : new Error(String(error));\n}\n\n// ============================================================================\n// Main Execute Function\n// ============================================================================\n\n/**\n * Execute an HTTP request\n * Returns a native fetch Response\n */\nexport async function execute(\n request: ExecuteRequest,\n options: ExecuteOptions = {}\n): Promise<Response> {\n return await executeWithTransport(request, options, createAutoTransport());\n}\n\n/**\n * Execute an HTTP request using an explicit transport.\n * Useful for renderer-safe environments (e.g. Tauri) or custom proxy/TLS handling.\n */\nexport async function executeWithTransport(\n request: ExecuteRequest,\n options: ExecuteOptions,\n transport: Transport\n): Promise<Response> {\n const timeout = options.timeout ?? 30000;\n\n const executionSignalOpts = setOptional<{ timeout: number; provided?: AbortSignal }>({ timeout })\n .ifDefined('provided', options.signal)\n .build();\n\n const { signal, isInternalTimeout, cleanup } = createExecutionSignal(executionSignalOpts);\n\n const runOpts = setOptional<{\n followRedirects: boolean;\n signal: AbortSignal;\n }>({\n followRedirects: options.followRedirects ?? true,\n signal\n }).build();\n\n const requestInit = buildRequestInit(request, runOpts);\n\n try {\n const ctx = setOptional<TransportContext>({})\n .ifDefined('proxy', options.proxy)\n .ifDefined('validateSSL', options.validateSSL)\n .build();\n\n return await transport.fetch(request.url, requestInit, ctx);\n } catch (error) {\n throw mapExecuteError(error, { timeout, isInternalTimeout });\n } finally {\n cleanup();\n }\n}\n",
|
|
10
10
|
"import type { IO } from './runtime/types';\n\n// ============================================================================\n// Types\n// ============================================================================\n\n/**\n * Options for loading a file body.\n */\nexport interface FileLoaderOptions {\n /**\n * Base path for resolving relative file paths.\n * File paths are validated to not escape this directory.\n * @default process.cwd()\n */\n basePath?: string;\n\n /**\n * Optional IO adapter for reading files (Node/Bun/Tauri).\n * If not provided, Bun runtime fallback is used when available.\n */\n io?: IO;\n}\n\n/**\n * Result of loading a file body.\n */\nexport interface LoadedFile {\n /** File content as string (text) or ArrayBuffer (binary) */\n content: string | ArrayBuffer;\n /** Whether the file was loaded as binary */\n isBinary: boolean;\n /** Inferred MIME type based on file extension */\n mimeType: string;\n}\n\n// ============================================================================\n// MIME Type Detection\n// ============================================================================\n\n/**\n * Map of file extensions to MIME types.\n */\nconst EXTENSION_MIME_MAP: Record<string, string> = {\n // Text types\n '.json': 'application/json',\n '.xml': 'application/xml',\n '.txt': 'text/plain',\n '.html': 'text/html',\n '.htm': 'text/html',\n '.css': 'text/css',\n '.js': 'application/javascript',\n '.mjs': 'application/javascript',\n '.ts': 'application/typescript',\n '.csv': 'text/csv',\n '.md': 'text/markdown',\n '.yaml': 'text/yaml',\n '.yml': 'text/yaml',\n '.svg': 'image/svg+xml',\n\n // Binary image types\n '.jpg': 'image/jpeg',\n '.jpeg': 'image/jpeg',\n '.png': 'image/png',\n '.gif': 'image/gif',\n '.webp': 'image/webp',\n '.ico': 'image/x-icon',\n '.bmp': 'image/bmp',\n '.tiff': 'image/tiff',\n '.tif': 'image/tiff',\n\n // Binary document types\n '.pdf': 'application/pdf',\n '.doc': 'application/msword',\n '.docx': 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',\n '.xls': 'application/vnd.ms-excel',\n '.xlsx': 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',\n '.ppt': 'application/vnd.ms-powerpoint',\n '.pptx': 'application/vnd.openxmlformats-officedocument.presentationml.presentation',\n\n // Binary archive types\n '.zip': 'application/zip',\n '.gz': 'application/gzip',\n '.tar': 'application/x-tar',\n '.rar': 'application/vnd.rar',\n '.7z': 'application/x-7z-compressed',\n\n // Binary audio types\n '.mp3': 'audio/mpeg',\n '.wav': 'audio/wav',\n '.ogg': 'audio/ogg',\n '.m4a': 'audio/mp4',\n\n // Binary video types\n '.mp4': 'video/mp4',\n '.webm': 'video/webm',\n '.avi': 'video/x-msvideo',\n '.mov': 'video/quicktime',\n\n // Other binary types\n '.wasm': 'application/wasm',\n '.bin': 'application/octet-stream',\n '.exe': 'application/octet-stream',\n '.dll': 'application/octet-stream',\n '.so': 'application/octet-stream',\n '.dylib': 'application/octet-stream'\n};\n\n/**\n * Set of MIME types that should be treated as text (not binary).\n */\nconst TEXT_MIME_TYPES = new Set([\n 'application/json',\n 'application/xml',\n 'application/javascript',\n 'application/typescript',\n 'text/plain',\n 'text/html',\n 'text/css',\n 'text/csv',\n 'text/xml',\n 'text/markdown',\n 'text/yaml',\n 'image/svg+xml' // SVG is XML-based text\n]);\n\n/**\n * Infer MIME type from file path based on extension.\n *\n * @param filePath - Path to the file\n * @returns Inferred MIME type, or 'application/octet-stream' if unknown\n *\n * @example\n * ```typescript\n * inferMimeType('./data.json'); // 'application/json'\n * inferMimeType('./image.png'); // 'image/png'\n * inferMimeType('./unknown'); // 'application/octet-stream'\n * ```\n */\nexport function inferMimeType(filePath: string): string {\n const ext = extname(filePath).toLowerCase();\n return EXTENSION_MIME_MAP[ext] ?? 'application/octet-stream';\n}\n\n/**\n * Determine if a MIME type represents binary content.\n *\n * @param mimeType - The MIME type to check\n * @returns true if binary, false if text\n *\n * @example\n * ```typescript\n * isBinaryMimeType('application/json'); // false\n * isBinaryMimeType('image/png'); // true\n * isBinaryMimeType('application/octet-stream'); // true\n * ```\n */\nexport function isBinaryMimeType(mimeType: string): boolean {\n // text/* types are text\n if (mimeType.startsWith('text/')) {\n return false;\n }\n // Known text application types\n if (TEXT_MIME_TYPES.has(mimeType)) {\n return false;\n }\n // Everything else is binary\n return true;\n}\n\n// ============================================================================\n// Path Utilities (runtime-neutral)\n// ============================================================================\n\nfunction sepForPath(p: string): string {\n return p.includes('\\\\') ? '\\\\' : '/';\n}\n\nfunction splitParts(p: string): string[] {\n return p.split(/[\\\\/]+/).filter(Boolean);\n}\n\nfunction isAbsolutePath(p: string): boolean {\n if (p.startsWith('/') || p.startsWith('\\\\')) return true;\n // Windows drive letters: C:\\ or C:/\n if (/^[A-Za-z]:[\\\\/]/.test(p)) return true;\n // UNC paths: \\\\server\\share\n if (p.startsWith('\\\\\\\\')) return true;\n return false;\n}\n\nfunction normalizeJoin(sep: string, parts: string[]): string {\n const out: string[] = [];\n for (const part of parts) {\n if (!part || part === '.') continue;\n if (part === '..') {\n out.pop();\n continue;\n }\n out.push(part);\n }\n return (sep === '\\\\' ? '\\\\' : '/') + out.join(sep);\n}\n\nfunction resolvePath(basePath: string, relativePath: string): string {\n const sep = sepForPath(basePath);\n\n const baseIsDrive = /^[A-Za-z]:$/.test(basePath);\n const baseDriveMatch = basePath.match(/^([A-Za-z]:)[\\\\/]?/);\n const drive = baseDriveMatch?.[1];\n\n const baseParts = splitParts(basePath);\n const relParts = splitParts(relativePath);\n\n if (drive) {\n const normalized = normalizeJoin(sep, [...baseParts.slice(1), ...relParts]);\n return `${drive}${normalized}`;\n }\n\n if (baseIsDrive) {\n const normalized = normalizeJoin(sep, relParts);\n return `${basePath}${normalized}`;\n }\n\n return normalizeJoin(sep, [...baseParts, ...relParts]);\n}\n\nfunction normalizeBase(basePath: string): string {\n if (basePath === '') return '/';\n const sep = sepForPath(basePath);\n const driveMatch = basePath.match(/^([A-Za-z]:)[\\\\/]?/);\n const drive = driveMatch?.[1];\n const parts = splitParts(basePath);\n if (drive) {\n const normalized = normalizeJoin(sep, parts.slice(1));\n return `${drive}${normalized}`;\n }\n return normalizeJoin(sep, parts);\n}\n\nfunction extname(p: string): string {\n const base = basename(p);\n const idx = base.lastIndexOf('.');\n if (idx <= 0) return '';\n return base.slice(idx);\n}\n\nfunction basename(p: string): string {\n const parts = p.split(/[\\\\/]/);\n const last = parts[parts.length - 1];\n return last ?? '';\n}\n\n/**\n * Validate that a file path is safe and doesn't escape the base directory.\n *\n * @param filePath - The file path to validate\n * @param basePath - The base directory to constrain paths within\n * @returns The resolved absolute path if valid\n * @throws Error if path is absolute or escapes the base directory\n *\n * @example\n * ```typescript\n * validateFilePath('./data.json', '/app');\n * // Returns '/app/data.json'\n *\n * validateFilePath('../../../etc/passwd', '/app');\n * // Throws: 'Path escapes base directory: ../../../etc/passwd'\n *\n * validateFilePath('/etc/passwd', '/app');\n * // Throws: 'Absolute paths not allowed: /etc/passwd'\n * ```\n */\nexport function validateFilePath(filePath: string, basePath: string): string {\n // Reject absolute paths\n if (isAbsolutePath(filePath)) {\n throw new Error(`Absolute paths not allowed: ${filePath}`);\n }\n\n // Resolve and ensure within basePath\n const resolved = resolvePath(basePath, filePath);\n const normalizedBase = normalizeBase(basePath);\n const sep = sepForPath(normalizedBase);\n\n // Check that resolved path starts with base path\n // Need to handle both exact match and subdirectory cases\n if (resolved !== normalizedBase && !resolved.startsWith(normalizedBase + sep)) {\n throw new Error(`Path escapes base directory: ${filePath}`);\n }\n\n return resolved;\n}\n\n// ============================================================================\n// File Loading\n// ============================================================================\n\n/**\n * Load a file body for use in HTTP requests.\n *\n * Automatically determines whether to load as text or binary based on\n * the file's MIME type (inferred from extension).\n *\n * @param filePath - Relative path to the file\n * @param options - Options including basePath for path resolution\n * @returns Loaded file content, binary flag, and MIME type\n * @throws Error if path validation fails or file doesn't exist\n *\n * @example\n * ```typescript\n * // Load JSON (text)\n * const jsonFile = await loadFileBody('./data.json', { basePath: '/app' });\n * // { content: '{\"key\": \"value\"}', isBinary: false, mimeType: 'application/json' }\n *\n * // Load image (binary)\n * const imageFile = await loadFileBody('./logo.png', { basePath: '/app' });\n * // { content: ArrayBuffer, isBinary: true, mimeType: 'image/png' }\n * ```\n */\nexport async function loadFileBody(\n filePath: string,\n options: FileLoaderOptions = {}\n): Promise<LoadedFile> {\n const basePath =\n options.basePath ??\n options.io?.cwd() ??\n (globalThis as unknown as { process?: { cwd?: () => string } }).process?.cwd?.() ??\n '.';\n\n // Validate path security\n const resolvedPath = validateFilePath(filePath, basePath);\n\n // Determine MIME type and binary mode\n const mimeType = inferMimeType(filePath);\n const isBinary = isBinaryMimeType(mimeType);\n\n let content: string | ArrayBuffer;\n\n if (options.io) {\n const exists = await options.io.exists(resolvedPath);\n if (!exists) {\n throw new Error(`File not found: ${filePath}`);\n }\n content = isBinary\n ? await options.io.readBinary(resolvedPath)\n : await options.io.readText(resolvedPath);\n } else if (typeof (globalThis as Record<string, unknown>)['Bun'] !== 'undefined') {\n type BunFile = {\n exists: () => Promise<boolean>;\n text: () => Promise<string>;\n arrayBuffer: () => Promise<ArrayBuffer>;\n };\n type BunGlobal = { file: (p: string) => BunFile };\n\n const bun = (globalThis as unknown as { Bun: BunGlobal }).Bun;\n const file = bun.file(resolvedPath);\n const exists = await file.exists();\n if (!exists) {\n throw new Error(`File not found: ${filePath}`);\n }\n content = isBinary ? await file.arrayBuffer() : await file.text();\n } else {\n throw new Error(\n 'No IO adapter provided. Provide `options.io` (Node/Bun/Tauri) to load file bodies.'\n );\n }\n\n return {\n content,\n isBinary,\n mimeType\n };\n}\n",
|
|
11
11
|
"import { loadFileBody } from './file-loader';\nimport type { IO } from './runtime/types';\nimport type { FormField } from './types';\n\n// ============================================================================\n// Types\n// ============================================================================\n\n/**\n * Options for building form data.\n */\nexport interface BuildFormDataOptions {\n /**\n * Base path for resolving relative file paths.\n * @default process.cwd()\n */\n basePath?: string;\n\n /**\n * Optional IO adapter for reading files and path operations.\n */\n io?: IO;\n}\n\n// ============================================================================\n// Utilities\n// ============================================================================\n\n/**\n * Check if any field in the array is a file upload.\n *\n * @param fields - Array of form fields\n * @returns true if any field is a file upload\n */\nexport function hasFileFields(fields: FormField[]): boolean {\n return fields.some((field) => field.isFile);\n}\n\n// ============================================================================\n// Form Data Building\n// ============================================================================\n\n/**\n * Build FormData object from form fields.\n * Used for multipart/form-data requests (file uploads).\n *\n * @param fields - Array of form fields\n * @param options - Options including basePath for file resolution\n * @returns FormData ready for use with fetch\n *\n * @example\n * ```typescript\n * const fields: FormField[] = [\n * { name: 'title', value: 'Report', isFile: false },\n * { name: 'document', value: '', isFile: true, path: './report.pdf' },\n * ];\n *\n * const formData = await buildFormData(fields, { basePath: '/app' });\n * ```\n */\nexport async function buildFormData(\n fields: FormField[],\n options: BuildFormDataOptions = {}\n): Promise<FormData> {\n const form = new FormData();\n const basePath =\n options.basePath ??\n options.io?.cwd() ??\n (globalThis as unknown as { process?: { cwd?: () => string } }).process?.cwd?.() ??\n '.';\n\n for (const field of fields) {\n if (field.isFile && field.path) {\n // Load file content\n const loadedFile = await loadFileBody(\n field.path,\n options.io ? { basePath, io: options.io } : { basePath }\n );\n\n // Determine filename: custom filename if provided, otherwise basename of path\n const filename =\n field.filename ?? options.io?.path.basename(field.path) ?? basename(field.path);\n\n // Create Blob with appropriate MIME type\n const blob = new Blob([loadedFile.content], { type: loadedFile.mimeType });\n\n // Append as file\n form.append(field.name, blob, filename);\n } else {\n // Append as text field\n form.append(field.name, field.value);\n }\n }\n\n return form;\n}\n\nfunction basename(p: string): string {\n const parts = p.split(/[\\\\/]/);\n const last = parts[parts.length - 1];\n return last ?? '';\n}\n\n/**\n * Build URLSearchParams from form fields.\n * Used for application/x-www-form-urlencoded requests.\n *\n * Note: File fields should not be present - use buildFormData for files.\n *\n * @param fields - Array of form fields (should not contain file fields)\n * @returns URLSearchParams ready for use with fetch\n *\n * @example\n * ```typescript\n * const fields: FormField[] = [\n * { name: 'username', value: 'john', isFile: false },\n * { name: 'password', value: 'secret', isFile: false },\n * ];\n *\n * const params = buildUrlEncoded(fields);\n * // username=john&password=secret\n * ```\n */\nexport function buildUrlEncoded(fields: FormField[]): URLSearchParams {\n const params = new URLSearchParams();\n\n for (const field of fields) {\n // Skip file fields - they can't be URL encoded\n if (field.isFile) {\n continue;\n }\n params.append(field.name, field.value);\n }\n\n return params;\n}\n",
|
|
12
|
-
"import type { InterpolateOptions, Interpolator } from './types';\n\n
|
|
12
|
+
"import type { InterpolateOptions, Interpolator } from './types';\n\ntype TemplatePart =\n | { type: 'text'; value: string }\n | { type: 'expr'; expression: string; raw: string };\n\nfunction isWord(s: string): boolean {\n return /^[A-Za-z0-9_]+$/.test(s);\n}\n\nfunction splitTemplate(input: string): TemplatePart[] {\n const parts: TemplatePart[] = [];\n let i = 0;\n let textStart = 0;\n\n while (i < input.length) {\n const c = input[i];\n const n = input[i + 1];\n\n if (c === '{' && n === '{') {\n if (i > textStart) {\n parts.push({ type: 'text', value: input.slice(textStart, i) });\n }\n\n const start = i;\n i += 2; // consume {{\n\n let depth = 1;\n let expr = '';\n\n while (i < input.length) {\n const cc = input[i];\n const nn = input[i + 1];\n\n if (cc === '{' && nn === '{') {\n depth++;\n expr += '{{';\n i += 2;\n continue;\n }\n\n if (cc === '}' && nn === '}') {\n depth--;\n if (depth === 0) {\n i += 2; // consume final }}\n break;\n }\n expr += '}}';\n i += 2;\n continue;\n }\n\n expr += cc;\n i++;\n }\n\n if (depth !== 0) {\n throw new Error('Unterminated interpolation: missing \"}}\"');\n }\n\n const raw = input.slice(start, i);\n parts.push({ type: 'expr', expression: expr, raw });\n textStart = i;\n continue;\n }\n\n i++;\n }\n\n if (textStart < input.length) {\n parts.push({ type: 'text', value: input.slice(textStart) });\n }\n\n return parts;\n}\n\n/**\n * Get a nested value from an object using dot notation\n */\nfunction getNestedValue(obj: Record<string, unknown>, path: string): unknown {\n const parts = path.split('.');\n let current: unknown = obj;\n\n for (const part of parts) {\n if (current === null || current === undefined) {\n return undefined;\n }\n if (typeof current !== 'object') {\n return undefined;\n }\n current = (current as Record<string, unknown>)[part];\n }\n\n return current;\n}\n\nfunction parseResolverCall(expression: string): { resolverKey: string; argText: string } | null {\n const trimmed = expression.trim();\n if (!trimmed.startsWith('$')) return null;\n if (!trimmed.endsWith(')')) return null;\n\n const openIdx = trimmed.indexOf('(');\n if (openIdx === -1) return null;\n\n const name = trimmed.slice(1, openIdx).trim();\n if (!name || !isWord(name)) return null;\n\n const argText = trimmed.slice(openIdx + 1, -1);\n return { resolverKey: `$${name}`, argText };\n}\n\nfunction interpolateVariablesOnly(\n str: string,\n variables: Record<string, unknown>,\n undefinedBehavior: NonNullable<InterpolateOptions['undefinedBehavior']>\n): string {\n const parts = splitTemplate(str);\n if (parts.length === 1 && parts[0]?.type === 'text') return str;\n\n let out = '';\n for (const part of parts) {\n if (part.type === 'text') {\n out += part.value;\n continue;\n }\n\n const expr = part.expression.trim();\n if (parseResolverCall(expr)) {\n throw new Error(`Resolver calls are not allowed inside resolver args: ${part.raw}`);\n }\n\n const value = getNestedValue(variables, expr);\n if (value === undefined) {\n if (undefinedBehavior === 'throw') {\n throw new Error(`Undefined variable: ${expr}`);\n }\n out += undefinedBehavior === 'keep' ? part.raw : '';\n continue;\n }\n\n out += String(value);\n }\n\n return out;\n}\n\nfunction parseResolverArgsFromText(\n argText: string,\n variables: Record<string, unknown>,\n undefinedBehavior: NonNullable<InterpolateOptions['undefinedBehavior']>\n): string[] {\n const interpolated = interpolateVariablesOnly(argText, variables, undefinedBehavior).trim();\n if (!interpolated) return [];\n\n try {\n const parsed = JSON.parse(interpolated) as unknown;\n if (Array.isArray(parsed)) {\n return parsed.map((v) => String(v));\n }\n } catch {\n // Fallback below\n }\n\n return [interpolated];\n}\n\n/**\n * Interpolate variables into a string synchronously\n */\nfunction interpolateString(\n str: string,\n variables: Record<string, unknown>,\n options: InterpolateOptions = {}\n): string {\n const { resolvers = {}, undefinedBehavior = 'throw' } = options;\n\n const parts = splitTemplate(str);\n if (parts.length === 1 && parts[0]?.type === 'text') return str;\n\n let out = '';\n for (const part of parts) {\n if (part.type === 'text') {\n out += part.value;\n continue;\n }\n\n const parsed = parseResolverCall(part.expression);\n if (parsed) {\n const resolver = resolvers[parsed.resolverKey];\n if (resolver) {\n const args = parseResolverArgsFromText(parsed.argText, variables, undefinedBehavior);\n const result = resolver(...args);\n if (result instanceof Promise) {\n throw new Error(\n `Resolver ${parsed.resolverKey} returned a Promise. Use createInterpolator() for async resolvers.`\n );\n }\n out += String(result);\n continue;\n }\n\n if (undefinedBehavior === 'throw') {\n throw new Error(`Unknown resolver: ${parsed.resolverKey}`);\n }\n out += undefinedBehavior === 'keep' ? part.raw : '';\n continue;\n }\n\n const expr = part.expression.trim();\n const value = getNestedValue(variables, expr);\n if (value === undefined) {\n if (undefinedBehavior === 'throw') {\n throw new Error(`Undefined variable: ${expr}`);\n }\n out += undefinedBehavior === 'keep' ? part.raw : '';\n continue;\n }\n\n out += String(value);\n }\n\n return out;\n}\n\n/**\n * Interpolate variables into a string asynchronously\n */\nasync function interpolateStringAsync(\n str: string,\n variables: Record<string, unknown>,\n options: InterpolateOptions = {}\n): Promise<string> {\n const { resolvers = {}, undefinedBehavior = 'throw' } = options;\n const parts = splitTemplate(str);\n if (parts.length === 1 && parts[0]?.type === 'text') return str;\n\n let out = '';\n for (const part of parts) {\n if (part.type === 'text') {\n out += part.value;\n continue;\n }\n\n const parsed = parseResolverCall(part.expression);\n if (parsed) {\n const resolver = resolvers[parsed.resolverKey];\n if (resolver) {\n const args = parseResolverArgsFromText(parsed.argText, variables, undefinedBehavior);\n const result = await resolver(...args);\n out += String(result);\n continue;\n }\n\n if (undefinedBehavior === 'throw') {\n throw new Error(`Unknown resolver: ${parsed.resolverKey}`);\n }\n out += undefinedBehavior === 'keep' ? part.raw : '';\n continue;\n }\n\n const expr = part.expression.trim();\n const value = getNestedValue(variables, expr);\n if (value === undefined) {\n if (undefinedBehavior === 'throw') {\n throw new Error(`Undefined variable: ${expr}`);\n }\n out += undefinedBehavior === 'keep' ? part.raw : '';\n continue;\n }\n\n out += String(value);\n }\n\n return out;\n}\n\n/**\n * Deep clone and interpolate an object\n */\nfunction interpolateValue<T>(\n value: T,\n variables: Record<string, unknown>,\n options: InterpolateOptions\n): T {\n if (typeof value === 'string') {\n return interpolateString(value, variables, options) as T;\n }\n\n if (Array.isArray(value)) {\n return value.map((item) => interpolateValue(item, variables, options)) as T;\n }\n\n if (value !== null && typeof value === 'object') {\n const result: Record<string, unknown> = {};\n for (const [key, val] of Object.entries(value as Record<string, unknown>)) {\n result[key] = interpolateValue(val, variables, options);\n }\n return result as T;\n }\n\n return value;\n}\n\n/**\n * Deep clone and interpolate an object asynchronously\n */\nasync function interpolateValueAsync<T>(\n value: T,\n variables: Record<string, unknown>,\n options: InterpolateOptions\n): Promise<T> {\n if (typeof value === 'string') {\n return (await interpolateStringAsync(value, variables, options)) as T;\n }\n\n if (Array.isArray(value)) {\n return (await Promise.all(\n value.map((item) => interpolateValueAsync(item, variables, options))\n )) as T;\n }\n\n if (value !== null && typeof value === 'object') {\n const result: Record<string, unknown> = {};\n const entries = Object.entries(value as Record<string, unknown>);\n const resolvedEntries = await Promise.all(\n entries.map(async ([key, val]) => [key, await interpolateValueAsync(val, variables, options)])\n );\n for (const [key, val] of resolvedEntries) {\n result[key as string] = val;\n }\n return result as T;\n }\n\n return value;\n}\n\n/**\n * Interpolate variables into a target object or string\n * For async resolvers, use createInterpolator() instead\n */\nexport function interpolate<T>(\n target: T,\n variables: Record<string, unknown>,\n options: InterpolateOptions = {}\n): T {\n return interpolateValue(target, variables, options);\n}\n\n/**\n * Create a reusable interpolator with async resolver support\n */\nexport function createInterpolator(options: InterpolateOptions = {}): Interpolator {\n return {\n async interpolate<T>(target: T, variables: Record<string, unknown>): Promise<T> {\n return interpolateValueAsync(target, variables, options);\n }\n };\n}\n",
|
|
13
13
|
"import type { IO } from './runtime/types';\nimport type { FileReference, FormField, ParsedRequest } from './types';\nimport { setOptional } from './utils/optional';\n\n// Pattern to match form field lines: name = value or name=value\nconst FORM_LINE_PATTERN = /^([^=]+?)\\s*=\\s*(.*)$/;\n\n/**\n * Check if body content matches form data syntax.\n * All non-empty lines must match the `name = value` pattern.\n *\n * This detects our friendly form syntax (multiline, spaces around =):\n * username = john\n * password = secret\n *\n * NOT traditional URL-encoded format (single line, & separators):\n * username=john&password=secret\n */\nfunction isFormBody(body: string, contentType?: string): boolean {\n // Explicit non-form Content-Type → not form\n if (\n contentType &&\n !contentType.includes('form-data') &&\n !contentType.includes('x-www-form-urlencoded')\n ) {\n return false;\n }\n\n // Empty body is not form\n const lines = body.split(/\\r?\\n/).filter((l) => l.trim());\n if (lines.length === 0) return false;\n\n // Single line with & is traditional URL-encoded format, not our form syntax\n if (lines.length === 1 && lines[0]?.includes('&')) {\n return false;\n }\n\n // All lines must match form pattern\n return lines.every((line) => FORM_LINE_PATTERN.test(line));\n}\n\n/**\n * Parse form body into FormField array.\n */\nfunction parseFormBody(body: string): FormField[] {\n const fields: FormField[] = [];\n const lines = body.split(/\\r?\\n/).filter((l) => l.trim());\n\n for (const line of lines) {\n const match = line.match(FORM_LINE_PATTERN);\n if (!match) continue;\n\n const nameRaw = match[1];\n const valueRaw = match[2];\n if (!nameRaw) continue;\n\n const name = nameRaw.trim();\n let value = (valueRaw ?? '').trim();\n\n // File reference: @./ or @{{ for variable paths\n const isFile = value.startsWith('@./') || value.startsWith('@{{');\n\n if (isFile) {\n // Remove @ prefix\n value = value.slice(1);\n\n // Check for custom filename: @./path | filename\n let filePath = value;\n let filename: string | undefined;\n\n const pipeIndex = value.indexOf(' | ');\n if (pipeIndex !== -1) {\n filePath = value.slice(0, pipeIndex).trim();\n filename = value.slice(pipeIndex + 3).trim();\n }\n\n fields.push(\n setOptional<FormField>({\n name,\n value: '',\n isFile: true,\n path: filePath\n })\n .ifDefined('filename', filename)\n .build()\n );\n } else {\n fields.push({ name, value, isFile: false });\n }\n }\n\n return fields;\n}\n\n/**\n * Parse .http file content into structured request objects\n */\nexport function parse(content: string): ParsedRequest[] {\n const requests: ParsedRequest[] = [];\n\n // Split by request separators (###)\n // The regex captures the optional name after ###\n // Handle both LF and CRLF line endings\n const lines = content.split(/\\r?\\n/);\n const blocks: Array<{ name?: string; lines: string[] }> = [];\n let currentBlock: { name?: string; lines: string[] } = { lines: [] };\n\n for (const line of lines) {\n const separatorMatch = line.match(/^###\\s*(.*)$/);\n if (separatorMatch) {\n // Save current block if it has content\n if (currentBlock.lines.length > 0 || blocks.length === 0) {\n blocks.push(currentBlock);\n }\n // Start new block with optional name from separator\n const blockName = separatorMatch[1]?.trim();\n currentBlock = setOptional<{ name?: string; lines: string[] }>({ lines: [] })\n .ifDefined('name', blockName || undefined)\n .build();\n } else {\n currentBlock.lines.push(line);\n }\n }\n\n // Don't forget the last block\n blocks.push(currentBlock);\n\n // Parse each block\n for (const block of blocks) {\n if (block.lines.length === 0 && !block.name) {\n continue;\n }\n\n const blockContent = block.lines.join('\\n');\n const request = parseRequestBlock(blockContent, block.name);\n if (request) {\n requests.push(request);\n }\n }\n\n return requests;\n}\n\n/**\n * Parse a single request block\n */\nfunction parseRequestBlock(block: string, defaultName?: string): ParsedRequest | null {\n const lines = block.split(/\\r?\\n/);\n const meta: Record<string, string> = {};\n let name = defaultName;\n let method = '';\n let url = '';\n const headers: Record<string, string> = {};\n let body: string | undefined;\n let bodyFile: FileReference | undefined;\n let formData: FormField[] | undefined;\n let inBody = false;\n const bodyLines: string[] = [];\n let requestLineFound = false;\n\n for (let i = 0; i < lines.length; i++) {\n const line = lines[i];\n if (line === undefined) continue;\n const trimmedLine = line.trim();\n\n // Skip empty lines before request line\n if (!requestLineFound && trimmedLine === '') {\n continue;\n }\n\n // Handle comments and directives\n if (!requestLineFound && (trimmedLine.startsWith('#') || trimmedLine.startsWith('//'))) {\n const commentContent = trimmedLine.replace(/^(#|\\/\\/)\\s*/, '');\n\n // Check for @directive\n const directiveMatch = commentContent.match(/^@(\\w+)\\s*(.*)?$/);\n if (directiveMatch) {\n const [, directive, value] = directiveMatch;\n if (directive === 'name') {\n name = value?.trim() || name;\n } else if (directive) {\n meta[directive] = value?.trim() || '';\n }\n }\n continue;\n }\n\n // Parse request line (METHOD URL HTTP/VERSION)\n if (!requestLineFound) {\n const requestLineMatch = trimmedLine.match(\n /^(GET|POST|PUT|DELETE|PATCH|HEAD|OPTIONS|TRACE|CONNECT)\\s+(\\S+)(\\s+HTTP\\/[\\d.]+)?$/i\n );\n if (requestLineMatch) {\n const methodMatch = requestLineMatch[1];\n const urlMatch = requestLineMatch[2];\n if (!methodMatch || !urlMatch) continue;\n method = methodMatch.toUpperCase();\n url = urlMatch;\n requestLineFound = true;\n continue;\n }\n\n // Also support just METHOD URL without HTTP version\n const simpleRequestMatch = trimmedLine.match(\n /^(GET|POST|PUT|DELETE|PATCH|HEAD|OPTIONS|TRACE|CONNECT)\\s+(.+)$/i\n );\n if (simpleRequestMatch) {\n const methodMatch = simpleRequestMatch[1];\n const urlMatch = simpleRequestMatch[2];\n if (!methodMatch || !urlMatch) continue;\n method = methodMatch.toUpperCase();\n url = urlMatch.trim();\n requestLineFound = true;\n continue;\n }\n\n // Skip non-matching lines before request\n continue;\n }\n\n // After request line - handle headers and body\n if (inBody) {\n bodyLines.push(line);\n } else if (trimmedLine === '') {\n // Empty line marks start of body\n inBody = true;\n } else {\n // Parse header\n const headerMatch = line.match(/^([^:]+):\\s*(.*)$/);\n if (headerMatch) {\n const headerName = headerMatch[1];\n if (!headerName) continue;\n const headerValue = headerMatch[2] ?? '';\n headers[headerName.trim()] = headerValue.trim();\n }\n }\n }\n\n // No valid request found\n if (!method || !url) {\n return null;\n }\n\n // Process body\n if (bodyLines.length > 0) {\n // Remove trailing empty lines\n while (bodyLines.length > 0 && bodyLines[bodyLines.length - 1]?.trim() === '') {\n bodyLines.pop();\n }\n\n // Check for file reference syntax: < ./path\n if (bodyLines.length === 1) {\n const fileRefMatch = bodyLines[0]?.trim().match(/^<\\s+(.+)$/);\n if (fileRefMatch) {\n const filePath = fileRefMatch[1];\n if (filePath) {\n bodyFile = { path: filePath };\n }\n } else if (bodyLines[0]?.trim()) {\n body = bodyLines[0];\n }\n } else if (bodyLines.length > 0) {\n body = bodyLines.join('\\n');\n }\n }\n\n // Check if body matches form data syntax\n // Get Content-Type header (case-insensitive)\n const contentType = headers['Content-Type'] || headers['content-type'];\n\n if (body && isFormBody(body, contentType)) {\n formData = parseFormBody(body);\n body = undefined; // Clear body since we're using formData\n }\n\n return setOptional<ParsedRequest>({\n method,\n url,\n headers,\n raw: block,\n meta\n })\n .ifDefined('name', name)\n .ifDefined('body', body)\n .ifDefined('bodyFile', bodyFile)\n .ifDefined('formData', formData)\n .build();\n}\n\n/**\n * Parse .http file from filesystem\n */\nexport async function parseFile(path: string): Promise<ParsedRequest[]> {\n return await parseFileWithIO(path);\n}\n\n/**\n * Parse .http file from filesystem using an IO adapter.\n * If `io` is omitted, Bun runtime fallback is used when available.\n */\nexport async function parseFileWithIO(path: string, io?: IO): Promise<ParsedRequest[]> {\n if (io) {\n const content = await io.readText(path);\n return parse(content);\n }\n\n if (typeof (globalThis as Record<string, unknown>)['Bun'] !== 'undefined') {\n type BunFile = { text: () => Promise<string> };\n type BunGlobal = { file: (p: string) => BunFile };\n\n const bun = (globalThis as unknown as { Bun: BunGlobal }).Bun;\n const file = bun.file(path);\n const content = await file.text();\n return parse(content);\n }\n\n throw new Error('No IO adapter provided. Use parseFileWithIO(path, io) in this runtime.');\n}\n",
|
|
14
|
-
"import { executeWithTransport } from '../execute';\nimport { loadFileBody } from '../file-loader';\nimport { buildFormData, buildUrlEncoded, hasFileFields } from '../form-data-builder';\nimport { createInterpolator } from '../interpolate';\nimport { parse, parseFileWithIO } from '../parser';\nimport { createAutoTransport } from '../runtime/auto-transport';\nimport type { CookieStore, EngineEvent, EventSink, IO, Transport } from '../runtime/types';\nimport type { ExecuteOptions, ExecuteRequest, FormField, Resolver } from '../types';\nimport { setOptional } from '../utils/optional';\n\nexport type EngineConfig = {\n transport?: Transport;\n io?: IO;\n cookieStore?: CookieStore;\n resolvers?: Record<string, Resolver>;\n onEvent?: EventSink;\n headerDefaults?: Record<string, string>;\n};\n\nexport type EngineRunOptions = {\n variables?: Record<string, unknown>;\n timeoutMs?: number;\n signal?: AbortSignal;\n followRedirects?: boolean;\n validateSSL?: boolean;\n proxy?: string;\n\n /**\n * Base path for resolving file references when running from string content.\n * When running from file, this is derived from the .http file directory.\n */\n basePath?: string;\n};\n\nexport type Engine = {\n parseString: (httpText: string) => ReturnType<typeof parse>;\n runString: (httpText: string, options?: EngineRunOptions) => Promise<Response>;\n runFile: (path: string, options?: EngineRunOptions) => Promise<Response>;\n};\n\nfunction emit(onEvent: EventSink | undefined, event: EngineEvent): void {\n onEvent?.(event);\n}\n\nfunction firstOrThrow<T>(arr: T[], ctx: string): T {\n const first = arr[0];\n if (!first) {\n throw new Error(ctx);\n }\n return first;\n}\n\nfunction dirnameFromPath(p: string): string {\n const idx = Math.max(p.lastIndexOf('/'), p.lastIndexOf('\\\\'));\n if (idx === -1) return '.';\n return idx === 0 ? p.slice(0, 1) : p.slice(0, idx);\n}\n\nfunction isAbsolutePath(p: string): boolean {\n if (p.startsWith('/') || p.startsWith('\\\\')) return true;\n if (/^[A-Za-z]:[\\\\/]/.test(p)) return true;\n if (p.startsWith('\\\\\\\\')) return true;\n return false;\n}\n\nfunction joinWithSep(a: string, b: string): string {\n if (!a) return b;\n if (!b) return a;\n const sep = a.includes('\\\\') ? '\\\\' : '/';\n const aTrim = a.endsWith('/') || a.endsWith('\\\\') ? a.slice(0, -1) : a;\n const bTrim = b.startsWith('/') || b.startsWith('\\\\') ? b.slice(1) : b;\n return `${aTrim}${sep}${bTrim}`;\n}\n\nfunction getFileBasePath(filePath: string, io?: IO): string {\n if (io) {\n return io.path.dirname(io.path.resolve(filePath));\n }\n\n const cwd =\n (globalThis as unknown as { process?: { cwd?: () => string } }).process?.cwd?.() ?? '.';\n const absolute = isAbsolutePath(filePath) ? filePath : joinWithSep(cwd, filePath);\n return dirnameFromPath(absolute);\n}\n\nfunction withCookieHeader(\n headers: Record<string, string>,\n cookie: string | undefined\n): Record<string, string> {\n if (!cookie) return headers;\n const existing = headers['Cookie'] || headers['cookie'] || '';\n return {\n ...headers,\n Cookie: existing ? `${existing}; ${cookie}` : cookie\n };\n}\n\nexport function createEngine(config: EngineConfig = {}): Engine {\n const transport = config.transport ?? createAutoTransport();\n const io = config.io;\n const cookieStore = config.cookieStore;\n const onEvent = config.onEvent;\n const headerDefaults = config.headerDefaults;\n\n const interpolator = createInterpolator({ resolvers: config.resolvers ?? {} });\n\n async function compileFromString(\n httpText: string,\n options: EngineRunOptions\n ): Promise<{\n executeRequest: ExecuteRequest;\n baseUrl: string;\n }> {\n emit(onEvent, { type: 'parseStarted', source: 'string' });\n const requests = parse(httpText);\n emit(onEvent, { type: 'parseFinished', source: 'string', requestCount: requests.length });\n\n const request = firstOrThrow(requests, 'No valid requests found in provided content.');\n\n emit(onEvent, { type: 'interpolateStarted' });\n const mergedVars = { ...(options.variables ?? {}) };\n const interpolated = await interpolator.interpolate(request, mergedVars);\n emit(onEvent, { type: 'interpolateFinished' });\n\n emit(onEvent, { type: 'compileStarted' });\n const basePath =\n options.basePath ??\n io?.cwd() ??\n (globalThis as unknown as { process?: { cwd?: () => string } }).process?.cwd?.() ??\n '.';\n\n const { executeRequest } = await compileExecuteRequest(\n interpolated,\n setOptional<{ basePath: string; io?: IO; headerDefaults?: Record<string, string> }>({\n basePath\n })\n .ifDefined('io', io)\n .ifDefined('headerDefaults', headerDefaults)\n .build()\n );\n emit(onEvent, { type: 'compileFinished' });\n\n return { executeRequest, baseUrl: interpolated.url };\n }\n\n async function compileFromFile(\n path: string,\n options: EngineRunOptions\n ): Promise<{\n executeRequest: ExecuteRequest;\n baseUrl: string;\n }> {\n emit(onEvent, { type: 'parseStarted', source: 'file' });\n const requests = await parseFileWithIO(path, io);\n emit(onEvent, { type: 'parseFinished', source: 'file', requestCount: requests.length });\n\n const request = firstOrThrow(requests, `No valid requests found in file: ${path}`);\n\n emit(onEvent, { type: 'interpolateStarted' });\n const mergedVars = { ...(options.variables ?? {}) };\n const interpolated = await interpolator.interpolate(request, mergedVars);\n emit(onEvent, { type: 'interpolateFinished' });\n\n emit(onEvent, { type: 'compileStarted' });\n const basePath = getFileBasePath(path, io);\n const { executeRequest } = await compileExecuteRequest(\n interpolated,\n setOptional<{ basePath: string; io?: IO; headerDefaults?: Record<string, string> }>({\n basePath\n })\n .ifDefined('io', io)\n .ifDefined('headerDefaults', headerDefaults)\n .build()\n );\n emit(onEvent, { type: 'compileFinished' });\n\n return { executeRequest, baseUrl: interpolated.url };\n }\n\n async function runCompiled(\n executeRequest: ExecuteRequest,\n urlForCookies: string,\n options: EngineRunOptions\n ): Promise<Response> {\n const headers = executeRequest.headers ?? {};\n\n const cookieHeader = cookieStore ? await cookieStore.getCookieHeader(urlForCookies) : undefined;\n const headersWithCookies = cookieStore ? withCookieHeader(headers, cookieHeader) : headers;\n\n const requestWithCookies: ExecuteRequest = {\n ...executeRequest,\n headers: headersWithCookies\n };\n\n emit(onEvent, { type: 'fetchStarted', method: requestWithCookies.method, url: urlForCookies });\n\n try {\n const execOptions = setOptional<ExecuteOptions>({})\n .ifDefined('timeout', options.timeoutMs)\n .ifDefined('signal', options.signal)\n .ifDefined('followRedirects', options.followRedirects)\n .ifDefined('validateSSL', options.validateSSL)\n .ifDefined('proxy', options.proxy)\n .build();\n\n const response = await executeWithTransport(requestWithCookies, execOptions, transport);\n\n emit(onEvent, {\n type: 'fetchFinished',\n method: requestWithCookies.method,\n url: urlForCookies,\n status: response.status\n });\n\n if (cookieStore) {\n await cookieStore.setFromResponse(urlForCookies, response);\n }\n\n return response;\n } catch (e) {\n emit(onEvent, {\n type: 'error',\n stage: 'fetch',\n message: e instanceof Error ? e.message : String(e)\n });\n throw e;\n }\n }\n\n return {\n parseString: parse,\n async runString(httpText, options = {}) {\n const { executeRequest, baseUrl } = await compileFromString(httpText, options);\n return await runCompiled(executeRequest, baseUrl, options);\n },\n async runFile(path, options = {}) {\n const { executeRequest, baseUrl } = await compileFromFile(path, options);\n return await runCompiled(executeRequest, baseUrl, options);\n }\n };\n}\n\nasync function compileExecuteRequest(\n interpolated: {\n method: string;\n url: string;\n headers: Record<string, string>;\n body?: string;\n bodyFile?: { path: string };\n formData?: FormField[];\n },\n ctx: { basePath: string; io?: IO; headerDefaults?: Record<string, string> }\n): Promise<{ executeRequest: ExecuteRequest }> {\n const headers: Record<string, string> = {\n ...(ctx.headerDefaults ?? {}),\n ...(interpolated.headers ?? {})\n };\n\n let body: ExecuteRequest['body'] = interpolated.body;\n\n if (interpolated.bodyFile) {\n const loadedFile = await loadFileBody(\n interpolated.bodyFile.path,\n setOptional<{ basePath: string; io?: IO }>({ basePath: ctx.basePath })\n .ifDefined('io', ctx.io)\n .build()\n );\n\n body = loadedFile.content;\n\n if (!headers['Content-Type'] && !headers['content-type']) {\n headers['Content-Type'] = loadedFile.mimeType;\n }\n } else if (interpolated.formData && interpolated.formData.length > 0) {\n const hasFiles = hasFileFields(interpolated.formData);\n\n if (hasFiles) {\n body = await buildFormData(\n interpolated.formData,\n setOptional<{ basePath: string; io?: IO }>({ basePath: ctx.basePath })\n .ifDefined('io', ctx.io)\n .build()\n );\n delete headers['Content-Type'];\n delete headers['content-type'];\n } else {\n body = buildUrlEncoded(interpolated.formData);\n if (!headers['Content-Type'] && !headers['content-type']) {\n headers['Content-Type'] = 'application/x-www-form-urlencoded';\n }\n }\n }\n\n const executeRequest: ExecuteRequest = {\n method: interpolated.method,\n url: interpolated.url,\n headers,\n ...(body !== undefined ? { body } : {})\n };\n\n return { executeRequest };\n}\n",
|
|
15
|
-
"import type { CookieJar } from 'tough-cookie';\nimport type { EngineConfig, EngineRunOptions } from './engine/engine';\nimport { createEngine } from './engine/engine';\nimport type { CookieStore } from './runtime/types';\nimport type { Client, ClientConfig, RunOptions } from './types';\nimport { setOptional } from './utils/optional';\n\n/**\n * Create a high-level HTTP client\n */\nexport function createClient(config: ClientConfig = {}): Client {\n let variables: Record<string, unknown> = { ...config.variables };\n const cookieJar = config.cookieJar;\n const defaultTimeout = config.timeout ?? 30000;\n const defaults = config.defaults || {};\n\n const cookieStore = cookieJar ? cookieJarToStore(cookieJar) : undefined;\n\n const engine = createEngine(\n setOptional<EngineConfig>({} as EngineConfig)\n .ifDefined('io', config.io)\n .ifDefined('transport', config.transport)\n .ifDefined('cookieStore', cookieStore)\n .ifDefined('resolvers', config.resolvers)\n .ifDefined('onEvent', config.onEvent)\n .ifDefined('headerDefaults', defaults.headers)\n .build()\n );\n\n function mergedVars(runVars?: Record<string, unknown>): Record<string, unknown> {\n return { ...variables, ...runVars };\n }\n\n return {\n async run(filePath: string, options: RunOptions = {}): Promise<Response> {\n return await engine.runFile(\n filePath,\n setOptional<EngineRunOptions>({\n variables: mergedVars(options.variables),\n timeoutMs: options.timeout ?? defaultTimeout\n })\n .ifDefined('signal', options.signal)\n .ifDefined('followRedirects', defaults.followRedirects)\n .ifDefined('validateSSL', defaults.validateSSL)\n .ifDefined('proxy', defaults.proxy)\n .build()\n );\n },\n\n async runString(content: string, options: RunOptions = {}): Promise<Response> {\n return await engine.runString(\n content,\n setOptional<EngineRunOptions>({\n variables: mergedVars(options.variables),\n timeoutMs: options.timeout ?? defaultTimeout\n })\n .ifDefined('signal', options.signal)\n .ifDefined('followRedirects', defaults.followRedirects)\n .ifDefined('validateSSL', defaults.validateSSL)\n .ifDefined('proxy', defaults.proxy)\n .ifDefined('basePath', options.basePath)\n .build()\n );\n },\n\n setVariables(vars: Record<string, unknown>): void {\n variables = { ...variables, ...vars };\n },\n\n setVariable(key: string, value: unknown): void {\n variables[key] = value;\n },\n\n getVariables(): Record<string, unknown> {\n return { ...variables };\n }\n };\n}\n\nfunction cookieJarToStore(cookieJar: CookieJar): CookieStore {\n return {\n getCookieHeader(url: string) {\n try {\n const v = cookieJar.getCookieStringSync(url);\n return v || undefined;\n } catch {\n return undefined;\n }\n },\n setFromResponse(url: string, response: Response) {\n const setCookieHeaders = getSetCookieHeaders(response);\n for (const header of setCookieHeaders) {\n cookieJar.setCookieSync(header, url, { ignoreError: true });\n }\n }\n };\n}\n\nfunction getSetCookieHeaders(response: Response): string[] {\n const anyHeaders = response.headers as unknown as { getSetCookie?: () => string[] };\n const fromGetSetCookie = anyHeaders.getSetCookie?.();\n if (Array.isArray(fromGetSetCookie) && fromGetSetCookie.length > 0) return fromGetSetCookie;\n\n const single = response.headers.get('set-cookie');\n return single ? [single] : [];\n}\n"
|
|
14
|
+
"import { executeWithTransport } from '../execute';\nimport { loadFileBody } from '../file-loader';\nimport { buildFormData, buildUrlEncoded, hasFileFields } from '../form-data-builder';\nimport { createInterpolator } from '../interpolate';\nimport { parse, parseFileWithIO } from '../parser';\nimport { createAutoTransport } from '../runtime/auto-transport';\nimport type { CookieStore, EngineEvent, EventSink, IO, Transport } from '../runtime/types';\nimport type { ExecuteOptions, ExecuteRequest, FormField, Resolver } from '../types';\nimport { setOptional } from '../utils/optional';\n\nexport type EngineConfig = {\n transport?: Transport;\n io?: IO;\n cookieStore?: CookieStore;\n resolvers?: Record<string, Resolver>;\n onEvent?: EventSink;\n headerDefaults?: Record<string, string>;\n};\n\nexport type EngineRunOptions = {\n variables?: Record<string, unknown>;\n timeoutMs?: number;\n signal?: AbortSignal;\n followRedirects?: boolean;\n validateSSL?: boolean;\n proxy?: string;\n basePath?: string;\n};\n\nexport type Engine = {\n parseString: (httpText: string) => ReturnType<typeof parse>;\n runString: (httpText: string, options?: EngineRunOptions) => Promise<Response>;\n runFile: (path: string, options?: EngineRunOptions) => Promise<Response>;\n};\n\nfunction emit(onEvent: EventSink | undefined, event: EngineEvent): void {\n onEvent?.(event);\n}\n\nfunction firstOrThrow<T>(arr: T[], ctx: string): T {\n const first = arr[0];\n if (!first) {\n throw new Error(ctx);\n }\n return first;\n}\n\nfunction dirnameFromPath(p: string): string {\n const idx = Math.max(p.lastIndexOf('/'), p.lastIndexOf('\\\\'));\n if (idx === -1) return '.';\n return idx === 0 ? p.slice(0, 1) : p.slice(0, idx);\n}\n\nfunction isAbsolutePath(p: string): boolean {\n if (p.startsWith('/') || p.startsWith('\\\\')) return true;\n if (/^[A-Za-z]:[\\\\/]/.test(p)) return true;\n if (p.startsWith('\\\\\\\\')) return true;\n return false;\n}\n\nfunction joinWithSep(a: string, b: string): string {\n if (!a) return b;\n if (!b) return a;\n const sep = a.includes('\\\\') ? '\\\\' : '/';\n const aTrim = a.endsWith('/') || a.endsWith('\\\\') ? a.slice(0, -1) : a;\n const bTrim = b.startsWith('/') || b.startsWith('\\\\') ? b.slice(1) : b;\n return `${aTrim}${sep}${bTrim}`;\n}\n\nfunction getFileBasePath(filePath: string, io?: IO): string {\n if (io) {\n return io.path.dirname(io.path.resolve(filePath));\n }\n\n const cwd =\n (globalThis as unknown as { process?: { cwd?: () => string } }).process?.cwd?.() ?? '.';\n const absolute = isAbsolutePath(filePath) ? filePath : joinWithSep(cwd, filePath);\n return dirnameFromPath(absolute);\n}\n\nfunction withCookieHeader(\n headers: Record<string, string>,\n cookie: string | undefined\n): Record<string, string> {\n if (!cookie) return headers;\n const existing = headers['Cookie'] || headers['cookie'] || '';\n return {\n ...headers,\n Cookie: existing ? `${existing}; ${cookie}` : cookie\n };\n}\n\nexport function createEngine(config: EngineConfig = {}): Engine {\n const transport = config.transport ?? createAutoTransport();\n const io = config.io;\n const cookieStore = config.cookieStore;\n const onEvent = config.onEvent;\n const headerDefaults = config.headerDefaults;\n\n const interpolator = createInterpolator({ resolvers: config.resolvers ?? {} });\n\n async function compileFromString(\n httpText: string,\n options: EngineRunOptions\n ): Promise<{\n executeRequest: ExecuteRequest;\n baseUrl: string;\n }> {\n emit(onEvent, { type: 'parseStarted', source: 'string' });\n const requests = parse(httpText);\n emit(onEvent, { type: 'parseFinished', source: 'string', requestCount: requests.length });\n\n const request = firstOrThrow(requests, 'No valid requests found in provided content.');\n\n emit(onEvent, { type: 'interpolateStarted' });\n const mergedVars = { ...(options.variables ?? {}) };\n const interpolated = await interpolator.interpolate(request, mergedVars);\n emit(onEvent, { type: 'interpolateFinished' });\n\n emit(onEvent, { type: 'compileStarted' });\n const basePath =\n options.basePath ??\n io?.cwd() ??\n (globalThis as unknown as { process?: { cwd?: () => string } }).process?.cwd?.() ??\n '.';\n\n const { executeRequest } = await compileExecuteRequest(\n interpolated,\n setOptional<{ basePath: string; io?: IO; headerDefaults?: Record<string, string> }>({\n basePath\n })\n .ifDefined('io', io)\n .ifDefined('headerDefaults', headerDefaults)\n .build()\n );\n emit(onEvent, { type: 'compileFinished' });\n\n return { executeRequest, baseUrl: interpolated.url };\n }\n\n async function compileFromFile(\n path: string,\n options: EngineRunOptions\n ): Promise<{\n executeRequest: ExecuteRequest;\n baseUrl: string;\n }> {\n emit(onEvent, { type: 'parseStarted', source: 'file' });\n const requests = await parseFileWithIO(path, io);\n emit(onEvent, { type: 'parseFinished', source: 'file', requestCount: requests.length });\n\n const request = firstOrThrow(requests, `No valid requests found in file: ${path}`);\n\n emit(onEvent, { type: 'interpolateStarted' });\n const mergedVars = { ...(options.variables ?? {}) };\n const interpolated = await interpolator.interpolate(request, mergedVars);\n emit(onEvent, { type: 'interpolateFinished' });\n\n emit(onEvent, { type: 'compileStarted' });\n const basePath = getFileBasePath(path, io);\n const { executeRequest } = await compileExecuteRequest(\n interpolated,\n setOptional<{ basePath: string; io?: IO; headerDefaults?: Record<string, string> }>({\n basePath\n })\n .ifDefined('io', io)\n .ifDefined('headerDefaults', headerDefaults)\n .build()\n );\n emit(onEvent, { type: 'compileFinished' });\n\n return { executeRequest, baseUrl: interpolated.url };\n }\n\n async function runCompiled(\n executeRequest: ExecuteRequest,\n urlForCookies: string,\n options: EngineRunOptions\n ): Promise<Response> {\n const headers = executeRequest.headers ?? {};\n\n const cookieHeader = cookieStore ? await cookieStore.getCookieHeader(urlForCookies) : undefined;\n const headersWithCookies = cookieStore ? withCookieHeader(headers, cookieHeader) : headers;\n\n const requestWithCookies: ExecuteRequest = {\n ...executeRequest,\n headers: headersWithCookies\n };\n\n emit(onEvent, { type: 'fetchStarted', method: requestWithCookies.method, url: urlForCookies });\n\n try {\n const execOptions = setOptional<ExecuteOptions>({})\n .ifDefined('timeout', options.timeoutMs)\n .ifDefined('signal', options.signal)\n .ifDefined('followRedirects', options.followRedirects)\n .ifDefined('validateSSL', options.validateSSL)\n .ifDefined('proxy', options.proxy)\n .build();\n\n const response = await executeWithTransport(requestWithCookies, execOptions, transport);\n\n emit(onEvent, {\n type: 'fetchFinished',\n method: requestWithCookies.method,\n url: urlForCookies,\n status: response.status\n });\n\n if (cookieStore) {\n await cookieStore.setFromResponse(urlForCookies, response);\n }\n\n return response;\n } catch (e) {\n emit(onEvent, {\n type: 'error',\n stage: 'fetch',\n message: e instanceof Error ? e.message : String(e)\n });\n throw e;\n }\n }\n\n return {\n parseString: parse,\n async runString(httpText, options = {}) {\n const { executeRequest, baseUrl } = await compileFromString(httpText, options);\n return await runCompiled(executeRequest, baseUrl, options);\n },\n async runFile(path, options = {}) {\n const { executeRequest, baseUrl } = await compileFromFile(path, options);\n return await runCompiled(executeRequest, baseUrl, options);\n }\n };\n}\n\nasync function compileExecuteRequest(\n interpolated: {\n method: string;\n url: string;\n headers: Record<string, string>;\n body?: string;\n bodyFile?: { path: string };\n formData?: FormField[];\n },\n ctx: { basePath: string; io?: IO; headerDefaults?: Record<string, string> }\n): Promise<{ executeRequest: ExecuteRequest }> {\n const headers: Record<string, string> = {\n ...(ctx.headerDefaults ?? {}),\n ...(interpolated.headers ?? {})\n };\n\n let body: ExecuteRequest['body'] = interpolated.body;\n\n if (interpolated.bodyFile) {\n const loadedFile = await loadFileBody(\n interpolated.bodyFile.path,\n setOptional<{ basePath: string; io?: IO }>({ basePath: ctx.basePath })\n .ifDefined('io', ctx.io)\n .build()\n );\n\n body = loadedFile.content;\n\n if (!headers['Content-Type'] && !headers['content-type']) {\n headers['Content-Type'] = loadedFile.mimeType;\n }\n } else if (interpolated.formData && interpolated.formData.length > 0) {\n const hasFiles = hasFileFields(interpolated.formData);\n\n if (hasFiles) {\n body = await buildFormData(\n interpolated.formData,\n setOptional<{ basePath: string; io?: IO }>({ basePath: ctx.basePath })\n .ifDefined('io', ctx.io)\n .build()\n );\n delete headers['Content-Type'];\n delete headers['content-type'];\n } else {\n body = buildUrlEncoded(interpolated.formData);\n if (!headers['Content-Type'] && !headers['content-type']) {\n headers['Content-Type'] = 'application/x-www-form-urlencoded';\n }\n }\n }\n\n const executeRequest: ExecuteRequest = {\n method: interpolated.method,\n url: interpolated.url,\n headers,\n ...(body !== undefined ? { body } : {})\n };\n\n return { executeRequest };\n}\n",
|
|
15
|
+
"/**\n * Server Client - Internal module for routing requests through treq server.\n *\n * This is the internal implementation used by createClient() when a server URL is configured.\n * The public API remains createClient({ server: 'http://localhost:4096' }).\n *\n * @internal\n */\n\nimport type { Client, ClientConfig, RunOptions } from './types';\n\n/**\n * Internal configuration for server client, extending public ClientConfig.\n */\nexport interface ServerClientConfig extends ClientConfig {\n /** Base URL of the treq server (required) */\n serverUrl: string;\n /** Bearer token for authentication */\n token?: string | undefined;\n}\n\n/**\n * Metadata about the server connection.\n * Accessible via getServerMetadata() utility.\n */\nexport interface ServerMetadata {\n /** The server URL the client is connected to */\n readonly serverUrl: string;\n /** Session ID created on the server */\n readonly sessionId: string | undefined;\n /** Flow ID for observability (TUI) */\n readonly flowId: string | undefined;\n}\n\n// Symbol for storing server metadata on the client instance\nexport const SERVER_METADATA = Symbol('serverMetadata');\n\n// ============================================================================\n// Server Response Types\n// ============================================================================\n\ninterface CreateSessionResponse {\n sessionId: string;\n}\n\ninterface CreateFlowResponse {\n flowId: string;\n}\n\ninterface ExecuteResponse {\n runId: string;\n reqExecId?: string;\n flowId?: string;\n request: {\n index: number;\n name?: string;\n method: string;\n url: string;\n };\n response: {\n status: number;\n statusText: string;\n headers: Array<{ name: string; value: string }>;\n body?: string;\n encoding: 'utf-8' | 'base64';\n truncated: boolean;\n bodyBytes: number;\n };\n timing: {\n startTime: number;\n endTime: number;\n durationMs: number;\n };\n}\n\n// ============================================================================\n// Implementation\n// ============================================================================\n\n/**\n * Read environment variable safely (works in both Node and Bun).\n */\nexport function getEnvVar(name: string): string | undefined {\n if (typeof process !== 'undefined' && process.env) {\n return process.env[name];\n }\n return undefined;\n}\n\n/**\n * Create a server-backed client (internal use).\n *\n * Called by createClient() when server URL is provided.\n * Automatically creates a session and flow on first use.\n *\n * When running as a server-spawned script:\n * - TREQ_FLOW_ID: Attach to existing flow (don't create new one)\n * - TREQ_SESSION_ID: Use pre-created session (skip session creation)\n * - TREQ_TOKEN: Use scoped script token for authentication\n *\n * @internal\n */\nexport function createServerClient(config: ServerClientConfig): Client {\n const { serverUrl } = config;\n const attachedFlowId = getEnvVar('TREQ_FLOW_ID');\n const preCreatedSessionId = getEnvVar('TREQ_SESSION_ID');\n const envToken = getEnvVar('TREQ_TOKEN');\n\n // Use environment token if available (for server-spawned scripts)\n // Fall back to config token if provided\n const token = envToken ?? config.token;\n\n const baseUrl = serverUrl.replace(/\\/$/, '');\n let variables: Record<string, unknown> = { ...config.variables };\n let sessionId: string | undefined = preCreatedSessionId;\n let flowId: string | undefined = attachedFlowId;\n let initialized = false;\n let initPromise: Promise<void> | undefined;\n\n // Track if we're using pre-created resources (server-spawned script mode)\n const isAttachedFlow = !!attachedFlowId;\n const hasPreCreatedSession = !!preCreatedSessionId;\n\n // Serialize variable sync operations so rapid updates cannot race.\n let variableSyncChain: Promise<void> = Promise.resolve();\n\n const defaultTimeout = config.timeout ?? 30000;\n\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json'\n };\n if (token) {\n headers['Authorization'] = `Bearer ${token}`;\n }\n\n // Initialize session and flow on first request\n async function ensureInitialized(): Promise<void> {\n if (initialized) return;\n if (initPromise) return initPromise;\n\n initPromise = (async () => {\n // Check for pre-created session (server-spawned scripts)\n // If TREQ_SESSION_ID is set, skip session creation - use the pre-created one\n if (hasPreCreatedSession) {\n // Session ID is already set from environment variable\n // Flow ID is also set from TREQ_FLOW_ID\n // Just mark as initialized and return\n initialized = true;\n return;\n }\n\n // 1. Create session (needed for variables/cookies)\n const sessionRes = await fetch(`${baseUrl}/session`, {\n method: 'POST',\n headers,\n body: JSON.stringify({ variables })\n });\n\n if (!sessionRes.ok) {\n const error = await sessionRes.text();\n throw new Error(`Failed to create session: ${error}`);\n }\n\n const sessionData = (await sessionRes.json()) as CreateSessionResponse;\n sessionId = sessionData.sessionId;\n\n // 2. Create flow with sessionId ONLY if not attached to an existing flow\n if (!isAttachedFlow) {\n const flowRes = await fetch(`${baseUrl}/flows`, {\n method: 'POST',\n headers,\n body: JSON.stringify({\n sessionId\n })\n });\n\n if (!flowRes.ok) {\n const error = await flowRes.text();\n throw new Error(`Failed to create flow: ${error}`);\n }\n\n const flowData = (await flowRes.json()) as CreateFlowResponse;\n flowId = flowData.flowId;\n }\n\n initialized = true;\n })();\n\n return initPromise;\n }\n\n function enqueueVariableSync(vars: Record<string, unknown>): void {\n const doSync = async (): Promise<void> => {\n if (!sessionId) return;\n try {\n const res = await fetch(`${baseUrl}/session/${sessionId}/variables`, {\n method: 'PUT',\n headers,\n body: JSON.stringify({ variables: vars, mode: 'merge' })\n });\n if (!res.ok) {\n console.error('Failed to sync variables to server:', res.statusText);\n }\n } catch (err) {\n console.error('Failed to sync variables to server:', err);\n }\n };\n\n if (sessionId) {\n variableSyncChain = variableSyncChain.catch(() => undefined).then(doSync);\n } else if (initPromise) {\n variableSyncChain = variableSyncChain\n .catch(() => undefined)\n .then(() => initPromise)\n .then(doSync);\n }\n }\n\n async function executeRequest(\n source: { path?: string; content?: string },\n options: RunOptions = {}\n ): Promise<Response> {\n await ensureInitialized();\n\n // Await any queued variable sync to ensure server has latest values\n await variableSyncChain;\n\n const body: Record<string, unknown> = {\n ...source,\n sessionId,\n flowId,\n profile: config.profile,\n timeoutMs: options.timeout ?? defaultTimeout\n };\n\n // Add reqLabel based on source\n if (source.path) {\n body['reqLabel'] = source.path;\n }\n\n // Add per-request variables if provided\n if (options.variables && Object.keys(options.variables).length > 0) {\n body['variables'] = options.variables;\n }\n\n // Add basePath if provided (for content execution)\n if (options.basePath) {\n body['basePath'] = options.basePath;\n }\n\n // Build fetch options\n const fetchOptions: RequestInit = {\n method: 'POST',\n headers,\n body: JSON.stringify(body)\n };\n if (options.signal) {\n fetchOptions.signal = options.signal;\n }\n\n const execRes = await fetch(`${baseUrl}/execute`, fetchOptions);\n\n if (!execRes.ok) {\n const error = await execRes.text();\n throw new Error(`Execution failed: ${error}`);\n }\n\n const data = (await execRes.json()) as ExecuteResponse;\n\n // Convert server response back to a fetch-like Response\n return serverResponseToFetchResponse(data.response);\n }\n\n function serverResponseToFetchResponse(serverRes: ExecuteResponse['response']): Response {\n // Decode body based on encoding\n let body: BodyInit | undefined;\n if (serverRes.body !== undefined) {\n if (serverRes.encoding === 'base64') {\n // Decode base64 to binary\n const binaryStr = atob(serverRes.body);\n const bytes = new Uint8Array(binaryStr.length);\n for (let i = 0; i < binaryStr.length; i++) {\n bytes[i] = binaryStr.charCodeAt(i);\n }\n body = bytes;\n } else {\n body = serverRes.body;\n }\n }\n\n // Build headers\n const responseHeaders = new Headers();\n for (const h of serverRes.headers) {\n responseHeaders.append(h.name, h.value);\n }\n\n return new Response(body, {\n status: serverRes.status,\n statusText: serverRes.statusText,\n headers: responseHeaders\n });\n }\n\n async function close(): Promise<void> {\n // Only finish the flow if we created it (not when attached to TUI's flow)\n if (!flowId || isAttachedFlow) return;\n\n try {\n await fetch(`${baseUrl}/flows/${flowId}/finish`, {\n method: 'POST',\n headers\n });\n } catch {\n // Best-effort - server will TTL anyway\n }\n }\n\n const client: Client & { [SERVER_METADATA]: () => ServerMetadata } = {\n async run(path: string, options: RunOptions = {}): Promise<Response> {\n return executeRequest({ path }, options);\n },\n\n async runString(content: string, options: RunOptions = {}): Promise<Response> {\n return executeRequest({ content }, options);\n },\n\n setVariables(vars: Record<string, unknown>): void {\n variables = { ...variables, ...vars };\n enqueueVariableSync(vars);\n },\n\n setVariable(key: string, value: unknown): void {\n variables[key] = value;\n enqueueVariableSync({ [key]: value });\n },\n\n getVariables(): Record<string, unknown> {\n return { ...variables };\n },\n\n close,\n\n [Symbol.asyncDispose]: close,\n\n // Internal: accessor for server metadata\n [SERVER_METADATA]: () => ({\n serverUrl: baseUrl,\n sessionId,\n flowId\n })\n };\n\n return client;\n}\n",
|
|
16
|
+
"import type { CookieJar } from 'tough-cookie';\nimport type { EngineConfig, EngineRunOptions } from './engine/engine';\nimport { createEngine } from './engine/engine';\nimport type { CookieStore } from './runtime/types';\nimport { createServerClient, getEnvVar } from './server-client';\nimport type { Client, ClientConfig, RunOptions } from './types';\nimport { setOptional } from './utils/optional';\n\nexport function createClient(config: ClientConfig = {}): Client {\n // Route to server implementation if server URL provided\n const serverUrl = config.server ?? getEnvVar('TREQ_SERVER');\n if (serverUrl) {\n return createServerClient({\n ...config,\n serverUrl,\n token: config.serverToken ?? getEnvVar('TREQ_TOKEN')\n });\n }\n\n // Local implementation\n let variables: Record<string, unknown> = { ...config.variables };\n const cookieJar = config.cookieJar;\n const defaultTimeout = config.timeout ?? 30000;\n const defaults = config.defaults || {};\n\n const cookieStore = cookieJar ? cookieJarToStore(cookieJar) : undefined;\n\n const engine = createEngine(\n setOptional<EngineConfig>({} as EngineConfig)\n .ifDefined('io', config.io)\n .ifDefined('transport', config.transport)\n .ifDefined('cookieStore', cookieStore)\n .ifDefined('resolvers', config.resolvers)\n .ifDefined('onEvent', config.onEvent)\n .ifDefined('headerDefaults', defaults.headers)\n .build()\n );\n\n function mergedVars(runVars?: Record<string, unknown>): Record<string, unknown> {\n return { ...variables, ...runVars };\n }\n\n // No-op close for local client\n async function close(): Promise<void> {\n // Local client has no external resources to release\n }\n\n return {\n async run(filePath: string, options: RunOptions = {}): Promise<Response> {\n return await engine.runFile(\n filePath,\n setOptional<EngineRunOptions>({\n variables: mergedVars(options.variables),\n timeoutMs: options.timeout ?? defaultTimeout\n })\n .ifDefined('signal', options.signal)\n .ifDefined('followRedirects', defaults.followRedirects)\n .ifDefined('validateSSL', defaults.validateSSL)\n .ifDefined('proxy', defaults.proxy)\n .build()\n );\n },\n\n async runString(content: string, options: RunOptions = {}): Promise<Response> {\n return await engine.runString(\n content,\n setOptional<EngineRunOptions>({\n variables: mergedVars(options.variables),\n timeoutMs: options.timeout ?? defaultTimeout\n })\n .ifDefined('signal', options.signal)\n .ifDefined('followRedirects', defaults.followRedirects)\n .ifDefined('validateSSL', defaults.validateSSL)\n .ifDefined('proxy', defaults.proxy)\n .ifDefined('basePath', options.basePath)\n .build()\n );\n },\n\n setVariables(vars: Record<string, unknown>): void {\n variables = { ...variables, ...vars };\n },\n\n setVariable(key: string, value: unknown): void {\n variables[key] = value;\n },\n\n getVariables(): Record<string, unknown> {\n return { ...variables };\n },\n\n close,\n\n [Symbol.asyncDispose]: close\n };\n}\n\nfunction cookieJarToStore(cookieJar: CookieJar): CookieStore {\n return {\n getCookieHeader(url: string) {\n try {\n const v = cookieJar.getCookieStringSync(url);\n return v || undefined;\n } catch {\n return undefined;\n }\n },\n setFromResponse(url: string, response: Response) {\n const setCookieHeaders = getSetCookieHeaders(response);\n for (const header of setCookieHeaders) {\n cookieJar.setCookieSync(header, url, { ignoreError: true });\n }\n }\n };\n}\n\nfunction getSetCookieHeaders(response: Response): string[] {\n const anyHeaders = response.headers as unknown as { getSetCookie?: () => string[] };\n const fromGetSetCookie = anyHeaders.getSetCookie?.();\n if (Array.isArray(fromGetSetCookie) && fromGetSetCookie.length > 0) return fromGetSetCookie;\n\n const single = response.headers.get('set-cookie');\n return single ? [single] : [];\n}\n",
|
|
17
|
+
"/**\n * Server metadata utilities for TUI and tooling.\n *\n * When a client is created with a server URL, it maintains internal metadata\n * about the server connection (session ID, flow ID). This utility allows\n * external tools to access this metadata for debugging, observability, or\n * orchestration purposes.\n */\n\nimport { SERVER_METADATA, type ServerMetadata } from './server-client';\nimport type { Client } from './types';\n\nexport type { ServerMetadata } from './server-client';\n\n/**\n * Get server connection metadata from a client.\n *\n * Returns metadata only for server-backed clients (created with `server` option).\n * Returns `undefined` for local clients.\n *\n * @example\n * ```typescript\n * const client = createClient({ server: 'http://localhost:4096' });\n * await client.run('./test.http');\n *\n * const meta = getServerMetadata(client);\n * if (meta) {\n * console.log('Session:', meta.sessionId);\n * console.log('Flow:', meta.flowId);\n * console.log('Server:', meta.serverUrl);\n * }\n * ```\n */\ninterface ClientWithServerMetadata extends Client {\n [SERVER_METADATA]?: () => ServerMetadata;\n}\n\nexport function getServerMetadata(client: Client): ServerMetadata | undefined {\n const serverClient = client as ClientWithServerMetadata;\n const accessor = serverClient[SERVER_METADATA];\n if (typeof accessor === 'function') {\n return accessor();\n }\n return undefined;\n}\n"
|
|
16
18
|
],
|
|
17
|
-
"mappings": "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",
|
|
18
|
-
"debugId": "
|
|
19
|
+
"mappings": "wEAIA,SAAS,CAAW,CAAC,EAAqB,CACxC,MAAU,MACR,GAAG,6EACL,EASK,SAAS,CAAoB,CAAC,EAAuB,MAAkB,CAC5E,MAAO,CACL,aAAc,CAAE,MAAO,GAAO,YAAa,EAAM,OAC3C,MAAK,CAAC,EAAK,EAAM,EAAK,CAC1B,GAAI,EAAI,MAAO,EAAY,OAAO,EAClC,GAAI,EAAI,cAAgB,GAAO,EAAY,mBAAmB,EAC9D,OAAO,MAAM,EAAU,EAAK,CAAI,EAEpC,ECnBF,SAAS,EAAM,EAAY,CACzB,OAAO,OAAQ,WAAuC,IAAW,IAS5D,SAAS,CAAmB,EAAc,CAC/C,GAAI,CAAC,GAAO,EACV,OAAO,EAAqB,KAAK,EAGnC,MAAO,CACL,aAAc,CAAE,MAAO,GAAM,YAAa,EAAK,OACzC,MAAK,CAAC,EAAK,EAAM,EAAK,CAC1B,IAAM,EAAwB,IAAM,GAAQ,CAAC,CAAG,EAEhD,GAAI,EAAI,QAAU,OAChB,EAAQ,MAAQ,EAAI,MAGtB,GAAI,EAAI,cAAgB,GACtB,EAAQ,IAAM,CAAE,mBAAoB,EAAM,EAG5C,OAAO,MAAM,MAAM,EAAK,CAAO,EAEnC,ECnCF,iBAAS,eAAQ,yBACjB,4BAGA,eAAe,EAAM,CAAC,EAA6B,CACjD,GAAI,CAEF,OADA,MAAM,GAAO,CAAC,EACP,GACP,KAAM,CACN,MAAO,IAOJ,SAAS,EAAY,EAAO,CACjC,MAAO,CACL,IAAK,IAAM,QAAQ,IAAI,EACvB,KAAM,CACJ,QAAS,IAAI,IAAe,UAAQ,GAAG,CAAK,EAC5C,QAAS,CAAC,IAAW,UAAQ,CAAC,EAC9B,SAAU,CAAC,IAAW,WAAS,CAAC,EAChC,QAAS,CAAC,IAAW,UAAQ,CAAC,EAC9B,WAAY,CAAC,IAAW,aAAW,CAAC,EACpC,IAAU,KACZ,EACA,UACA,SAAU,MAAO,IAAM,MAAM,EAAS,EAAG,MAAM,EAC/C,WAAY,MAAO,IAAM,CACvB,IAAM,EAAM,MAAM,EAAS,CAAC,EAG5B,OAAO,EAAI,OAAO,MAAM,EAAI,WAAY,EAAI,WAAa,EAAI,UAAU,EAE3E,ECMK,SAAS,CAA6B,CAAC,EAA6B,CACzE,IAAI,EAAS,IAAK,CAAK,EAEjB,EAA8B,CAClC,IAAI,CAAC,EAAW,EAAO,CACrB,GAAI,EACF,EAAS,IAAK,KAAW,CAAM,EAEjC,OAAO,GAGT,SAAS,CAAC,EAAK,EAAO,CACpB,GAAI,IAAU,OACZ,EAAS,IAAK,GAAS,GAAM,CAAM,EAErC,OAAO,GAGT,KAAK,EAAG,CACN,OAAO,EAEX,EAEA,OAAO,ECvDT,SAAS,EAAgB,CAAC,EAAgB,EAAuC,CAC/E,OAAO,IAAS,QAAa,CAAC,CAAC,MAAO,MAAM,EAAE,SAAS,EAAO,YAAY,CAAC,EAM7E,SAAS,EAAgB,CACvB,EACA,EAIa,CACb,IAAM,EACJ,GAAiB,EAAQ,OAAQ,EAAQ,IAAI,GAAK,EAAQ,OAAS,OAErE,OAAO,EAAyB,CAC9B,OAAQ,EAAQ,OAChB,SAAU,EAAK,gBAAkB,SAAW,SAC5C,OAAQ,EAAK,MACf,CAAC,EACE,UAAU,UAAW,EAAQ,OAAO,EACpC,UAAU,OAAQ,EAAoB,EAAQ,KAAO,MAAS,EAC9D,MAAM,EAOX,SAAS,EAAqB,CAAC,EAI7B,CACA,GAAI,EAAK,SACP,MAAO,CACL,OAAQ,EAAK,SACb,kBAAmB,GACnB,QAAS,IAAM,EACjB,EAGF,IAAM,EAAa,IAAI,gBACjB,EAAY,WAAW,IAAM,EAAW,MAAM,EAAG,EAAK,OAAO,EAEnE,MAAO,CACL,OAAQ,EAAW,OACnB,kBAAmB,GACnB,QAAS,IAAM,aAAa,CAAS,CACvC,EAOF,SAAS,EAAe,CACtB,EACA,EACO,CACP,GAAI,aAAiB,OAAS,EAAM,OAAS,cAC3C,GAAI,EAAI,kBACN,OAAW,MAAM,yBAAyB,EAAI,WAAW,EAG7D,OAAO,aAAiB,MAAQ,EAAY,MAAM,OAAO,CAAK,CAAC,EAsBjE,eAAsB,CAAoB,CACxC,EACA,EACA,EACmB,CACnB,IAAM,EAAU,EAAQ,SAAW,MAE7B,EAAsB,EAAyD,CAAE,SAAQ,CAAC,EAC7F,UAAU,WAAY,EAAQ,MAAM,EACpC,MAAM,GAED,SAAQ,oBAAmB,WAAY,GAAsB,CAAmB,EAElF,EAAU,EAGb,CACD,gBAAiB,EAAQ,iBAAmB,GAC5C,QACF,CAAC,EAAE,MAAM,EAEH,EAAc,GAAiB,EAAS,CAAO,EAErD,GAAI,CACF,IAAM,EAAM,EAA8B,CAAC,CAAC,EACzC,UAAU,QAAS,EAAQ,KAAK,EAChC,UAAU,cAAe,EAAQ,WAAW,EAC5C,MAAM,EAET,OAAO,MAAM,EAAU,MAAM,EAAQ,IAAK,EAAa,CAAG,EAC1D,MAAO,EAAO,CACd,MAAM,GAAgB,EAAO,CAAE,UAAS,mBAAkB,CAAC,SAC3D,CACA,EAAQ,GCxFZ,IAAM,GAA6C,CAEjD,QAAS,mBACT,OAAQ,kBACR,OAAQ,aACR,QAAS,YACT,OAAQ,YACR,OAAQ,WACR,MAAO,yBACP,OAAQ,yBACR,MAAO,yBACP,OAAQ,WACR,MAAO,gBACP,QAAS,YACT,OAAQ,YACR,OAAQ,gBAGR,OAAQ,aACR,QAAS,aACT,OAAQ,YACR,OAAQ,YACR,QAAS,aACT,OAAQ,eACR,OAAQ,YACR,QAAS,aACT,OAAQ,aAGR,OAAQ,kBACR,OAAQ,qBACR,QAAS,0EACT,OAAQ,2BACR,QAAS,oEACT,OAAQ,gCACR,QAAS,4EAGT,OAAQ,kBACR,MAAO,mBACP,OAAQ,oBACR,OAAQ,sBACR,MAAO,8BAGP,OAAQ,aACR,OAAQ,YACR,OAAQ,YACR,OAAQ,YAGR,OAAQ,YACR,QAAS,aACT,OAAQ,kBACR,OAAQ,kBAGR,QAAS,mBACT,OAAQ,2BACR,OAAQ,2BACR,OAAQ,2BACR,MAAO,2BACP,SAAU,0BACZ,EAKM,GAAkB,IAAI,IAAI,CAC9B,mBACA,kBACA,yBACA,yBACA,aACA,YACA,WACA,WACA,WACA,gBACA,YACA,eACF,CAAC,EAeM,SAAS,EAAa,CAAC,EAA0B,CACtD,IAAM,EAAM,GAAQ,CAAQ,EAAE,YAAY,EAC1C,OAAO,GAAmB,IAAQ,2BAgB7B,SAAS,EAAgB,CAAC,EAA2B,CAE1D,GAAI,EAAS,WAAW,OAAO,EAC7B,MAAO,GAGT,GAAI,GAAgB,IAAI,CAAQ,EAC9B,MAAO,GAGT,MAAO,GAOT,SAAS,CAAU,CAAC,EAAmB,CACrC,OAAO,EAAE,SAAS,IAAI,EAAI,KAAO,IAGnC,SAAS,CAAU,CAAC,EAAqB,CACvC,OAAO,EAAE,MAAM,QAAQ,EAAE,OAAO,OAAO,EAGzC,SAAS,EAAc,CAAC,EAAoB,CAC1C,GAAI,EAAE,WAAW,GAAG,GAAK,EAAE,WAAW,IAAI,EAAG,MAAO,GAEpD,GAAI,kBAAkB,KAAK,CAAC,EAAG,MAAO,GAEtC,GAAI,EAAE,WAAW,MAAM,EAAG,MAAO,GACjC,MAAO,GAGT,SAAS,CAAa,CAAC,EAAa,EAAyB,CAC3D,IAAM,EAAgB,CAAC,EACvB,QAAW,KAAQ,EAAO,CACxB,GAAI,CAAC,GAAQ,IAAS,IAAK,SAC3B,GAAI,IAAS,KAAM,CACjB,EAAI,IAAI,EACR,SAEF,EAAI,KAAK,CAAI,EAEf,OAAQ,IAAQ,KAAO,KAAO,KAAO,EAAI,KAAK,CAAG,EAGnD,SAAS,EAAW,CAAC,EAAkB,EAA8B,CACnE,IAAM,EAAM,EAAW,CAAQ,EAEzB,EAAc,cAAc,KAAK,CAAQ,EAEzC,EADiB,EAAS,MAAM,oBAAoB,IAC3B,GAEzB,EAAY,EAAW,CAAQ,EAC/B,EAAW,EAAW,CAAY,EAExC,GAAI,EAAO,CACT,IAAM,EAAa,EAAc,EAAK,CAAC,GAAG,EAAU,MAAM,CAAC,EAAG,GAAG,CAAQ,CAAC,EAC1E,MAAO,GAAG,IAAQ,IAGpB,GAAI,EAAa,CACf,IAAM,EAAa,EAAc,EAAK,CAAQ,EAC9C,MAAO,GAAG,IAAW,IAGvB,OAAO,EAAc,EAAK,CAAC,GAAG,EAAW,GAAG,CAAQ,CAAC,EAGvD,SAAS,EAAa,CAAC,EAA0B,CAC/C,GAAI,IAAa,GAAI,MAAO,IAC5B,IAAM,EAAM,EAAW,CAAQ,EAEzB,EADa,EAAS,MAAM,oBAAoB,IAC3B,GACrB,EAAQ,EAAW,CAAQ,EACjC,GAAI,EAAO,CACT,IAAM,EAAa,EAAc,EAAK,EAAM,MAAM,CAAC,CAAC,EACpD,MAAO,GAAG,IAAQ,IAEpB,OAAO,EAAc,EAAK,CAAK,EAGjC,SAAS,EAAO,CAAC,EAAmB,CAClC,IAAM,EAAO,GAAS,CAAC,EACjB,EAAM,EAAK,YAAY,GAAG,EAChC,GAAI,GAAO,EAAG,MAAO,GACrB,OAAO,EAAK,MAAM,CAAG,EAGvB,SAAS,EAAQ,CAAC,EAAmB,CACnC,IAAM,EAAQ,EAAE,MAAM,OAAO,EAE7B,OADa,EAAM,EAAM,OAAS,IACnB,GAuBV,SAAS,EAAgB,CAAC,EAAkB,EAA0B,CAE3E,GAAI,GAAe,CAAQ,EACzB,MAAU,MAAM,+BAA+B,GAAU,EAI3D,IAAM,EAAW,GAAY,EAAU,CAAQ,EACzC,EAAiB,GAAc,CAAQ,EACvC,EAAM,EAAW,CAAc,EAIrC,GAAI,IAAa,GAAkB,CAAC,EAAS,WAAW,EAAiB,CAAG,EAC1E,MAAU,MAAM,gCAAgC,GAAU,EAG5D,OAAO,EA6BT,eAAsB,CAAY,CAChC,EACA,EAA6B,CAAC,EACT,CACrB,IAAM,EACJ,EAAQ,UACR,EAAQ,IAAI,IAAI,GACf,WAA+D,SAAS,MAAM,GAC/E,IAGI,EAAe,GAAiB,EAAU,CAAQ,EAGlD,EAAW,GAAc,CAAQ,EACjC,EAAW,GAAiB,CAAQ,EAEtC,EAEJ,GAAI,EAAQ,GAAI,CAEd,GAAI,CADW,MAAM,EAAQ,GAAG,OAAO,CAAY,EAEjD,MAAU,MAAM,mBAAmB,GAAU,EAE/C,EAAU,EACN,MAAM,EAAQ,GAAG,WAAW,CAAY,EACxC,MAAM,EAAQ,GAAG,SAAS,CAAY,EACrC,QAAI,OAAQ,WAAuC,IAAW,IAAa,CAShF,IAAM,EADO,WAA6C,IACzC,KAAK,CAAY,EAElC,GAAI,CADW,MAAM,EAAK,OAAO,EAE/B,MAAU,MAAM,mBAAmB,GAAU,EAE/C,EAAU,EAAW,MAAM,EAAK,YAAY,EAAI,MAAM,EAAK,KAAK,EAEhE,WAAU,MACR,oFACF,EAGF,MAAO,CACL,UACA,WACA,UACF,ECjVK,SAAS,CAAa,CAAC,EAA8B,CAC1D,OAAO,EAAO,KAAK,CAAC,IAAU,EAAM,MAAM,EAyB5C,eAAsB,CAAa,CACjC,EACA,EAAgC,CAAC,EACd,CACnB,IAAM,EAAO,IAAI,SACX,EACJ,EAAQ,UACR,EAAQ,IAAI,IAAI,GACf,WAA+D,SAAS,MAAM,GAC/E,IAEF,QAAW,KAAS,EAClB,GAAI,EAAM,QAAU,EAAM,KAAM,CAE9B,IAAM,EAAa,MAAM,EACvB,EAAM,KACN,EAAQ,GAAK,CAAE,WAAU,GAAI,EAAQ,EAAG,EAAI,CAAE,UAAS,CACzD,EAGM,EACJ,EAAM,UAAY,EAAQ,IAAI,KAAK,SAAS,EAAM,IAAI,GAAK,GAAS,EAAM,IAAI,EAG1E,EAAO,IAAI,KAAK,CAAC,EAAW,OAAO,EAAG,CAAE,KAAM,EAAW,QAAS,CAAC,EAGzE,EAAK,OAAO,EAAM,KAAM,EAAM,CAAQ,EAGtC,OAAK,OAAO,EAAM,KAAM,EAAM,KAAK,EAIvC,OAAO,EAGT,SAAS,EAAQ,CAAC,EAAmB,CACnC,IAAM,EAAQ,EAAE,MAAM,OAAO,EAE7B,OADa,EAAM,EAAM,OAAS,IACnB,GAuBV,SAAS,CAAe,CAAC,EAAsC,CACpE,IAAM,EAAS,IAAI,gBAEnB,QAAW,KAAS,EAAQ,CAE1B,GAAI,EAAM,OACR,SAEF,EAAO,OAAO,EAAM,KAAM,EAAM,KAAK,EAGvC,OAAO,EChIT,SAAS,EAAM,CAAC,EAAoB,CAClC,MAAO,kBAAkB,KAAK,CAAC,EAGjC,SAAS,CAAa,CAAC,EAA+B,CACpD,IAAM,EAAwB,CAAC,EAC3B,EAAI,EACJ,EAAY,EAEhB,MAAO,EAAI,EAAM,OAAQ,CACvB,IAAM,EAAI,EAAM,GACV,EAAI,EAAM,EAAI,GAEpB,GAAI,IAAM,KAAO,IAAM,IAAK,CAC1B,GAAI,EAAI,EACN,EAAM,KAAK,CAAE,KAAM,OAAQ,MAAO,EAAM,MAAM,EAAW,CAAC,CAAE,CAAC,EAG/D,IAAM,EAAQ,EACd,GAAK,EAEL,IAAI,EAAQ,EACR,EAAO,GAEX,MAAO,EAAI,EAAM,OAAQ,CACvB,IAAM,EAAK,EAAM,GACX,EAAK,EAAM,EAAI,GAErB,GAAI,IAAO,KAAO,IAAO,IAAK,CAC5B,IACA,GAAQ,KACR,GAAK,EACL,SAGF,GAAI,IAAO,KAAO,IAAO,IAAK,CAE5B,GADA,IACI,IAAU,EAAG,CACf,GAAK,EACL,MAEF,GAAQ,KACR,GAAK,EACL,SAGF,GAAQ,EACR,IAGF,GAAI,IAAU,EACZ,MAAU,MAAM,0CAA0C,EAG5D,IAAM,EAAM,EAAM,MAAM,EAAO,CAAC,EAChC,EAAM,KAAK,CAAE,KAAM,OAAQ,WAAY,EAAM,KAAI,CAAC,EAClD,EAAY,EACZ,SAGF,IAGF,GAAI,EAAY,EAAM,OACpB,EAAM,KAAK,CAAE,KAAM,OAAQ,MAAO,EAAM,MAAM,CAAS,CAAE,CAAC,EAG5D,OAAO,EAMT,SAAS,CAAc,CAAC,EAA8B,EAAuB,CAC3E,IAAM,EAAQ,EAAK,MAAM,GAAG,EACxB,EAAmB,EAEvB,QAAW,KAAQ,EAAO,CACxB,GAAI,IAAY,MAAQ,IAAY,OAClC,OAEF,GAAI,OAAO,IAAY,SACrB,OAEF,EAAW,EAAoC,GAGjD,OAAO,EAGT,SAAS,CAAiB,CAAC,EAAqE,CAC9F,IAAM,EAAU,EAAW,KAAK,EAChC,GAAI,CAAC,EAAQ,WAAW,GAAG,EAAG,OAAO,KACrC,GAAI,CAAC,EAAQ,SAAS,GAAG,EAAG,OAAO,KAEnC,IAAM,EAAU,EAAQ,QAAQ,GAAG,EACnC,GAAI,IAAY,GAAI,OAAO,KAE3B,IAAM,EAAO,EAAQ,MAAM,EAAG,CAAO,EAAE,KAAK,EAC5C,GAAI,CAAC,GAAQ,CAAC,GAAO,CAAI,EAAG,OAAO,KAEnC,IAAM,EAAU,EAAQ,MAAM,EAAU,EAAG,EAAE,EAC7C,MAAO,CAAE,YAAa,IAAI,IAAQ,SAAQ,EAG5C,SAAS,EAAwB,CAC/B,EACA,EACA,EACQ,CACR,IAAM,EAAQ,EAAc,CAAG,EAC/B,GAAI,EAAM,SAAW,GAAK,EAAM,IAAI,OAAS,OAAQ,OAAO,EAE5D,IAAI,EAAM,GACV,QAAW,KAAQ,EAAO,CACxB,GAAI,EAAK,OAAS,OAAQ,CACxB,GAAO,EAAK,MACZ,SAGF,IAAM,EAAO,EAAK,WAAW,KAAK,EAClC,GAAI,EAAkB,CAAI,EACxB,MAAU,MAAM,wDAAwD,EAAK,KAAK,EAGpF,IAAM,EAAQ,EAAe,EAAW,CAAI,EAC5C,GAAI,IAAU,OAAW,CACvB,GAAI,IAAsB,QACxB,MAAU,MAAM,uBAAuB,GAAM,EAE/C,GAAO,IAAsB,OAAS,EAAK,IAAM,GACjD,SAGF,GAAO,OAAO,CAAK,EAGrB,OAAO,EAGT,SAAS,EAAyB,CAChC,EACA,EACA,EACU,CACV,IAAM,EAAe,GAAyB,EAAS,EAAW,CAAiB,EAAE,KAAK,EAC1F,GAAI,CAAC,EAAc,MAAO,CAAC,EAE3B,GAAI,CACF,IAAM,EAAS,KAAK,MAAM,CAAY,EACtC,GAAI,MAAM,QAAQ,CAAM,EACtB,OAAO,EAAO,IAAI,CAAC,IAAM,OAAO,CAAC,CAAC,EAEpC,KAAM,EAIR,MAAO,CAAC,CAAY,EAMtB,SAAS,EAAiB,CACxB,EACA,EACA,EAA8B,CAAC,EACvB,CACR,IAAQ,YAAY,CAAC,EAAG,oBAAoB,SAAY,EAElD,EAAQ,EAAc,CAAG,EAC/B,GAAI,EAAM,SAAW,GAAK,EAAM,IAAI,OAAS,OAAQ,OAAO,EAE5D,IAAI,EAAM,GACV,QAAW,KAAQ,EAAO,CACxB,GAAI,EAAK,OAAS,OAAQ,CACxB,GAAO,EAAK,MACZ,SAGF,IAAM,EAAS,EAAkB,EAAK,UAAU,EAChD,GAAI,EAAQ,CACV,IAAM,EAAW,EAAU,EAAO,aAClC,GAAI,EAAU,CACZ,IAAM,EAAO,GAA0B,EAAO,QAAS,EAAW,CAAiB,EAC7E,EAAS,EAAS,GAAG,CAAI,EAC/B,GAAI,aAAkB,QACpB,MAAU,MACR,YAAY,EAAO,+EACrB,EAEF,GAAO,OAAO,CAAM,EACpB,SAGF,GAAI,IAAsB,QACxB,MAAU,MAAM,qBAAqB,EAAO,aAAa,EAE3D,GAAO,IAAsB,OAAS,EAAK,IAAM,GACjD,SAGF,IAAM,EAAO,EAAK,WAAW,KAAK,EAC5B,EAAQ,EAAe,EAAW,CAAI,EAC5C,GAAI,IAAU,OAAW,CACvB,GAAI,IAAsB,QACxB,MAAU,MAAM,uBAAuB,GAAM,EAE/C,GAAO,IAAsB,OAAS,EAAK,IAAM,GACjD,SAGF,GAAO,OAAO,CAAK,EAGrB,OAAO,EAMT,eAAe,EAAsB,CACnC,EACA,EACA,EAA8B,CAAC,EACd,CACjB,IAAQ,YAAY,CAAC,EAAG,oBAAoB,SAAY,EAClD,EAAQ,EAAc,CAAG,EAC/B,GAAI,EAAM,SAAW,GAAK,EAAM,IAAI,OAAS,OAAQ,OAAO,EAE5D,IAAI,EAAM,GACV,QAAW,KAAQ,EAAO,CACxB,GAAI,EAAK,OAAS,OAAQ,CACxB,GAAO,EAAK,MACZ,SAGF,IAAM,EAAS,EAAkB,EAAK,UAAU,EAChD,GAAI,EAAQ,CACV,IAAM,EAAW,EAAU,EAAO,aAClC,GAAI,EAAU,CACZ,IAAM,EAAO,GAA0B,EAAO,QAAS,EAAW,CAAiB,EAC7E,EAAS,MAAM,EAAS,GAAG,CAAI,EACrC,GAAO,OAAO,CAAM,EACpB,SAGF,GAAI,IAAsB,QACxB,MAAU,MAAM,qBAAqB,EAAO,aAAa,EAE3D,GAAO,IAAsB,OAAS,EAAK,IAAM,GACjD,SAGF,IAAM,EAAO,EAAK,WAAW,KAAK,EAC5B,EAAQ,EAAe,EAAW,CAAI,EAC5C,GAAI,IAAU,OAAW,CACvB,GAAI,IAAsB,QACxB,MAAU,MAAM,uBAAuB,GAAM,EAE/C,GAAO,IAAsB,OAAS,EAAK,IAAM,GACjD,SAGF,GAAO,OAAO,CAAK,EAGrB,OAAO,EAMT,SAAS,CAAmB,CAC1B,EACA,EACA,EACG,CACH,GAAI,OAAO,IAAU,SACnB,OAAO,GAAkB,EAAO,EAAW,CAAO,EAGpD,GAAI,MAAM,QAAQ,CAAK,EACrB,OAAO,EAAM,IAAI,CAAC,IAAS,EAAiB,EAAM,EAAW,CAAO,CAAC,EAGvE,GAAI,IAAU,MAAQ,OAAO,IAAU,SAAU,CAC/C,IAAM,EAAkC,CAAC,EACzC,QAAY,EAAK,KAAQ,OAAO,QAAQ,CAAgC,EACtE,EAAO,GAAO,EAAiB,EAAK,EAAW,CAAO,EAExD,OAAO,EAGT,OAAO,EAMT,eAAe,CAAwB,CACrC,EACA,EACA,EACY,CACZ,GAAI,OAAO,IAAU,SACnB,OAAQ,MAAM,GAAuB,EAAO,EAAW,CAAO,EAGhE,GAAI,MAAM,QAAQ,CAAK,EACrB,OAAQ,MAAM,QAAQ,IACpB,EAAM,IAAI,CAAC,IAAS,EAAsB,EAAM,EAAW,CAAO,CAAC,CACrE,EAGF,GAAI,IAAU,MAAQ,OAAO,IAAU,SAAU,CAC/C,IAAM,EAAkC,CAAC,EACnC,EAAU,OAAO,QAAQ,CAAgC,EACzD,EAAkB,MAAM,QAAQ,IACpC,EAAQ,IAAI,OAAQ,EAAK,KAAS,CAAC,EAAK,MAAM,EAAsB,EAAK,EAAW,CAAO,CAAC,CAAC,CAC/F,EACA,QAAY,EAAK,KAAQ,EACvB,EAAO,GAAiB,EAE1B,OAAO,EAGT,OAAO,EAOF,SAAS,EAAc,CAC5B,EACA,EACA,EAA8B,CAAC,EAC5B,CACH,OAAO,EAAiB,EAAQ,EAAW,CAAO,EAM7C,SAAS,CAAkB,CAAC,EAA8B,CAAC,EAAiB,CACjF,MAAO,MACC,YAAc,CAAC,EAAW,EAAgD,CAC9E,OAAO,EAAsB,EAAQ,EAAW,CAAO,EAE3D,EC/VF,IAAM,GAAoB,wBAa1B,SAAS,EAAU,CAAC,EAAc,EAA+B,CAE/D,GACE,GACA,CAAC,EAAY,SAAS,WAAW,GACjC,CAAC,EAAY,SAAS,uBAAuB,EAE7C,MAAO,GAIT,IAAM,EAAQ,EAAK,MAAM,OAAO,EAAE,OAAO,CAAC,IAAM,EAAE,KAAK,CAAC,EACxD,GAAI,EAAM,SAAW,EAAG,MAAO,GAG/B,GAAI,EAAM,SAAW,GAAK,EAAM,IAAI,SAAS,GAAG,EAC9C,MAAO,GAIT,OAAO,EAAM,MAAM,CAAC,IAAS,GAAkB,KAAK,CAAI,CAAC,EAM3D,SAAS,EAAa,CAAC,EAA2B,CAChD,IAAM,EAAsB,CAAC,EACvB,EAAQ,EAAK,MAAM,OAAO,EAAE,OAAO,CAAC,IAAM,EAAE,KAAK,CAAC,EAExD,QAAW,KAAQ,EAAO,CACxB,IAAM,EAAQ,EAAK,MAAM,EAAiB,EAC1C,GAAI,CAAC,EAAO,SAEZ,IAAM,EAAU,EAAM,GAChB,EAAW,EAAM,GACvB,GAAI,CAAC,EAAS,SAEd,IAAM,EAAO,EAAQ,KAAK,EACtB,GAAS,GAAY,IAAI,KAAK,EAKlC,GAFe,EAAM,WAAW,KAAK,GAAK,EAAM,WAAW,KAAK,EAEpD,CAEV,EAAQ,EAAM,MAAM,CAAC,EAGrB,IAAI,EAAW,EACX,EAEE,EAAY,EAAM,QAAQ,KAAK,EACrC,GAAI,IAAc,GAChB,EAAW,EAAM,MAAM,EAAG,CAAS,EAAE,KAAK,EAC1C,EAAW,EAAM,MAAM,EAAY,CAAC,EAAE,KAAK,EAG7C,EAAO,KACL,EAAuB,CACrB,OACA,MAAO,GACP,OAAQ,GACR,KAAM,CACR,CAAC,EACE,UAAU,WAAY,CAAQ,EAC9B,MAAM,CACX,EAEA,OAAO,KAAK,CAAE,OAAM,QAAO,OAAQ,EAAM,CAAC,EAI9C,OAAO,EAMF,SAAS,CAAK,CAAC,EAAkC,CACtD,IAAM,EAA4B,CAAC,EAK7B,EAAQ,EAAQ,MAAM,OAAO,EAC7B,EAAoD,CAAC,EACvD,EAAmD,CAAE,MAAO,CAAC,CAAE,EAEnE,QAAW,KAAQ,EAAO,CACxB,IAAM,EAAiB,EAAK,MAAM,cAAc,EAChD,GAAI,EAAgB,CAElB,GAAI,EAAa,MAAM,OAAS,GAAK,EAAO,SAAW,EACrD,EAAO,KAAK,CAAY,EAG1B,IAAM,EAAY,EAAe,IAAI,KAAK,EAC1C,EAAe,EAAgD,CAAE,MAAO,CAAC,CAAE,CAAC,EACzE,UAAU,OAAQ,GAAa,MAAS,EACxC,MAAM,EAET,OAAa,MAAM,KAAK,CAAI,EAKhC,EAAO,KAAK,CAAY,EAGxB,QAAW,KAAS,EAAQ,CAC1B,GAAI,EAAM,MAAM,SAAW,GAAK,CAAC,EAAM,KACrC,SAGF,IAAM,EAAe,EAAM,MAAM,KAAK;AAAA,CAAI,EACpC,EAAU,GAAkB,EAAc,EAAM,IAAI,EAC1D,GAAI,EACF,EAAS,KAAK,CAAO,EAIzB,OAAO,EAMT,SAAS,EAAiB,CAAC,EAAe,EAA4C,CACpF,IAAM,EAAQ,EAAM,MAAM,OAAO,EAC3B,EAA+B,CAAC,EAClC,EAAO,EACP,EAAS,GACT,EAAM,GACJ,EAAkC,CAAC,EACrC,EACA,EACA,EACA,EAAS,GACP,EAAsB,CAAC,EACzB,EAAmB,GAEvB,QAAS,EAAI,EAAG,EAAI,EAAM,OAAQ,IAAK,CACrC,IAAM,EAAO,EAAM,GACnB,GAAI,IAAS,OAAW,SACxB,IAAM,EAAc,EAAK,KAAK,EAG9B,GAAI,CAAC,GAAoB,IAAgB,GACvC,SAIF,GAAI,CAAC,IAAqB,EAAY,WAAW,GAAG,GAAK,EAAY,WAAW,IAAI,GAAI,CAItF,IAAM,EAHiB,EAAY,QAAQ,eAAgB,EAAE,EAGvB,MAAM,kBAAkB,EAC9D,GAAI,EAAgB,CAClB,KAAS,EAAW,GAAS,EAC7B,GAAI,IAAc,OAChB,EAAO,GAAO,KAAK,GAAK,EACnB,QAAI,EACT,EAAK,GAAa,GAAO,KAAK,GAAK,GAGvC,SAIF,GAAI,CAAC,EAAkB,CACrB,IAAM,EAAmB,EAAY,MACnC,qFACF,EACA,GAAI,EAAkB,CACpB,IAAM,EAAc,EAAiB,GAC/B,EAAW,EAAiB,GAClC,GAAI,CAAC,GAAe,CAAC,EAAU,SAC/B,EAAS,EAAY,YAAY,EACjC,EAAM,EACN,EAAmB,GACnB,SAIF,IAAM,EAAqB,EAAY,MACrC,kEACF,EACA,GAAI,EAAoB,CACtB,IAAM,EAAc,EAAmB,GACjC,EAAW,EAAmB,GACpC,GAAI,CAAC,GAAe,CAAC,EAAU,SAC/B,EAAS,EAAY,YAAY,EACjC,EAAM,EAAS,KAAK,EACpB,EAAmB,GACnB,SAIF,SAIF,GAAI,EACF,EAAU,KAAK,CAAI,EACd,QAAI,IAAgB,GAEzB,EAAS,GACJ,KAEL,IAAM,EAAc,EAAK,MAAM,mBAAmB,EAClD,GAAI,EAAa,CACf,IAAM,EAAa,EAAY,GAC/B,GAAI,CAAC,EAAY,SACjB,IAAM,EAAc,EAAY,IAAM,GACtC,EAAQ,EAAW,KAAK,GAAK,EAAY,KAAK,IAMpD,GAAI,CAAC,GAAU,CAAC,EACd,OAAO,KAIT,GAAI,EAAU,OAAS,EAAG,CAExB,MAAO,EAAU,OAAS,GAAK,EAAU,EAAU,OAAS,IAAI,KAAK,IAAM,GACzE,EAAU,IAAI,EAIhB,GAAI,EAAU,SAAW,EAAG,CAC1B,IAAM,EAAe,EAAU,IAAI,KAAK,EAAE,MAAM,YAAY,EAC5D,GAAI,EAAc,CAChB,IAAM,EAAW,EAAa,GAC9B,GAAI,EACF,EAAW,CAAE,KAAM,CAAS,EAEzB,QAAI,EAAU,IAAI,KAAK,EAC5B,EAAO,EAAU,GAEd,QAAI,EAAU,OAAS,EAC5B,EAAO,EAAU,KAAK;AAAA,CAAI,EAM9B,IAAM,EAAc,EAAQ,iBAAmB,EAAQ,gBAEvD,GAAI,GAAQ,GAAW,EAAM,CAAW,EACtC,EAAW,GAAc,CAAI,EAC7B,EAAO,OAGT,OAAO,EAA2B,CAChC,SACA,MACA,UACA,IAAK,EACL,MACF,CAAC,EACE,UAAU,OAAQ,CAAI,EACtB,UAAU,OAAQ,CAAI,EACtB,UAAU,WAAY,CAAQ,EAC9B,UAAU,WAAY,CAAQ,EAC9B,MAAM,EAMX,eAAsB,EAAS,CAAC,EAAwC,CACtE,OAAO,MAAM,EAAgB,CAAI,EAOnC,eAAsB,CAAe,CAAC,EAAc,EAAmC,CACrF,GAAI,EAAI,CACN,IAAM,EAAU,MAAM,EAAG,SAAS,CAAI,EACtC,OAAO,EAAM,CAAO,EAGtB,GAAI,OAAQ,WAAuC,IAAW,IAAa,CAMzE,IAAM,EAAU,MAFH,WAA6C,IACzC,KAAK,CAAI,EACC,KAAK,EAChC,OAAO,EAAM,CAAO,EAGtB,MAAU,MAAM,wEAAwE,ECzR1F,SAAS,CAAI,CAAC,EAAgC,EAA0B,CACtE,IAAU,CAAK,EAGjB,SAAS,EAAe,CAAC,EAAU,EAAgB,CACjD,IAAM,EAAQ,EAAI,GAClB,GAAI,CAAC,EACH,MAAU,MAAM,CAAG,EAErB,OAAO,EAGT,SAAS,EAAe,CAAC,EAAmB,CAC1C,IAAM,EAAM,KAAK,IAAI,EAAE,YAAY,GAAG,EAAG,EAAE,YAAY,IAAI,CAAC,EAC5D,GAAI,IAAQ,GAAI,MAAO,IACvB,OAAO,IAAQ,EAAI,EAAE,MAAM,EAAG,CAAC,EAAI,EAAE,MAAM,EAAG,CAAG,EAGnD,SAAS,EAAc,CAAC,EAAoB,CAC1C,GAAI,EAAE,WAAW,GAAG,GAAK,EAAE,WAAW,IAAI,EAAG,MAAO,GACpD,GAAI,kBAAkB,KAAK,CAAC,EAAG,MAAO,GACtC,GAAI,EAAE,WAAW,MAAM,EAAG,MAAO,GACjC,MAAO,GAGT,SAAS,EAAW,CAAC,EAAW,EAAmB,CACjD,GAAI,CAAC,EAAG,OAAO,EACf,GAAI,CAAC,EAAG,OAAO,EACf,IAAM,EAAM,EAAE,SAAS,IAAI,EAAI,KAAO,IAChC,EAAQ,EAAE,SAAS,GAAG,GAAK,EAAE,SAAS,IAAI,EAAI,EAAE,MAAM,EAAG,EAAE,EAAI,EAC/D,EAAQ,EAAE,WAAW,GAAG,GAAK,EAAE,WAAW,IAAI,EAAI,EAAE,MAAM,CAAC,EAAI,EACrE,MAAO,GAAG,IAAQ,IAAM,IAG1B,SAAS,EAAe,CAAC,EAAkB,EAAiB,CAC1D,GAAI,EACF,OAAO,EAAG,KAAK,QAAQ,EAAG,KAAK,QAAQ,CAAQ,CAAC,EAGlD,IAAM,EACH,WAA+D,SAAS,MAAM,GAAK,IAChF,EAAW,GAAe,CAAQ,EAAI,EAAW,GAAY,EAAK,CAAQ,EAChF,OAAO,GAAgB,CAAQ,EAGjC,SAAS,EAAgB,CACvB,EACA,EACwB,CACxB,GAAI,CAAC,EAAQ,OAAO,EACpB,IAAM,EAAW,EAAQ,QAAa,EAAQ,QAAa,GAC3D,MAAO,IACF,EACH,OAAQ,EAAW,GAAG,MAAa,IAAW,CAChD,EAGK,SAAS,CAAY,CAAC,EAAuB,CAAC,EAAW,CAC9D,IAAM,EAAY,EAAO,WAAa,EAAoB,EACpD,EAAK,EAAO,GACZ,EAAc,EAAO,YACrB,EAAU,EAAO,QACjB,EAAiB,EAAO,eAExB,EAAe,EAAmB,CAAE,UAAW,EAAO,WAAa,CAAC,CAAE,CAAC,EAE7E,eAAe,CAAiB,CAC9B,EACA,EAIC,CACD,EAAK,EAAS,CAAE,KAAM,eAAgB,OAAQ,QAAS,CAAC,EACxD,IAAM,EAAW,EAAM,CAAQ,EAC/B,EAAK,EAAS,CAAE,KAAM,gBAAiB,OAAQ,SAAU,aAAc,EAAS,MAAO,CAAC,EAExF,IAAM,EAAU,GAAa,EAAU,8CAA8C,EAErF,EAAK,EAAS,CAAE,KAAM,oBAAqB,CAAC,EAC5C,IAAM,EAAa,IAAM,EAAQ,WAAa,CAAC,CAAG,EAC5C,EAAe,MAAM,EAAa,YAAY,EAAS,CAAU,EACvE,EAAK,EAAS,CAAE,KAAM,qBAAsB,CAAC,EAE7C,EAAK,EAAS,CAAE,KAAM,gBAAiB,CAAC,EACxC,IAAM,EACJ,EAAQ,UACR,GAAI,IAAI,GACP,WAA+D,SAAS,MAAM,GAC/E,KAEM,kBAAmB,MAAM,GAC/B,EACA,EAAoF,CAClF,UACF,CAAC,EACE,UAAU,KAAM,CAAE,EAClB,UAAU,iBAAkB,CAAc,EAC1C,MAAM,CACX,EAGA,OAFA,EAAK,EAAS,CAAE,KAAM,iBAAkB,CAAC,EAElC,CAAE,iBAAgB,QAAS,EAAa,GAAI,EAGrD,eAAe,CAAe,CAC5B,EACA,EAIC,CACD,EAAK,EAAS,CAAE,KAAM,eAAgB,OAAQ,MAAO,CAAC,EACtD,IAAM,EAAW,MAAM,EAAgB,EAAM,CAAE,EAC/C,EAAK,EAAS,CAAE,KAAM,gBAAiB,OAAQ,OAAQ,aAAc,EAAS,MAAO,CAAC,EAEtF,IAAM,EAAU,GAAa,EAAU,oCAAoC,GAAM,EAEjF,EAAK,EAAS,CAAE,KAAM,oBAAqB,CAAC,EAC5C,IAAM,EAAa,IAAM,EAAQ,WAAa,CAAC,CAAG,EAC5C,EAAe,MAAM,EAAa,YAAY,EAAS,CAAU,EACvE,EAAK,EAAS,CAAE,KAAM,qBAAsB,CAAC,EAE7C,EAAK,EAAS,CAAE,KAAM,gBAAiB,CAAC,EACxC,IAAM,EAAW,GAAgB,EAAM,CAAE,GACjC,kBAAmB,MAAM,GAC/B,EACA,EAAoF,CAClF,UACF,CAAC,EACE,UAAU,KAAM,CAAE,EAClB,UAAU,iBAAkB,CAAc,EAC1C,MAAM,CACX,EAGA,OAFA,EAAK,EAAS,CAAE,KAAM,iBAAkB,CAAC,EAElC,CAAE,iBAAgB,QAAS,EAAa,GAAI,EAGrD,eAAe,CAAW,CACxB,EACA,EACA,EACmB,CACnB,IAAM,EAAU,EAAe,SAAW,CAAC,EAErC,EAAe,EAAc,MAAM,EAAY,gBAAgB,CAAa,EAAI,OAChF,EAAqB,EAAc,GAAiB,EAAS,CAAY,EAAI,EAE7E,EAAqC,IACtC,EACH,QAAS,CACX,EAEA,EAAK,EAAS,CAAE,KAAM,eAAgB,OAAQ,EAAmB,OAAQ,IAAK,CAAc,CAAC,EAE7F,GAAI,CACF,IAAM,EAAc,EAA4B,CAAC,CAAC,EAC/C,UAAU,UAAW,EAAQ,SAAS,EACtC,UAAU,SAAU,EAAQ,MAAM,EAClC,UAAU,kBAAmB,EAAQ,eAAe,EACpD,UAAU,cAAe,EAAQ,WAAW,EAC5C,UAAU,QAAS,EAAQ,KAAK,EAChC,MAAM,EAEH,EAAW,MAAM,EAAqB,EAAoB,EAAa,CAAS,EAStF,GAPA,EAAK,EAAS,CACZ,KAAM,gBACN,OAAQ,EAAmB,OAC3B,IAAK,EACL,OAAQ,EAAS,MACnB,CAAC,EAEG,EACF,MAAM,EAAY,gBAAgB,EAAe,CAAQ,EAG3D,OAAO,EACP,MAAO,EAAG,CAMV,MALA,EAAK,EAAS,CACZ,KAAM,QACN,MAAO,QACP,QAAS,aAAa,MAAQ,EAAE,QAAU,OAAO,CAAC,CACpD,CAAC,EACK,GAIV,MAAO,CACL,YAAa,OACP,UAAS,CAAC,EAAU,EAAU,CAAC,EAAG,CACtC,IAAQ,iBAAgB,WAAY,MAAM,EAAkB,EAAU,CAAO,EAC7E,OAAO,MAAM,EAAY,EAAgB,EAAS,CAAO,QAErD,QAAO,CAAC,EAAM,EAAU,CAAC,EAAG,CAChC,IAAQ,iBAAgB,WAAY,MAAM,EAAgB,EAAM,CAAO,EACvE,OAAO,MAAM,EAAY,EAAgB,EAAS,CAAO,EAE7D,EAGF,eAAe,EAAqB,CAClC,EAQA,EAC6C,CAC7C,IAAM,EAAkC,IAClC,EAAI,gBAAkB,CAAC,KACvB,EAAa,SAAW,CAAC,CAC/B,EAEI,EAA+B,EAAa,KAEhD,GAAI,EAAa,SAAU,CACzB,IAAM,EAAa,MAAM,EACvB,EAAa,SAAS,KACtB,EAA2C,CAAE,SAAU,EAAI,QAAS,CAAC,EAClE,UAAU,KAAM,EAAI,EAAE,EACtB,MAAM,CACX,EAIA,GAFA,EAAO,EAAW,QAEd,CAAC,EAAQ,iBAAmB,CAAC,EAAQ,gBACvC,EAAQ,gBAAkB,EAAW,SAElC,QAAI,EAAa,UAAY,EAAa,SAAS,OAAS,GAGjE,GAFiB,EAAc,EAAa,QAAQ,EAGlD,EAAO,MAAM,EACX,EAAa,SACb,EAA2C,CAAE,SAAU,EAAI,QAAS,CAAC,EAClE,UAAU,KAAM,EAAI,EAAE,EACtB,MAAM,CACX,EACA,OAAO,EAAQ,gBACf,OAAO,EAAQ,gBAGf,QADA,EAAO,EAAgB,EAAa,QAAQ,EACxC,CAAC,EAAQ,iBAAmB,CAAC,EAAQ,gBACvC,EAAQ,gBAAkB,oCAYhC,MAAO,CAAE,eAP8B,CACrC,OAAQ,EAAa,OACrB,IAAK,EAAa,IAClB,aACI,IAAS,OAAY,CAAE,MAAK,EAAI,CAAC,CACvC,CAEwB,ECpQnB,IAAM,EAAkB,OAAO,gBAAgB,EA+C/C,SAAS,CAAS,CAAC,EAAkC,CAC1D,GAAI,OAAO,QAAY,KAAe,QAAQ,IAC5C,OAAO,QAAQ,IAAI,GAErB,OAgBK,SAAS,EAAkB,CAAC,EAAoC,CACrE,IAAQ,aAAc,EAChB,EAAiB,EAAU,cAAc,EACzC,EAAsB,EAAU,iBAAiB,EAKjD,EAJW,EAAU,YAAY,GAIb,EAAO,MAE3B,EAAU,EAAU,QAAQ,MAAO,EAAE,EACvC,EAAqC,IAAK,EAAO,SAAU,EAC3D,EAAgC,EAChC,EAA6B,EAC7B,EAAc,GACd,EAGE,EAAiB,CAAC,CAAC,EACnB,EAAuB,CAAC,CAAC,EAG3B,EAAmC,QAAQ,QAAQ,EAEjD,EAAiB,EAAO,SAAW,MAEnC,EAAkC,CACtC,eAAgB,kBAClB,EACA,GAAI,EACF,EAAQ,cAAmB,UAAU,IAIvC,eAAe,CAAiB,EAAkB,CAChD,GAAI,EAAa,OACjB,GAAI,EAAa,OAAO,EAkDxB,OAhDA,GAAe,SAAY,CAGzB,GAAI,EAAsB,CAIxB,EAAc,GACd,OAIF,IAAM,EAAa,MAAM,MAAM,GAAG,YAAmB,CACnD,OAAQ,OACR,UACA,KAAM,KAAK,UAAU,CAAE,WAAU,CAAC,CACpC,CAAC,EAED,GAAI,CAAC,EAAW,GAAI,CAClB,IAAM,EAAQ,MAAM,EAAW,KAAK,EACpC,MAAU,MAAM,6BAA6B,GAAO,EAOtD,GAHA,GADqB,MAAM,EAAW,KAAK,GACnB,UAGpB,CAAC,EAAgB,CACnB,IAAM,EAAU,MAAM,MAAM,GAAG,UAAiB,CAC9C,OAAQ,OACR,UACA,KAAM,KAAK,UAAU,CACnB,WACF,CAAC,CACH,CAAC,EAED,GAAI,CAAC,EAAQ,GAAI,CACf,IAAM,EAAQ,MAAM,EAAQ,KAAK,EACjC,MAAU,MAAM,0BAA0B,GAAO,EAInD,GADkB,MAAM,EAAQ,KAAK,GACnB,OAGpB,EAAc,KACb,EAEI,EAGT,SAAS,CAAmB,CAAC,EAAqC,CAChE,IAAM,EAAS,SAA2B,CACxC,GAAI,CAAC,EAAW,OAChB,GAAI,CACF,IAAM,EAAM,MAAM,MAAM,GAAG,aAAmB,cAAuB,CACnE,OAAQ,MACR,UACA,KAAM,KAAK,UAAU,CAAE,UAAW,EAAM,KAAM,OAAQ,CAAC,CACzD,CAAC,EACD,GAAI,CAAC,EAAI,GACP,QAAQ,MAAM,sCAAuC,EAAI,UAAU,EAErE,MAAO,EAAK,CACZ,QAAQ,MAAM,sCAAuC,CAAG,IAI5D,GAAI,EACF,EAAoB,EAAkB,MAAM,IAAG,CAAG,OAAS,EAAE,KAAK,CAAM,EACnE,QAAI,EACT,EAAoB,EACjB,MAAM,IAAG,CAAG,OAAS,EACrB,KAAK,IAAM,CAAW,EACtB,KAAK,CAAM,EAIlB,eAAe,CAAc,CAC3B,EACA,EAAsB,CAAC,EACJ,CACnB,MAAM,EAAkB,EAGxB,MAAM,EAEN,IAAM,EAAgC,IACjC,EACH,YACA,SACA,QAAS,EAAO,QAChB,UAAW,EAAQ,SAAW,CAChC,EAGA,GAAI,EAAO,KACT,EAAK,SAAc,EAAO,KAI5B,GAAI,EAAQ,WAAa,OAAO,KAAK,EAAQ,SAAS,EAAE,OAAS,EAC/D,EAAK,UAAe,EAAQ,UAI9B,GAAI,EAAQ,SACV,EAAK,SAAc,EAAQ,SAI7B,IAAM,EAA4B,CAChC,OAAQ,OACR,UACA,KAAM,KAAK,UAAU,CAAI,CAC3B,EACA,GAAI,EAAQ,OACV,EAAa,OAAS,EAAQ,OAGhC,IAAM,EAAU,MAAM,MAAM,GAAG,YAAmB,CAAY,EAE9D,GAAI,CAAC,EAAQ,GAAI,CACf,IAAM,GAAQ,MAAM,EAAQ,KAAK,EACjC,MAAU,MAAM,qBAAqB,IAAO,EAG9C,IAAM,EAAQ,MAAM,EAAQ,KAAK,EAGjC,OAAO,EAA8B,EAAK,QAAQ,EAGpD,SAAS,CAA6B,CAAC,EAAkD,CAEvF,IAAI,EACJ,GAAI,EAAU,OAAS,OACrB,GAAI,EAAU,WAAa,SAAU,CAEnC,IAAM,EAAY,KAAK,EAAU,IAAI,EAC/B,EAAQ,IAAI,WAAW,EAAU,MAAM,EAC7C,QAAS,EAAI,EAAG,EAAI,EAAU,OAAQ,IACpC,EAAM,GAAK,EAAU,WAAW,CAAC,EAEnC,EAAO,EAEP,OAAO,EAAU,KAKrB,IAAM,EAAkB,IAAI,QAC5B,QAAW,KAAK,EAAU,QACxB,EAAgB,OAAO,EAAE,KAAM,EAAE,KAAK,EAGxC,OAAO,IAAI,SAAS,EAAM,CACxB,OAAQ,EAAU,OAClB,WAAY,EAAU,WACtB,QAAS,CACX,CAAC,EAGH,eAAe,CAAK,EAAkB,CAEpC,GAAI,CAAC,GAAU,EAAgB,OAE/B,GAAI,CACF,MAAM,MAAM,GAAG,WAAiB,WAAiB,CAC/C,OAAQ,OACR,SACF,CAAC,EACD,KAAM,GAwCV,MAnCqE,MAC7D,IAAG,CAAC,EAAc,EAAsB,CAAC,EAAsB,CACnE,OAAO,EAAe,CAAE,MAAK,EAAG,CAAO,QAGnC,UAAS,CAAC,EAAiB,EAAsB,CAAC,EAAsB,CAC5E,OAAO,EAAe,CAAE,SAAQ,EAAG,CAAO,GAG5C,YAAY,CAAC,EAAqC,CAChD,EAAY,IAAK,KAAc,CAAK,EACpC,EAAoB,CAAI,GAG1B,WAAW,CAAC,EAAa,EAAsB,CAC7C,EAAU,GAAO,EACjB,EAAoB,EAAG,GAAM,CAAM,CAAC,GAGtC,YAAY,EAA4B,CACtC,MAAO,IAAK,CAAU,GAGxB,SAEC,OAAO,cAAe,GAGtB,GAAkB,KAAO,CACxB,UAAW,EACX,YACA,QACF,EACF,ECtVK,SAAS,EAAY,CAAC,EAAuB,CAAC,EAAW,CAE9D,IAAM,EAAY,EAAO,QAAU,EAAU,aAAa,EAC1D,GAAI,EACF,OAAO,GAAmB,IACrB,EACH,YACA,MAAO,EAAO,aAAe,EAAU,YAAY,CACrD,CAAC,EAIH,IAAI,EAAqC,IAAK,EAAO,SAAU,EACzD,EAAY,EAAO,UACnB,EAAiB,EAAO,SAAW,MACnC,EAAW,EAAO,UAAY,CAAC,EAE/B,EAAc,EAAY,GAAiB,CAAS,EAAI,OAExD,EAAS,EACb,EAA0B,CAAC,CAAiB,EACzC,UAAU,KAAM,EAAO,EAAE,EACzB,UAAU,YAAa,EAAO,SAAS,EACvC,UAAU,cAAe,CAAW,EACpC,UAAU,YAAa,EAAO,SAAS,EACvC,UAAU,UAAW,EAAO,OAAO,EACnC,UAAU,iBAAkB,EAAS,OAAO,EAC5C,MAAM,CACX,EAEA,SAAS,CAAU,CAAC,EAA4D,CAC9E,MAAO,IAAK,KAAc,CAAQ,EAIpC,eAAe,CAAK,EAAkB,EAItC,MAAO,MACC,IAAG,CAAC,EAAkB,EAAsB,CAAC,EAAsB,CACvE,OAAO,MAAM,EAAO,QAClB,EACA,EAA8B,CAC5B,UAAW,EAAW,EAAQ,SAAS,EACvC,UAAW,EAAQ,SAAW,CAChC,CAAC,EACE,UAAU,SAAU,EAAQ,MAAM,EAClC,UAAU,kBAAmB,EAAS,eAAe,EACrD,UAAU,cAAe,EAAS,WAAW,EAC7C,UAAU,QAAS,EAAS,KAAK,EACjC,MAAM,CACX,QAGI,UAAS,CAAC,EAAiB,EAAsB,CAAC,EAAsB,CAC5E,OAAO,MAAM,EAAO,UAClB,EACA,EAA8B,CAC5B,UAAW,EAAW,EAAQ,SAAS,EACvC,UAAW,EAAQ,SAAW,CAChC,CAAC,EACE,UAAU,SAAU,EAAQ,MAAM,EAClC,UAAU,kBAAmB,EAAS,eAAe,EACrD,UAAU,cAAe,EAAS,WAAW,EAC7C,UAAU,QAAS,EAAS,KAAK,EACjC,UAAU,WAAY,EAAQ,QAAQ,EACtC,MAAM,CACX,GAGF,YAAY,CAAC,EAAqC,CAChD,EAAY,IAAK,KAAc,CAAK,GAGtC,WAAW,CAAC,EAAa,EAAsB,CAC7C,EAAU,GAAO,GAGnB,YAAY,EAA4B,CACtC,MAAO,IAAK,CAAU,GAGxB,SAEC,OAAO,cAAe,CACzB,EAGF,SAAS,EAAgB,CAAC,EAAmC,CAC3D,MAAO,CACL,eAAe,CAAC,EAAa,CAC3B,GAAI,CAEF,OADU,EAAU,oBAAoB,CAAG,GAC/B,OACZ,KAAM,CACN,SAGJ,eAAe,CAAC,EAAa,EAAoB,CAC/C,IAAM,EAAmB,GAAoB,CAAQ,EACrD,QAAW,KAAU,EACnB,EAAU,cAAc,EAAQ,EAAK,CAAE,YAAa,EAAK,CAAC,EAGhE,EAGF,SAAS,EAAmB,CAAC,EAA8B,CAEzD,IAAM,EADa,EAAS,QACQ,eAAe,EACnD,GAAI,MAAM,QAAQ,CAAgB,GAAK,EAAiB,OAAS,EAAG,OAAO,EAE3E,IAAM,EAAS,EAAS,QAAQ,IAAI,YAAY,EAChD,OAAO,EAAS,CAAC,CAAM,EAAI,CAAC,ECrFvB,SAAS,EAAiB,CAAC,EAA4C,CAE5E,IAAM,EADe,EACS,GAC9B,GAAI,OAAO,IAAa,WACtB,OAAO,EAAS,EAElB",
|
|
20
|
+
"debugId": "26111198FAA8274364756E2164756E21",
|
|
19
21
|
"names": []
|
|
20
22
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"interpolate.d.ts","sourceRoot":"","sources":["../src/interpolate.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,kBAAkB,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;
|
|
1
|
+
{"version":3,"file":"interpolate.d.ts","sourceRoot":"","sources":["../src/interpolate.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,kBAAkB,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AAgVhE;;;GAGG;AACH,wBAAgB,WAAW,CAAC,CAAC,EAC3B,MAAM,EAAE,CAAC,EACT,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAClC,OAAO,GAAE,kBAAuB,GAC/B,CAAC,CAEH;AAED;;GAEG;AACH,wBAAgB,kBAAkB,CAAC,OAAO,GAAE,kBAAuB,GAAG,YAAY,CAMjF"}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import type { CommandResolverDef } from '../config/types';
|
|
2
|
+
import type { Resolver } from '../types';
|
|
3
|
+
/**
|
|
4
|
+
* Create a Resolver function from a CommandResolverDef.
|
|
5
|
+
*
|
|
6
|
+
* The created resolver executes the external command with the NDJSON protocol:
|
|
7
|
+
* - Writes `{"resolver":"name","args":["arg1","arg2"]}` to stdin
|
|
8
|
+
* - Expects `{"value":"result"}` on stdout
|
|
9
|
+
*
|
|
10
|
+
* @param def - The command resolver definition
|
|
11
|
+
* @param name - The resolver name (used for error messages)
|
|
12
|
+
* @param projectRoot - The project root directory (used as cwd)
|
|
13
|
+
*/
|
|
14
|
+
export declare function createCommandResolver(def: CommandResolverDef, projectRoot: string, name?: string): Resolver;
|
|
15
|
+
/**
|
|
16
|
+
* Check if a resolver definition is a command resolver.
|
|
17
|
+
*/
|
|
18
|
+
export declare function isCommandResolverDef(def: unknown): def is CommandResolverDef;
|
|
19
|
+
//# sourceMappingURL=command.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"command.d.ts","sourceRoot":"","sources":["../../src/resolver/command.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,iBAAiB,CAAC;AAC1D,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AAsLzC;;;;;;;;;;GAUG;AACH,wBAAgB,qBAAqB,CACnC,GAAG,EAAE,kBAAkB,EACvB,WAAW,EAAE,MAAM,EACnB,IAAI,CAAC,EAAE,MAAM,GACZ,QAAQ,CAKV;AAED;;GAEG;AACH,wBAAgB,oBAAoB,CAAC,GAAG,EAAE,OAAO,GAAG,GAAG,IAAI,kBAAkB,CAI5E"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/resolver/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,qBAAqB,EACrB,oBAAoB,EACrB,MAAM,WAAW,CAAC"}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import{createRequire as A}from"node:module";var g=A(import.meta.url);import{spawn as O}from"node:child_process";var B=2000,S=500,E=1048576,M=65536;async function R(V,z,C,F){let w=V.timeoutMs??B;if(V.command.length===0)throw Error(`Resolver "${z}" has empty command array`);let[b,...U]=V.command;if(!b)throw Error(`Resolver "${z}" has empty command`);return new Promise((v,$)=>{let W=O(b,U,{cwd:F,stdio:["pipe","pipe","pipe"],env:process.env}),H="",q="",X=0,Y=0,G=!1,N=!1,x=!1,D=!1,I=setTimeout(()=>{D=!0,W.kill("SIGTERM"),setTimeout(()=>{if(!x)W.kill("SIGKILL")},S)},w);W.stdout?.on("data",(J)=>{let K=E-X;if(K>0){let Q=J.slice(0,K);if(H+=Q.toString("utf-8"),X+=Q.length,Q.length<J.length)G=!0}else G=!0}),W.stderr?.on("data",(J)=>{let K=M-Y;if(K>0){let Q=J.slice(0,K);if(q+=Q.toString("utf-8"),Y+=Q.length,Q.length<J.length)N=!0}else N=!0}),W.on("close",(J,K)=>{if(clearTimeout(I),x=!0,D){$(Error(`Resolver "${z}" timed out after ${w}ms${q?`: ${q}`:""}`));return}if(J!==0){let Z=K?`killed by signal ${K}`:`exit code ${J}`;$(Error(`Resolver "${z}" failed (${Z})${q?`: ${q}`:""}`));return}let Q=H.trim();if(!Q){$(Error(`Resolver "${z}" returned no output`));return}let L=Q.split(/\r?\n/).map((Z)=>Z.trim()).find((Z)=>Z.length>0)??"",P;try{P=JSON.parse(L)}catch{let Z=G?" (stdout exceeded 1MB limit and was truncated)":"";$(Error(`Resolver "${z}" returned invalid JSON: ${L.slice(0,100)}${Z}`));return}if(typeof P.value!=="string"){$(Error(`Resolver "${z}" returned no value (expected { "value": "..." })`));return}v(P.value)}),W.on("error",(J)=>{clearTimeout(I);let K=N?" (stderr exceeded 64KB limit and was truncated)":"";$(Error(`Resolver "${z}" failed to execute: ${J.message}${K}`))});let y={resolver:z,args:C};W.stdin?.write(`${JSON.stringify(y)}
|
|
2
|
+
`),W.stdin?.end()})}function p(V,z,C){let F=C??"$command";return async(...w)=>{return await R(V,F,w,z)}}function k(V){if(!V||typeof V!=="object")return!1;let z=V;return z.type==="command"&&Array.isArray(z.command)}export{k as isCommandResolverDef,p as createCommandResolver};
|
|
3
|
+
|
|
4
|
+
//# debugId=559AE90E803B432464756E2164756E21
|
|
5
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../src/resolver/command.ts"],
|
|
4
|
+
"sourcesContent": [
|
|
5
|
+
"import { spawn } from 'node:child_process';\nimport type { CommandResolverDef } from '../config/types';\nimport type { Resolver } from '../types';\n\n// ============================================================================\n// Constants\n// ============================================================================\n\nconst DEFAULT_TIMEOUT_MS = 2000;\nconst GRACE_PERIOD_MS = 500;\nconst MAX_STDOUT_BYTES = 1024 * 1024; // 1MB\nconst MAX_STDERR_BYTES = 64 * 1024; // 64KB\n\n// ============================================================================\n// Types\n// ============================================================================\n\ntype ResolverRequest = {\n resolver: string;\n args: string[];\n};\n\ntype ResolverResponse = {\n value: string;\n};\n\n// ============================================================================\n// Command Execution\n// ============================================================================\n\n/**\n * Execute a command resolver and return its value.\n */\nasync function executeCommandResolver(\n def: CommandResolverDef,\n name: string,\n args: string[],\n projectRoot: string\n): Promise<string> {\n const timeoutMs = def.timeoutMs ?? DEFAULT_TIMEOUT_MS;\n\n if (def.command.length === 0) {\n throw new Error(`Resolver \"${name}\" has empty command array`);\n }\n\n const [cmd, ...cmdArgs] = def.command;\n if (!cmd) {\n throw new Error(`Resolver \"${name}\" has empty command`);\n }\n\n return new Promise((resolve, reject) => {\n const child = spawn(cmd, cmdArgs, {\n cwd: projectRoot,\n stdio: ['pipe', 'pipe', 'pipe'],\n env: process.env\n });\n\n let stdout = '';\n let stderr = '';\n let stdoutBytes = 0;\n let stderrBytes = 0;\n let stdoutTruncated = false;\n let stderrTruncated = false;\n let killed = false;\n let timedOut = false;\n\n // Set up timeout\n const timeoutHandle = setTimeout(() => {\n timedOut = true;\n child.kill('SIGTERM');\n\n // Grace period before SIGKILL\n setTimeout(() => {\n if (!killed) {\n child.kill('SIGKILL');\n }\n }, GRACE_PERIOD_MS);\n }, timeoutMs);\n\n // Handle stdout\n child.stdout?.on('data', (data: Buffer) => {\n const remaining = MAX_STDOUT_BYTES - stdoutBytes;\n if (remaining > 0) {\n const chunk = data.slice(0, remaining);\n stdout += chunk.toString('utf-8');\n stdoutBytes += chunk.length;\n if (chunk.length < data.length) {\n stdoutTruncated = true;\n }\n } else {\n stdoutTruncated = true;\n }\n });\n\n // Handle stderr\n child.stderr?.on('data', (data: Buffer) => {\n const remaining = MAX_STDERR_BYTES - stderrBytes;\n if (remaining > 0) {\n const chunk = data.slice(0, remaining);\n stderr += chunk.toString('utf-8');\n stderrBytes += chunk.length;\n if (chunk.length < data.length) {\n stderrTruncated = true;\n }\n } else {\n stderrTruncated = true;\n }\n });\n\n // Handle close\n child.on('close', (code, signal) => {\n clearTimeout(timeoutHandle);\n killed = true;\n\n if (timedOut) {\n reject(\n new Error(\n `Resolver \"${name}\" timed out after ${timeoutMs}ms${stderr ? `: ${stderr}` : ''}`\n )\n );\n return;\n }\n\n if (code !== 0) {\n const exitInfo = signal ? `killed by signal ${signal}` : `exit code ${code}`;\n reject(new Error(`Resolver \"${name}\" failed (${exitInfo})${stderr ? `: ${stderr}` : ''}`));\n return;\n }\n\n // Parse output as NDJSON\n const trimmed = stdout.trim();\n if (!trimmed) {\n reject(new Error(`Resolver \"${name}\" returned no output`));\n return;\n }\n\n // Take first non-empty line (NDJSON protocol) and tolerate CRLF\n const firstLine =\n trimmed\n .split(/\\r?\\n/)\n .map((l) => l.trim())\n .find((l) => l.length > 0) ?? '';\n\n let response: ResolverResponse;\n try {\n response = JSON.parse(firstLine) as ResolverResponse;\n } catch {\n const suffix = stdoutTruncated ? ' (stdout exceeded 1MB limit and was truncated)' : '';\n reject(\n new Error(`Resolver \"${name}\" returned invalid JSON: ${firstLine.slice(0, 100)}${suffix}`)\n );\n return;\n }\n\n if (typeof response.value !== 'string') {\n reject(new Error(`Resolver \"${name}\" returned no value (expected { \"value\": \"...\" })`));\n return;\n }\n\n resolve(response.value);\n });\n\n // Handle error\n child.on('error', (err) => {\n clearTimeout(timeoutHandle);\n const suffix = stderrTruncated ? ' (stderr exceeded 64KB limit and was truncated)' : '';\n reject(new Error(`Resolver \"${name}\" failed to execute: ${err.message}${suffix}`));\n });\n\n // Write request to stdin\n const request: ResolverRequest = {\n resolver: name,\n args\n };\n\n child.stdin?.write(`${JSON.stringify(request)}\\n`);\n child.stdin?.end();\n });\n}\n\n// ============================================================================\n// Public API\n// ============================================================================\n\n/**\n * Create a Resolver function from a CommandResolverDef.\n *\n * The created resolver executes the external command with the NDJSON protocol:\n * - Writes `{\"resolver\":\"name\",\"args\":[\"arg1\",\"arg2\"]}` to stdin\n * - Expects `{\"value\":\"result\"}` on stdout\n *\n * @param def - The command resolver definition\n * @param name - The resolver name (used for error messages)\n * @param projectRoot - The project root directory (used as cwd)\n */\nexport function createCommandResolver(\n def: CommandResolverDef,\n projectRoot: string,\n name?: string\n): Resolver {\n const resolverName = name ?? '$command';\n return async (...args: string[]): Promise<string> => {\n return await executeCommandResolver(def, resolverName, args, projectRoot);\n };\n}\n\n/**\n * Check if a resolver definition is a command resolver.\n */\nexport function isCommandResolverDef(def: unknown): def is CommandResolverDef {\n if (!def || typeof def !== 'object') return false;\n const obj = def as Record<string, unknown>;\n return obj['type'] === 'command' && Array.isArray(obj['command']);\n}\n"
|
|
6
|
+
],
|
|
7
|
+
"mappings": "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",
|
|
8
|
+
"debugId": "559AE90E803B432464756E2164756E21",
|
|
9
|
+
"names": []
|
|
10
|
+
}
|
package/dist/runtime/index.js
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
function j(E){throw Error(`${E} is not supported by this transport. Provide a transport that supports it.`)}function P(E=fetch){return{capabilities:{proxy:!1,validateSSL:!1},async fetch(T,O,g){if(g.proxy)j("proxy");if(g.validateSSL===!1)j("validateSSL=false");return await E(T,O)}}}function
|
|
1
|
+
import{createRequire as z}from"node:module";var M=z(import.meta.url);function j(E){throw Error(`${E} is not supported by this transport. Provide a transport that supports it.`)}function P(E=fetch){return{capabilities:{proxy:!1,validateSSL:!1},async fetch(T,O,g){if(g.proxy)j("proxy");if(g.validateSSL===!1)j("validateSSL=false");return await E(T,O)}}}function A(){return typeof globalThis.Bun<"u"}function D(){if(!A())return P(fetch);return{capabilities:{proxy:!0,validateSSL:!0},async fetch(E,T,O){let g={...T??{}};if(O.proxy!==void 0)g.proxy=O.proxy;if(O.validateSSL===!1)g.tls={rejectUnauthorized:!1};return await fetch(E,g)}}}import{access as G,readFile as q}from"node:fs/promises";import*as C from"node:path";async function H(E){try{return await G(E),!0}catch{return!1}}function J(){return{cwd:()=>process.cwd(),path:{resolve:(...E)=>C.resolve(...E),dirname:(E)=>C.dirname(E),basename:(E)=>C.basename(E),extname:(E)=>C.extname(E),isAbsolute:(E)=>C.isAbsolute(E),sep:C.sep},exists:H,readText:async(E)=>await q(E,"utf8"),readBinary:async(E)=>{let T=await q(E);return T.buffer.slice(T.byteOffset,T.byteOffset+T.byteLength)}}}export{J as createNodeIO,P as createFetchTransport,D as createAutoTransport};
|
|
2
2
|
|
|
3
|
-
//# debugId=
|
|
3
|
+
//# debugId=BABB27F01082AFD864756E2164756E21
|
|
4
4
|
//# sourceMappingURL=index.js.map
|
|
@@ -6,7 +6,7 @@
|
|
|
6
6
|
"import { createFetchTransport } from './fetch-transport';\nimport type { Transport } from './types';\n\ntype BunFetchInit = RequestInit & { tls?: { rejectUnauthorized: boolean }; proxy?: string };\n\nfunction hasBun(): boolean {\n return typeof (globalThis as Record<string, unknown>)['Bun'] !== 'undefined';\n}\n\n/**\n * Auto-select a transport for the current runtime.\n *\n * - Bun: supports `proxy` and `validateSSL=false` via Bun fetch extensions.\n * - Others: falls back to baseline fetch transport (no proxy/validateSSL).\n */\nexport function createAutoTransport(): Transport {\n if (!hasBun()) {\n return createFetchTransport(fetch);\n }\n\n return {\n capabilities: { proxy: true, validateSSL: true },\n async fetch(url, init, ctx) {\n const bunInit: BunFetchInit = { ...(init ?? {}) };\n\n if (ctx.proxy !== undefined) {\n bunInit.proxy = ctx.proxy;\n }\n\n if (ctx.validateSSL === false) {\n bunInit.tls = { rejectUnauthorized: false };\n }\n\n return await fetch(url, bunInit);\n }\n };\n}\n",
|
|
7
7
|
"import { access, readFile } from 'node:fs/promises';\nimport * as path from 'node:path';\nimport type { IO } from './types';\n\nasync function exists(p: string): Promise<boolean> {\n try {\n await access(p);\n return true;\n } catch {\n return false;\n }\n}\n\n/**\n * Node IO adapter (fs + path) for running `runFile()` and file references.\n */\nexport function createNodeIO(): IO {\n return {\n cwd: () => process.cwd(),\n path: {\n resolve: (...parts) => path.resolve(...parts),\n dirname: (p) => path.dirname(p),\n basename: (p) => path.basename(p),\n extname: (p) => path.extname(p),\n isAbsolute: (p) => path.isAbsolute(p),\n sep: path.sep\n },\n exists,\n readText: async (p) => await readFile(p, 'utf8'),\n readBinary: async (p) => {\n const buf = await readFile(p);\n // Ensure we return an ArrayBuffer exactly sized to the file contents.\n // Note: Node's Buffer.buffer is never a SharedArrayBuffer, but TypeScript doesn't know that.\n return buf.buffer.slice(buf.byteOffset, buf.byteOffset + buf.byteLength) as ArrayBuffer;\n }\n };\n}\n"
|
|
8
8
|
],
|
|
9
|
-
"mappings": "
|
|
10
|
-
"debugId": "
|
|
9
|
+
"mappings": "qEAIA,SAAS,CAAW,CAAC,EAAqB,CACxC,MAAU,MACR,GAAG,6EACL,EASK,SAAS,CAAoB,CAAC,EAAuB,MAAkB,CAC5E,MAAO,CACL,aAAc,CAAE,MAAO,GAAO,YAAa,EAAM,OAC3C,MAAK,CAAC,EAAK,EAAM,EAAK,CAC1B,GAAI,EAAI,MAAO,EAAY,OAAO,EAClC,GAAI,EAAI,cAAgB,GAAO,EAAY,mBAAmB,EAC9D,OAAO,MAAM,EAAU,EAAK,CAAI,EAEpC,ECnBF,SAAS,CAAM,EAAY,CACzB,OAAO,OAAQ,WAAuC,IAAW,IAS5D,SAAS,CAAmB,EAAc,CAC/C,GAAI,CAAC,EAAO,EACV,OAAO,EAAqB,KAAK,EAGnC,MAAO,CACL,aAAc,CAAE,MAAO,GAAM,YAAa,EAAK,OACzC,MAAK,CAAC,EAAK,EAAM,EAAK,CAC1B,IAAM,EAAwB,IAAM,GAAQ,CAAC,CAAG,EAEhD,GAAI,EAAI,QAAU,OAChB,EAAQ,MAAQ,EAAI,MAGtB,GAAI,EAAI,cAAgB,GACtB,EAAQ,IAAM,CAAE,mBAAoB,EAAM,EAG5C,OAAO,MAAM,MAAM,EAAK,CAAO,EAEnC,ECnCF,iBAAS,cAAQ,yBACjB,4BAGA,eAAe,CAAM,CAAC,EAA6B,CACjD,GAAI,CAEF,OADA,MAAM,EAAO,CAAC,EACP,GACP,KAAM,CACN,MAAO,IAOJ,SAAS,CAAY,EAAO,CACjC,MAAO,CACL,IAAK,IAAM,QAAQ,IAAI,EACvB,KAAM,CACJ,QAAS,IAAI,IAAe,UAAQ,GAAG,CAAK,EAC5C,QAAS,CAAC,IAAW,UAAQ,CAAC,EAC9B,SAAU,CAAC,IAAW,WAAS,CAAC,EAChC,QAAS,CAAC,IAAW,UAAQ,CAAC,EAC9B,WAAY,CAAC,IAAW,aAAW,CAAC,EACpC,IAAU,KACZ,EACA,SACA,SAAU,MAAO,IAAM,MAAM,EAAS,EAAG,MAAM,EAC/C,WAAY,MAAO,IAAM,CACvB,IAAM,EAAM,MAAM,EAAS,CAAC,EAG5B,OAAO,EAAI,OAAO,MAAM,EAAI,WAAY,EAAI,WAAa,EAAI,UAAU,EAE3E",
|
|
10
|
+
"debugId": "BABB27F01082AFD864756E2164756E21",
|
|
11
11
|
"names": []
|
|
12
12
|
}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Server Client - Internal module for routing requests through treq server.
|
|
3
|
+
*
|
|
4
|
+
* This is the internal implementation used by createClient() when a server URL is configured.
|
|
5
|
+
* The public API remains createClient({ server: 'http://localhost:4096' }).
|
|
6
|
+
*
|
|
7
|
+
* @internal
|
|
8
|
+
*/
|
|
9
|
+
import type { Client, ClientConfig } from './types';
|
|
10
|
+
/**
|
|
11
|
+
* Internal configuration for server client, extending public ClientConfig.
|
|
12
|
+
*/
|
|
13
|
+
export interface ServerClientConfig extends ClientConfig {
|
|
14
|
+
/** Base URL of the treq server (required) */
|
|
15
|
+
serverUrl: string;
|
|
16
|
+
/** Bearer token for authentication */
|
|
17
|
+
token?: string | undefined;
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Metadata about the server connection.
|
|
21
|
+
* Accessible via getServerMetadata() utility.
|
|
22
|
+
*/
|
|
23
|
+
export interface ServerMetadata {
|
|
24
|
+
/** The server URL the client is connected to */
|
|
25
|
+
readonly serverUrl: string;
|
|
26
|
+
/** Session ID created on the server */
|
|
27
|
+
readonly sessionId: string | undefined;
|
|
28
|
+
/** Flow ID for observability (TUI) */
|
|
29
|
+
readonly flowId: string | undefined;
|
|
30
|
+
}
|
|
31
|
+
export declare const SERVER_METADATA: unique symbol;
|
|
32
|
+
/**
|
|
33
|
+
* Read environment variable safely (works in both Node and Bun).
|
|
34
|
+
*/
|
|
35
|
+
export declare function getEnvVar(name: string): string | undefined;
|
|
36
|
+
/**
|
|
37
|
+
* Create a server-backed client (internal use).
|
|
38
|
+
*
|
|
39
|
+
* Called by createClient() when server URL is provided.
|
|
40
|
+
* Automatically creates a session and flow on first use.
|
|
41
|
+
*
|
|
42
|
+
* When running as a server-spawned script:
|
|
43
|
+
* - TREQ_FLOW_ID: Attach to existing flow (don't create new one)
|
|
44
|
+
* - TREQ_SESSION_ID: Use pre-created session (skip session creation)
|
|
45
|
+
* - TREQ_TOKEN: Use scoped script token for authentication
|
|
46
|
+
*
|
|
47
|
+
* @internal
|
|
48
|
+
*/
|
|
49
|
+
export declare function createServerClient(config: ServerClientConfig): Client;
|
|
50
|
+
//# sourceMappingURL=server-client.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"server-client.d.ts","sourceRoot":"","sources":["../src/server-client.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAO,KAAK,EAAE,MAAM,EAAE,YAAY,EAAc,MAAM,SAAS,CAAC;AAEhE;;GAEG;AACH,MAAM,WAAW,kBAAmB,SAAQ,YAAY;IACtD,6CAA6C;IAC7C,SAAS,EAAE,MAAM,CAAC;IAClB,sCAAsC;IACtC,KAAK,CAAC,EAAE,MAAM,GAAG,SAAS,CAAC;CAC5B;AAED;;;GAGG;AACH,MAAM,WAAW,cAAc;IAC7B,gDAAgD;IAChD,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC;IAC3B,uCAAuC;IACvC,QAAQ,CAAC,SAAS,EAAE,MAAM,GAAG,SAAS,CAAC;IACvC,sCAAsC;IACtC,QAAQ,CAAC,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC;CACrC;AAGD,eAAO,MAAM,eAAe,eAA2B,CAAC;AA4CxD;;GAEG;AACH,wBAAgB,SAAS,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS,CAK1D;AAED;;;;;;;;;;;;GAYG;AACH,wBAAgB,kBAAkB,CAAC,MAAM,EAAE,kBAAkB,GAAG,MAAM,CA2PrE"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Server metadata utilities for TUI and tooling.
|
|
3
|
+
*
|
|
4
|
+
* When a client is created with a server URL, it maintains internal metadata
|
|
5
|
+
* about the server connection (session ID, flow ID). This utility allows
|
|
6
|
+
* external tools to access this metadata for debugging, observability, or
|
|
7
|
+
* orchestration purposes.
|
|
8
|
+
*/
|
|
9
|
+
import { type ServerMetadata } from './server-client';
|
|
10
|
+
import type { Client } from './types';
|
|
11
|
+
export type { ServerMetadata } from './server-client';
|
|
12
|
+
export declare function getServerMetadata(client: Client): ServerMetadata | undefined;
|
|
13
|
+
//# sourceMappingURL=server-metadata.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"server-metadata.d.ts","sourceRoot":"","sources":["../src/server-metadata.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAO,EAAmB,KAAK,cAAc,EAAE,MAAM,iBAAiB,CAAC;AACvE,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAEtC,YAAY,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AAyBtD,wBAAgB,iBAAiB,CAAC,MAAM,EAAE,MAAM,GAAG,cAAc,GAAG,SAAS,CAO5E"}
|
package/dist/types.d.ts
CHANGED
|
@@ -229,6 +229,26 @@ export interface ClientConfig {
|
|
|
229
229
|
/** Proxy URL */
|
|
230
230
|
proxy?: string;
|
|
231
231
|
};
|
|
232
|
+
/**
|
|
233
|
+
* Server URL to route requests through.
|
|
234
|
+
* When set, the client operates in "server mode" - requests are executed
|
|
235
|
+
* by the treq server instead of locally.
|
|
236
|
+
*
|
|
237
|
+
* If not provided, will try to read from TREQ_SERVER environment variable.
|
|
238
|
+
*
|
|
239
|
+
* @example 'http://localhost:4096'
|
|
240
|
+
*/
|
|
241
|
+
server?: string;
|
|
242
|
+
/**
|
|
243
|
+
* Bearer token for server authentication.
|
|
244
|
+
* If not provided, will try to read from TREQ_TOKEN environment variable.
|
|
245
|
+
*/
|
|
246
|
+
serverToken?: string;
|
|
247
|
+
/**
|
|
248
|
+
* Server-side config profile to use.
|
|
249
|
+
* Selects a named profile from the server's treq.jsonc configuration.
|
|
250
|
+
*/
|
|
251
|
+
profile?: string;
|
|
232
252
|
}
|
|
233
253
|
/**
|
|
234
254
|
* Options for running a request.
|
|
@@ -271,6 +291,16 @@ export interface RunOptions {
|
|
|
271
291
|
*
|
|
272
292
|
* // Run another request
|
|
273
293
|
* const profile = await client.run('./users/profile.http');
|
|
294
|
+
*
|
|
295
|
+
* // Clean up when done
|
|
296
|
+
* await client.close();
|
|
297
|
+
* ```
|
|
298
|
+
*
|
|
299
|
+
* @example Using explicit resource management (TypeScript 5.2+)
|
|
300
|
+
* ```typescript
|
|
301
|
+
* await using client = createClient({ server: 'http://localhost:4096' });
|
|
302
|
+
* const res = await client.run('./auth/login.http');
|
|
303
|
+
* // client.close() is called automatically when scope exits
|
|
274
304
|
* ```
|
|
275
305
|
*/
|
|
276
306
|
export interface Client {
|
|
@@ -293,5 +323,16 @@ export interface Client {
|
|
|
293
323
|
setVariable(key: string, value: unknown): void;
|
|
294
324
|
/** Get a copy of all current variables */
|
|
295
325
|
getVariables(): Record<string, unknown>;
|
|
326
|
+
/**
|
|
327
|
+
* Close the client and release any resources.
|
|
328
|
+
* - For local clients: no-op
|
|
329
|
+
* - For server clients: finishes the flow (best-effort)
|
|
330
|
+
*/
|
|
331
|
+
close(): Promise<void>;
|
|
332
|
+
/**
|
|
333
|
+
* Async dispose for explicit resource management (TypeScript 5.2+).
|
|
334
|
+
* Equivalent to calling `close()`.
|
|
335
|
+
*/
|
|
336
|
+
[Symbol.asyncDispose](): Promise<void>;
|
|
296
337
|
}
|
|
297
338
|
//# sourceMappingURL=types.d.ts.map
|