@canaryai/cli 0.1.1-alpha.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/runner/instrumentation.ts","../../src/runner/config.ts","../../src/runner/env.ts","../../src/runner/healer.ts","../../src/runner/ai-client.ts","../../src/runner/state.ts","../../src/runner/preload.ts"],"sourcesContent":["import Module from \"node:module\";\nimport { createRequire } from \"node:module\";\nimport { pathToFileURL } from \"node:url\";\nimport { loadCanaryConfig } from \"./config\";\nimport { classifyFailure, executeHealActions, type FailureContext } from \"./healer\";\nimport { alreadyPatched, markPatched, recordHealingEvent, setEventLogPath } from \"./state\";\n\nlet DEBUG = false;\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nconst requireFn: any = typeof require !== \"undefined\" ? require : createRequire(import.meta.url);\n\nconst LOCATOR_ACTIONS = new Set([\n \"click\",\n \"dblclick\",\n \"fill\",\n \"check\",\n \"uncheck\",\n \"hover\",\n \"press\",\n \"type\",\n \"selectOption\",\n \"tap\",\n]);\n\nconst PAGE_ACTIONS = new Set([\n \"goto\",\n \"click\",\n \"dblclick\",\n \"fill\",\n \"check\",\n \"uncheck\",\n \"hover\",\n \"press\",\n \"type\",\n \"selectOption\",\n \"tap\",\n \"waitForSelector\",\n]);\n\nconst LOCATOR_FACTORIES = new Set([\n \"locator\",\n \"getByRole\",\n \"getByText\",\n \"getByLabel\",\n \"getByPlaceholder\",\n \"getByAltText\",\n \"getByTitle\",\n \"getByTestId\",\n \"frameLocator\",\n]);\n\nconst LOCATOR_CHAIN_METHODS = new Set([\n \"locator\",\n \"first\",\n \"last\",\n \"nth\",\n \"filter\",\n \"getByRole\",\n \"getByText\",\n \"getByLabel\",\n \"getByPlaceholder\",\n \"getByAltText\",\n \"getByTitle\",\n \"getByTestId\",\n]);\n\nexport function installInstrumentation(): void {\n if (alreadyPatched()) return;\n\n const config = loadCanaryConfig();\n DEBUG = !!config.debug;\n if (!config.enabled) {\n markPatched();\n return;\n }\n\n setEventLogPath(config.eventLogPath);\n if (!ensurePlaywrightVersion(config.allowedPlaywrightVersion)) {\n recordHealingEvent({\n kind: \"unknown\",\n action: \"playwright_version_mismatch\",\n errorMessage: \"Playwright version does not match CANARY_PLAYWRIGHT_VERSION\",\n healed: false,\n });\n markPatched();\n return;\n }\n hookPlaywrightModuleLoad();\n\n if (DEBUG) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] instrumentation installed (eventLog=${config.eventLogPath})`);\n }\n\n recordHealingEvent({\n kind: \"unknown\",\n action: \"instrumentation_ready\",\n healed: false,\n });\n markPatched();\n}\n\nfunction hookPlaywrightModuleLoad(): void {\n // Prevent double-hooking.\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const mod = Module as any;\n const originalLoad: typeof mod._load = mod._canaryOriginalLoad ?? mod._load;\n if (!mod._canaryOriginalLoad) {\n mod._canaryOriginalLoad = originalLoad;\n }\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n mod._load = function patchedLoad(request: string, parent: any, isMain: boolean) {\n if (request === \"@playwright/test\" || request === \"playwright/test\") {\n if (DEBUG) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] patching playwright module (request=${request})`);\n }\n const real = originalLoad.call(this, request, parent, isMain);\n return createPatchedPlaywright(real);\n }\n return originalLoad.call(this, request, parent, isMain);\n };\n}\n\ntype PlaywrightExports = {\n test: { extend: typeof import(\"@playwright/test\").test.extend };\n expect: typeof import(\"@playwright/test\").expect;\n [key: string]: unknown;\n};\n\nfunction createPatchedPlaywright(real: PlaywrightExports): PlaywrightExports {\n if ((real as { __canaryPatched?: boolean }).__canaryPatched) return real;\n\n const extended = real.test.extend({\n page: async ({ page }, use) => {\n recordHealingEvent({\n kind: \"page\",\n action: \"fixture_initialized\",\n healed: false,\n });\n const wrappedPage = wrapPage(page);\n await use(wrappedPage);\n },\n });\n\n const patched: PlaywrightExports = {\n ...real,\n test: extended,\n expect: wrapExpect(real.expect),\n };\n\n (patched as { __canaryPatched?: boolean }).__canaryPatched = true;\n if (DEBUG) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] playwright patched`);\n }\n return patched;\n}\n\nfunction ensurePlaywrightVersion(expected?: string): boolean {\n if (!expected) return true;\n try {\n const pkg = requireFn(\"playwright/package.json\") as { version?: string };\n if (pkg.version && pkg.version.startsWith(expected)) {\n return true;\n }\n // eslint-disable-next-line no-console\n console.warn(`[canary] Playwright version mismatch. Expected ${expected}, found ${pkg.version ?? \"unknown\"}`);\n return false;\n } catch {\n return true;\n }\n}\n\nfunction wrapExpect(expectImpl: PlaywrightExports[\"expect\"]): PlaywrightExports[\"expect\"] {\n const wrapMatchers = (expectation: unknown, mode: \"hard\" | \"soft\") => {\n return new Proxy(expectation ?? {}, {\n get(target, prop, receiver) {\n const value = Reflect.get(target, prop, receiver);\n if (typeof prop === \"string\" && typeof value === \"function\") {\n return (...args: unknown[]) => runMatcherWithHealing({ matcher: value as (...args: unknown[]) => unknown, matcherName: prop, args, expectTarget: target, mode });\n }\n return value;\n },\n });\n };\n\n const proxy = new Proxy(expectImpl, {\n apply(target, thisArg, argArray) {\n const expectation = (target as (...args: unknown[]) => unknown).apply(thisArg, argArray as unknown[]);\n return wrapMatchers(expectation, \"hard\");\n },\n get(target, prop, receiver) {\n const value = Reflect.get(target, prop, receiver);\n if (prop === \"soft\" && typeof value === \"function\") {\n return (...args: unknown[]) => wrapMatchers((value as (...args: unknown[]) => unknown).apply(target, args), \"soft\");\n }\n return value;\n },\n });\n\n return proxy as PlaywrightExports[\"expect\"];\n}\n\nfunction wrapPage<PageType extends object>(page: PageType): PageType {\n return new Proxy(page, {\n get(target, prop, receiver) {\n const value = Reflect.get(target, prop, receiver);\n\n if (typeof prop === \"string\") {\n if (LOCATOR_FACTORIES.has(prop) && typeof value === \"function\") {\n return (...args: unknown[]) => {\n const locator = (value as (...args: unknown[]) => unknown).apply(target, args);\n return wrapLocator(locator);\n };\n }\n\n if (PAGE_ACTIONS.has(prop) && typeof value === \"function\") {\n return async (...args: unknown[]) => {\n return attemptWithHealing({\n kind: \"page\",\n action: prop,\n target: safeTargetString(target),\n locator: undefined,\n page: target as unknown,\n invoke: () => (value as (...args: unknown[]) => unknown).apply(target, args),\n });\n };\n }\n }\n\n return value;\n },\n });\n}\n\nfunction wrapLocator(locator: unknown): unknown {\n if (!locator || typeof locator !== \"object\") return locator;\n\n return new Proxy(locator, {\n get(target, prop, receiver) {\n const value = Reflect.get(target, prop, receiver);\n\n if (typeof prop === \"string\" && LOCATOR_ACTIONS.has(prop) && typeof value === \"function\") {\n return async (...args: unknown[]) => {\n return attemptWithHealing({\n kind: \"locator\",\n action: prop,\n target: safeTargetString(target),\n locator: target as unknown,\n page: undefined,\n invoke: () => (value as (...args: unknown[]) => unknown).apply(target, args),\n });\n };\n }\n\n if (typeof prop === \"string\" && LOCATOR_CHAIN_METHODS.has(prop) && typeof value === \"function\") {\n return (...args: unknown[]) => wrapLocator((value as (...args: unknown[]) => unknown).apply(target, args));\n }\n\n return value;\n },\n });\n}\n\ntype AttemptContext = {\n kind: FailureContext[\"kind\"];\n action: string;\n target?: string;\n locator?: unknown;\n page?: unknown;\n invoke: () => unknown;\n};\n\nasync function attemptWithHealing<T>(ctx: AttemptContext): Promise<T> {\n try {\n return await Promise.resolve(ctx.invoke()) as T;\n } catch (error) {\n const failure = buildFailureContext(ctx.kind, ctx.action, ctx.target, error);\n recordHealingEvent({ ...failure, healed: false });\n const decision = classifyFailure(failure);\n if (DEBUG) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] failure intercepted: kind=${ctx.kind} action=${ctx.action} reason=${decision.healable ? decision.reason ?? \"healable\" : decision.reason}`);\n }\n if (!decision.healable) {\n throw error;\n }\n\n const outcome = await executeHealActions(decision, failure, {\n kind: ctx.kind,\n action: ctx.action,\n target: ctx.target,\n locator: isLocatorLike(ctx.locator) ? ctx.locator : undefined,\n page: isPageLike(ctx.page) ? ctx.page : undefined,\n testContext: {\n testFile: testContext?.testFile,\n testTitle: testContext?.testTitle,\n testSource: loadTestSource(testContext?.testFile),\n },\n });\n\n // Try to get test context for correlation\n const testContext = getTestContext();\n\n const summaryBase = {\n ...failure,\n healed: outcome.healed,\n strategy: 'agentic',\n reason: outcome.reason ?? decision.reason,\n actions: actionsToEventItems(outcome.actionsRun),\n durationMs: outcome.durationMs,\n mode: outcome.mode,\n decision: decision.reason,\n modelId: outcome.modelId,\n summary: outcome.summary,\n testFile: testContext?.testFile,\n testTitle: testContext?.testTitle,\n } as const;\n\n if (outcome.healed && !outcome.shouldRetryOriginal) {\n recordHealingEvent({ ...summaryBase, healed: true });\n return undefined as T;\n }\n\n if (outcome.shouldRetryOriginal) {\n try {\n const retried = await Promise.resolve(ctx.invoke()) as T;\n if (DEBUG) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] retry_original succeeded for action=${ctx.action}`);\n }\n recordHealingEvent({ ...summaryBase, healed: true });\n return retried;\n } catch (retryError) {\n const retryInfo = errorInfo(retryError);\n recordHealingEvent({ ...summaryBase, healed: false, errorMessage: retryInfo.message });\n if (DEBUG) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] retry_original failed for action=${ctx.action}: ${retryInfo.message ?? retryError}`);\n }\n throw retryError;\n }\n }\n\n recordHealingEvent(summaryBase);\n throw error;\n }\n}\n\nasync function runMatcherWithHealing(params: {\n matcher: (...args: unknown[]) => unknown;\n matcherName: string;\n args: unknown[];\n expectTarget: unknown;\n mode: \"hard\" | \"soft\";\n}): Promise<unknown> {\n const { matcher, matcherName, args, expectTarget, mode } = params;\n\n const invoke = () => matcher.apply(expectTarget, args);\n\n // Skip healing for soft assertions to avoid altering semantics.\n if (mode === \"soft\") {\n return Promise.resolve(invoke());\n }\n\n try {\n return await Promise.resolve(invoke());\n } catch (error) {\n const target = stringifyTarget(args?.[0]);\n const failure = buildFailureContext(\"expect\", matcherName, target, error);\n recordHealingEvent({ ...failure, healed: false });\n const decision = classifyFailure(failure);\n if (!decision.healable) {\n throw error;\n }\n\n const outcome = await executeHealActions(decision, failure, {\n kind: \"expect\",\n action: matcherName,\n target,\n locator: isLocatorLike(args?.[0]) ? args[0] : undefined,\n });\n\n // Try to get test context for correlation\n const testContext = getTestContext();\n\n const summaryBase = {\n ...failure,\n healed: false,\n strategy: 'agentic',\n reason: outcome.reason ?? decision.reason,\n actions: actionsToEventItems(outcome.actionsRun),\n durationMs: outcome.durationMs,\n mode: outcome.mode,\n decision: decision.reason,\n modelId: outcome.modelId,\n summary: outcome.summary,\n testFile: testContext?.testFile,\n testTitle: testContext?.testTitle,\n } as const;\n\n if (outcome.shouldRetryOriginal) {\n try {\n const retried = await Promise.resolve(invoke());\n recordHealingEvent({ ...summaryBase, healed: true });\n return retried;\n } catch (retryError) {\n const retryInfo = errorInfo(retryError);\n recordHealingEvent({ ...summaryBase, healed: false, errorMessage: retryInfo.message });\n throw retryError;\n }\n }\n\n recordHealingEvent(summaryBase);\n throw error;\n }\n}\n\nfunction buildFailureContext(\n kind: FailureContext[\"kind\"],\n action: string,\n target: string | undefined,\n error: unknown\n): FailureContext {\n const info = errorInfo(error);\n return {\n kind,\n action,\n target,\n errorMessage: info.message,\n errorName: info.name,\n stack: info.stack,\n };\n}\n\nfunction actionsToEventItems(actions: string[]): Array<{ type: string; detail?: string }> {\n return actions.map((action) => ({ type: action }));\n}\n\nfunction errorInfo(error: unknown): { message?: string; name?: string; stack?: string } {\n if (error instanceof Error) {\n return { message: error.message, name: error.name, stack: error.stack };\n }\n return { message: typeof error === \"string\" ? error : JSON.stringify(error) };\n}\n\nfunction isLocatorLike(candidate: unknown): candidate is { scrollIntoViewIfNeeded: () => Promise<unknown> } {\n return Boolean(candidate && typeof candidate === \"object\" && \"scrollIntoViewIfNeeded\" in (candidate as Record<string, unknown>));\n}\n\nfunction isPageLike(candidate: unknown): candidate is { waitForTimeout: (ms: number) => Promise<unknown> } {\n return Boolean(candidate && typeof candidate === \"object\" && \"waitForTimeout\" in (candidate as Record<string, unknown>));\n}\n\nfunction safeTargetString(target: unknown): string | undefined {\n try {\n if (typeof target === \"object\" && target !== null && \"toString\" in target) {\n const s = String((target as { toString: () => string }).toString());\n if (s && s !== \"[object Object]\") return s;\n }\n } catch {\n // ignore\n }\n return undefined;\n}\n\nfunction loadTestSource(filePath?: string): string | undefined {\n if (!filePath) return undefined;\n try {\n return requireFn(\"fs\").readFileSync(filePath, \"utf-8\") as string;\n } catch {\n return undefined;\n }\n}\n\nfunction stringifyTarget(candidate: unknown): string | undefined {\n if (!candidate) return undefined;\n if (typeof candidate === \"string\") return candidate;\n if (typeof candidate === \"object\") {\n if (\"selector\" in (candidate as Record<string, unknown>) && typeof (candidate as Record<string, unknown>).selector === \"string\") {\n return String((candidate as Record<string, unknown>).selector);\n }\n }\n return safeTargetString(candidate);\n}\n\n/**\n * Try to get test context from Playwright's test.info().\n * This is used to correlate healing events with tests in the report.\n */\nfunction getTestContext(): { testFile?: string; testTitle?: string } | undefined {\n try {\n // Try to dynamically require @playwright/test and call test.info()\n const playwright = requireFn(\"@playwright/test\") as { test?: { info?: () => { file?: string; title?: string } } };\n if (playwright?.test?.info) {\n const info = playwright.test.info();\n if (info) {\n return {\n testFile: info.file,\n testTitle: info.title,\n };\n }\n }\n } catch {\n // Not in a test context or @playwright/test not available\n }\n return undefined;\n}\n","import path from \"node:path\";\nimport { loadCanaryEnv } from \"./env\";\n\nexport type CanaryConfig = {\n enabled: boolean;\n allowedPlaywrightVersion?: string;\n aiProvider?: string;\n aiModel?: string;\n apiKey?: string;\n healTimeoutMs: number;\n maxActions: number;\n dryRun: boolean;\n warnOnly: boolean;\n visionEnabled: boolean;\n debug: boolean;\n eventLogPath: string;\n eventLoggingEnabled: boolean;\n readOnly: boolean;\n allowRunCode: boolean;\n allowEvaluate: boolean;\n maxPayloadBytes: number;\n};\n\nexport function loadCanaryConfig(): CanaryConfig {\n loadCanaryEnv();\n\n const eventLogPath = (() => {\n const userPath = process.env.CANARY_EVENT_LOG;\n if (userPath) return userPath;\n const worker = process.env.TEST_WORKER_INDEX ?? \"runner\";\n return path.join(process.cwd(), \"test-results\", \"ai-healer\", `events-worker-${worker}.jsonl`);\n })();\n\n const baseConfig: CanaryConfig = {\n enabled: true,\n allowedPlaywrightVersion: process.env.CANARY_PLAYWRIGHT_VERSION,\n aiProvider: process.env.AI_PROVIDER,\n aiModel: process.env.AI_MODEL,\n apiKey: process.env.AI_API_KEY,\n healTimeoutMs: Number(process.env.AI_TIMEOUT_MS ?? 120000), // 2 minutes for agentic healing\n maxActions: Number(process.env.CANARY_MAX_ACTIONS ?? 50), // Generous step limit for agentic healing\n dryRun: process.env.CANARY_DRY_RUN === \"1\",\n warnOnly: process.env.CANARY_WARN_ONLY === \"1\",\n visionEnabled: process.env.CANARY_VISION === \"1\" || process.env.AI_VISION === \"1\",\n debug: process.env.CANARY_DEBUG === \"1\",\n readOnly: process.env.CANARY_READ_ONLY === \"1\",\n allowRunCode: process.env.CANARY_ALLOW_RUN_CODE === \"1\",\n allowEvaluate: process.env.CANARY_ALLOW_EVALUATE !== \"0\",\n maxPayloadBytes: Number(process.env.CANARY_MAX_PAYLOAD_BYTES ?? 60000), // cap snapshots/screenshots/text\n eventLogPath,\n eventLoggingEnabled: process.env.CANARY_EVENT_LOG !== \"0\",\n };\n\n const disabled =\n process.env.CANARY_ENABLED === \"0\" ||\n process.env.CANARY_DISABLED === \"1\" ||\n process.env.AI_HEALING === \"0\";\n\n return {\n ...baseConfig,\n enabled: !disabled && baseConfig.enabled,\n };\n}\n","import fs from \"node:fs\";\nimport path from \"node:path\";\n\nlet loaded = false;\n\nexport function loadCanaryEnv(): void {\n if (loaded) return;\n loaded = true;\n\n const explicitPath = process.env.CANARY_ENV_FILE;\n const defaultPath = path.join(process.cwd(), \".env\");\n const envPath = explicitPath || defaultPath;\n\n if (fs.existsSync(envPath)) {\n loadFile(envPath);\n }\n}\n\nfunction loadFile(filePath: string): void {\n try {\n const raw = fs.readFileSync(filePath, \"utf-8\");\n for (const line of raw.split(\"\\n\")) {\n const trimmed = line.trim();\n if (!trimmed || trimmed.startsWith(\"#\")) continue;\n const idx = trimmed.indexOf(\"=\");\n if (idx === -1) continue;\n const key = trimmed.slice(0, idx).trim();\n const value = trimmed.slice(idx + 1).trim();\n if (!key) continue;\n if (process.env[key] === undefined) {\n process.env[key] = stripQuotes(value);\n }\n }\n } catch {\n // ignore\n }\n}\n\nfunction stripQuotes(value: string): string {\n if ((value.startsWith(\"\\\"\") && value.endsWith(\"\\\"\")) || (value.startsWith(\"'\") && value.endsWith(\"'\"))) {\n return value.slice(1, -1);\n }\n return value;\n}\n","import { stepCountIs, streamText, tool, type Tool } from 'ai';\nimport { z } from 'zod';\nimport { loadCanaryConfig } from './config';\nimport { resolveHealerModel } from './ai-client';\n\nconst REDACTION_PATTERNS: Array<{ regex: RegExp; replacement: string }> = [\n { regex: /bearer\\s+[a-z0-9._-]+/gi, replacement: '[REDACTED_BEARER]' },\n { regex: /api[_-]?key[:\\s\"']+[a-z0-9._-]+/gi, replacement: '[REDACTED_API_KEY]' },\n { regex: /secret[:\\s\"']+[a-z0-9._-]+/gi, replacement: '[REDACTED_SECRET]' },\n { regex: /[A-Za-z0-9._%+-]+@[A-Za-z0-9.-]+\\.[A-Za-z]{2,}/g, replacement: '[REDACTED_EMAIL]' },\n { regex: /[0-9]{12,}/g, replacement: '[REDACTED_NUMBER]' },\n];\nconst CODE_DENY_PATTERNS = [/process\\.env/i, /child_process/i, /\\brequire\\s*\\(/i, /\\bimport\\s*\\(/i];\n\n// === Types ===\n\nexport type FailureContext = {\n kind: 'locator' | 'page' | 'expect' | 'unknown';\n action?: string;\n target?: string;\n errorMessage?: string;\n errorName?: string;\n stack?: string;\n};\n\nexport type TestContext = {\n testFile?: string;\n testTitle?: string;\n testSource?: string;\n currentStep: string;\n expectedAfter?: string;\n action: string;\n target?: string;\n errorMessage?: string;\n};\n\nexport type HealDecision =\n | { healable: false; reason: string }\n | { healable: true; reason?: string };\n\nexport type HealExecutionContext = {\n kind: FailureContext['kind'];\n target?: string;\n action?: string;\n locator?: LocatorLike;\n page?: PageLike;\n testContext?: Partial<TestContext>;\n};\n\nexport type HealOutcome = {\n healed: boolean;\n shouldRetryOriginal: boolean;\n actionsRun: string[];\n timedOut: boolean;\n durationMs: number;\n errors: string[];\n mode: 'dry-run' | 'warn' | 'full';\n reason?: string;\n modelId?: string;\n summary?: string; // Human-readable one-line summary for report\n};\n\ntype LocatorLike = {\n scrollIntoViewIfNeeded?: (options?: { timeout?: number }) => Promise<unknown>;\n};\n\ntype PageLike = {\n waitForTimeout?: (ms: number) => Promise<unknown>;\n accessibility?: { snapshot: () => Promise<unknown> };\n screenshot?: (options?: { fullPage?: boolean }) => Promise<Buffer | Uint8Array | string>;\n locator?: (selector: string) => LocatorLike & {\n click?: (opts?: unknown) => Promise<unknown>;\n fill?: (value: string, opts?: unknown) => Promise<unknown>;\n pressSequentially?: (text: string, opts?: unknown) => Promise<unknown>;\n hover?: () => Promise<unknown>;\n selectOption?: (value: string) => Promise<unknown>;\n textContent?: () => Promise<string | null>;\n isVisible?: () => Promise<boolean>;\n waitFor?: (opts?: { state?: string }) => Promise<unknown>;\n };\n getByText?: (text: string) => LocatorLike & {\n click?: (opts?: unknown) => Promise<unknown>;\n };\n keyboard?: { press?: (key: string) => Promise<unknown> };\n goto?: (url: string) => Promise<unknown>;\n goBack?: () => Promise<unknown>;\n reload?: () => Promise<unknown>;\n evaluate?: (script: string) => Promise<unknown>;\n};\n\n// === Failure Classification ===\n\nexport function classifyFailure(context: FailureContext): HealDecision {\n const message = (context.errorMessage ?? '').toLowerCase();\n\n // Non-healable: context/browser closed\n if (\n message.includes('closed') ||\n message.includes('target page, context or browser has been closed')\n ) {\n return { healable: false, reason: 'context_closed' };\n }\n\n // Non-healable: navigation failed completely\n if (message.includes('navigation failed') && message.includes('net::')) {\n return { healable: false, reason: 'navigation_failed' };\n }\n\n // Everything else is potentially healable - let the agent decide\n return { healable: true, reason: 'agent_healing' };\n}\n\n// === Main Entry Point ===\n\nexport async function executeHealActions(\n decision: HealDecision,\n failure: FailureContext,\n execCtx: HealExecutionContext\n): Promise<HealOutcome> {\n const config = loadCanaryConfig();\n const started = Date.now();\n const mode = resolveMode(config);\n const actionsRun: string[] = [];\n const errors: string[] = [];\n const logTools = config.debug || process.env.CANARY_TOOL_LOG === '1';\n const initialContext = await buildInitialPageContext(execCtx.page, config);\n\n if (!decision.healable) {\n return baseOutcome({\n mode,\n started,\n healed: false,\n shouldRetryOriginal: false,\n reason: decision.reason,\n actionsRun,\n errors,\n });\n }\n\n const { model, modelId, reason: modelReason } = resolveHealerModel(config);\n if (!model) {\n return baseOutcome({\n mode,\n started,\n healed: false,\n shouldRetryOriginal: false,\n reason: modelReason ?? 'no_model',\n actionsRun,\n errors,\n });\n }\n\n // Build test context for the agent\n const testContext: TestContext = {\n testFile: execCtx.testContext?.testFile,\n testTitle: execCtx.testContext?.testTitle,\n testSource: execCtx.testContext?.testSource ? sanitizeString(execCtx.testContext.testSource) : undefined,\n currentStep: `${failure.action ?? 'unknown'}(${failure.target ?? 'unknown'})`,\n expectedAfter: execCtx.testContext?.expectedAfter,\n action: failure.action ?? 'unknown',\n target: failure.target,\n errorMessage: failure.errorMessage,\n };\n\n // Create the agentic toolset\n const toolset = createAgenticTools({\n page: execCtx.page,\n config,\n actionsRun,\n debug: config.debug,\n });\n\n let completionReason: string | undefined;\n\n try {\n const abort = AbortSignal.timeout(config.healTimeoutMs);\n const result = await streamText({\n model,\n system: buildSystemPrompt(testContext, config),\n messages: buildInitialMessages(testContext, failure, initialContext),\n tools: toolset.tools,\n stopWhen: stepCountIs(Math.max(1, config.maxActions)),\n abortSignal: abort,\n maxRetries: 0,\n onStepFinish: (step) => {\n if (config.debug) {\n // eslint-disable-next-line no-console\n console.log(\n `[canary][debug] step finish: finishReason=${step.finishReason}; toolCalls=${step.toolCalls?.length ?? 0}; toolResults=${step.toolResults?.length ?? 0}`\n );\n }\n if (logTools) {\n logToolStep(step);\n }\n },\n });\n\n // Consume stream until agent calls mark_complete\n for await (const _ of result.fullStream) {\n if (toolset.isComplete()) {\n completionReason = toolset.getCompletionReason();\n if (config.debug) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] agent marked complete: ${completionReason}`);\n }\n break;\n }\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n errors.push(message);\n if (config.debug) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] agent error: ${message}`);\n }\n }\n\n const healed = toolset.isComplete();\n\n return {\n healed,\n shouldRetryOriginal: false, // Agent-controlled completion, no retry needed\n actionsRun,\n timedOut: Date.now() - started > config.healTimeoutMs,\n durationMs: Date.now() - started,\n errors,\n mode,\n reason: completionReason ?? (healed ? 'agent_healed' : 'agent_incomplete'),\n modelId,\n summary: toolset.getSummary(),\n };\n}\n\n// === Agentic Toolset ===\n\nexport function createAgenticTools(params: {\n page?: PageLike;\n config: ReturnType<typeof loadCanaryConfig>;\n actionsRun: string[];\n debug?: boolean;\n}): {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n tools: Record<string, Tool<any, any>>;\n isComplete: () => boolean;\n getCompletionReason: () => string | undefined;\n getSummary: () => string | undefined;\n} {\n const { page, config, actionsRun, debug } = params;\n let complete = false;\n let completionReason: string | undefined;\n let completionSummary: string | undefined;\n const maxPayloadBytes = Math.max(1000, config.maxPayloadBytes || 60000);\n\n const log = (msg: string) => {\n if (debug) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] ${msg}`);\n }\n };\n\n const track = (action: string) => {\n actionsRun.push(action);\n };\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const tools: Record<string, Tool<any, any>> = {\n // === Core Action Tools ===\n click: tool<{ text: string }, { clicked: string; mode?: string; error?: string }>({\n description: 'Click an element by its visible text',\n inputSchema: z.object({ text: z.string().describe('The visible text to click') }),\n execute: async ({ text }) => {\n track(`click:${text}`);\n log(`click called with text=\"${text}\"`);\n if (config.readOnly) return { clicked: '', error: 'read_only_mode' };\n if (config.dryRun) return { clicked: text, mode: 'dry-run' };\n if (!page?.getByText) return { clicked: '', error: 'Page unavailable' };\n try {\n const el = page.getByText(text);\n if (el?.scrollIntoViewIfNeeded) {\n await el.scrollIntoViewIfNeeded({ timeout: 2000 });\n }\n const clickable = el as { click?: (opts?: unknown) => Promise<unknown> };\n if (clickable?.click) {\n await clickable.click({ timeout: 5000 });\n }\n log(`click succeeded for \"${text}\"`);\n return { clicked: text };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`click error: ${msg}`);\n return { clicked: '', error: msg };\n }\n },\n }),\n\n click_selector: tool<{ selector: string }, { clicked: string; mode?: string; error?: string }>({\n description: 'Click an element by CSS or Playwright selector',\n inputSchema: z.object({ selector: z.string().describe('CSS or Playwright selector') }),\n execute: async ({ selector }) => {\n track(`click_selector:${selector}`);\n log(`click_selector called with selector=\"${selector}\"`);\n if (config.readOnly) return { clicked: '', error: 'read_only_mode' };\n if (config.dryRun) return { clicked: selector, mode: 'dry-run' };\n if (!page?.locator) return { clicked: '', error: 'Page unavailable' };\n try {\n const el = page.locator(selector);\n if (el?.scrollIntoViewIfNeeded) {\n await el.scrollIntoViewIfNeeded({ timeout: 2000 });\n }\n if (el?.click) {\n await el.click({ timeout: 5000 });\n }\n log(`click_selector succeeded for \"${selector}\"`);\n return { clicked: selector };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`click_selector error: ${msg}`);\n return { clicked: '', error: msg };\n }\n },\n }),\n\n fill: tool<{ selector: string; value: string }, { filled: string; value?: string; mode?: string; error?: string }>({\n description: 'Fill an input field with a value using CSS selector (clears existing content)',\n inputSchema: z.object({\n selector: z.string().describe('CSS selector for the input (e.g. input[placeholder=\"...\"])'),\n value: z.string().describe('Value to fill'),\n }),\n execute: async ({ selector, value }) => {\n track(`fill:${selector}`);\n log(`fill called with selector=\"${selector}\" value=\"${value}\"`);\n if (config.readOnly) return { filled: '', error: 'read_only_mode' };\n if (config.dryRun) return { filled: selector, mode: 'dry-run' };\n if (!page?.locator) return { filled: '', error: 'Page unavailable' };\n try {\n const el = page.locator(selector);\n if (el?.fill) {\n await el.fill(value, { timeout: 5000 });\n }\n log(`fill succeeded`);\n return { filled: selector, value };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`fill error: ${msg}`);\n return { filled: '', error: msg };\n }\n },\n }),\n\n fill_by_name: tool<{ name: string; value: string }, { filled: string; value?: string; mode?: string; error?: string }>({\n description: 'Fill a textbox by its accessible name (shown in snapshot as textbox \"NAME\"). This is the PREFERRED way to fill inputs - use the name from the snapshot directly.',\n inputSchema: z.object({\n name: z.string().describe('The accessible name of the textbox (from snapshot, e.g. \"Enter 6-digit code\")'),\n value: z.string().describe('Value to fill'),\n }),\n execute: async ({ name, value }) => {\n track(`fill_by_name:${name}`);\n log(`fill_by_name called with name=\"${name}\" value=\"${value}\"`);\n if (config.readOnly) return { filled: '', error: 'read_only_mode' };\n if (config.dryRun) return { filled: name, mode: 'dry-run' };\n if (!page?.locator) return { filled: '', error: 'Page unavailable' };\n try {\n // Use role selector which matches the accessible name from the snapshot\n const el = page.locator(`role=textbox[name=\"${name}\"]`);\n if (el?.fill) {\n await el.fill(value, { timeout: 5000 });\n }\n log(`fill_by_name succeeded`);\n return { filled: name, value };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`fill_by_name error: ${msg}`);\n return { filled: '', error: msg };\n }\n },\n }),\n\n type: tool<{ selector: string; text: string }, { typed: string; into?: string; mode?: string; error?: string }>({\n description: 'Type text character by character (triggers key events)',\n inputSchema: z.object({\n selector: z.string().describe('CSS or Playwright selector'),\n text: z.string().describe('Text to type'),\n }),\n execute: async ({ selector, text }) => {\n track(`type:${selector}`);\n log(`type called with selector=\"${selector}\" text=\"${text}\"`);\n if (config.readOnly) return { typed: '', error: 'read_only_mode' };\n if (config.dryRun) return { typed: text, mode: 'dry-run' };\n if (!page?.locator) return { typed: '', error: 'Page unavailable' };\n try {\n const el = page.locator(selector);\n if (el?.pressSequentially) {\n await el.pressSequentially(text);\n }\n log(`type succeeded`);\n return { typed: text, into: selector };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`type error: ${msg}`);\n return { typed: '', error: msg };\n }\n },\n }),\n\n press_key: tool<{ key: string }, { pressed: string; mode?: string; error?: string }>({\n description: 'Press a keyboard key (Enter, Tab, Escape, ArrowDown, etc.)',\n inputSchema: z.object({ key: z.string().describe('Key to press') }),\n execute: async ({ key }) => {\n track(`press_key:${key}`);\n log(`press_key called with key=\"${key}\"`);\n if (config.readOnly) return { pressed: '', error: 'read_only_mode' };\n if (config.dryRun) return { pressed: key, mode: 'dry-run' };\n if (!page?.keyboard?.press) return { pressed: '', error: 'Page unavailable' };\n try {\n await page.keyboard.press(key);\n log(`press_key succeeded`);\n return { pressed: key };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`press_key error: ${msg}`);\n return { pressed: '', error: msg };\n }\n },\n }),\n\n hover: tool<{ selector: string }, { hovered: string; mode?: string; error?: string }>({\n description: 'Hover over an element',\n inputSchema: z.object({ selector: z.string().describe('CSS or Playwright selector') }),\n execute: async ({ selector }) => {\n track(`hover:${selector}`);\n log(`hover called with selector=\"${selector}\"`);\n if (config.readOnly) return { hovered: '', error: 'read_only_mode' };\n if (config.dryRun) return { hovered: selector, mode: 'dry-run' };\n if (!page?.locator) return { hovered: '', error: 'Page unavailable' };\n try {\n const el = page.locator(selector);\n if (el?.hover) {\n await el.hover();\n }\n log(`hover succeeded`);\n return { hovered: selector };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`hover error: ${msg}`);\n return { hovered: '', error: msg };\n }\n },\n }),\n\n select_option: tool<{ selector: string; value: string }, { selected: string; from?: string; mode?: string; error?: string }>({\n description: 'Select an option from a dropdown',\n inputSchema: z.object({\n selector: z.string().describe('CSS or Playwright selector for the select element'),\n value: z.string().describe('Value or label to select'),\n }),\n execute: async ({ selector, value }) => {\n track(`select_option:${selector}`);\n log(`select_option called with selector=\"${selector}\" value=\"${value}\"`);\n if (config.readOnly) return { selected: '', error: 'read_only_mode' };\n if (config.dryRun) return { selected: value, mode: 'dry-run' };\n if (!page?.locator) return { selected: '', error: 'Page unavailable' };\n try {\n const el = page.locator(selector);\n if (el?.selectOption) {\n await el.selectOption(value);\n }\n log(`select_option succeeded`);\n return { selected: value, from: selector };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`select_option error: ${msg}`);\n return { selected: '', error: msg };\n }\n },\n }),\n\n // === Navigation & Waiting ===\n wait: tool<{ ms: number }, { waited: number; mode?: string }>({\n description: 'Wait for a duration in milliseconds',\n inputSchema: z.object({ ms: z.number().min(100).max(30000).describe('Milliseconds to wait') }),\n execute: async ({ ms }) => {\n track(`wait:${ms}`);\n log(`wait called with ms=${ms}`);\n if (config.dryRun) return { waited: ms, mode: 'dry-run' };\n if (page?.waitForTimeout) {\n await page.waitForTimeout(ms);\n } else {\n await new Promise((resolve) => setTimeout(resolve, ms));\n }\n log(`wait completed`);\n return { waited: ms };\n },\n }),\n\n wait_for_selector: tool<{ selector: string; state?: 'visible' | 'hidden' | 'attached' | 'detached' }, { found: string; state?: string; mode?: string; error?: string }>({\n description: 'Wait for an element to reach a state (visible, hidden, attached, detached)',\n inputSchema: z.object({\n selector: z.string().describe('CSS or Playwright selector'),\n state: z.enum(['visible', 'hidden', 'attached', 'detached']).optional().describe('State to wait for'),\n }),\n execute: async ({ selector, state }) => {\n track(`wait_for_selector:${selector}`);\n log(`wait_for_selector called with selector=\"${selector}\" state=\"${state ?? 'visible'}\"`);\n if (config.dryRun) return { found: selector, mode: 'dry-run' };\n if (!page?.locator) return { found: '', error: 'Page unavailable' };\n try {\n const el = page.locator(selector);\n if (el?.waitFor) {\n await el.waitFor({ state: state ?? 'visible' });\n }\n log(`wait_for_selector succeeded`);\n return { found: selector, state: state ?? 'visible' };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`wait_for_selector error: ${msg}`);\n return { found: '', error: msg };\n }\n },\n }),\n\n navigate: tool<{ url: string }, { navigated: string; mode?: string; error?: string }>({\n description: 'Navigate to a URL',\n inputSchema: z.object({ url: z.string().describe('URL to navigate to') }),\n execute: async ({ url }) => {\n track(`navigate:${url}`);\n log(`navigate called with url=\"${url}\"`);\n if (config.readOnly) return { navigated: '', error: 'read_only_mode' };\n if (config.dryRun) return { navigated: url, mode: 'dry-run' };\n if (!page?.goto) return { navigated: '', error: 'Page unavailable' };\n try {\n await page.goto(url);\n log(`navigate succeeded`);\n return { navigated: url };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`navigate error: ${msg}`);\n return { navigated: '', error: msg };\n }\n },\n }),\n\n go_back: tool<Record<string, never>, { action: string; mode?: string; error?: string }>({\n description: 'Go back in browser history',\n inputSchema: z.object({}),\n execute: async () => {\n track('go_back');\n log(`go_back called`);\n if (config.readOnly) return { action: '', error: 'read_only_mode' };\n if (config.dryRun) return { action: 'back', mode: 'dry-run' };\n if (!page?.goBack) return { action: '', error: 'Page unavailable' };\n try {\n await page.goBack();\n log(`go_back succeeded`);\n return { action: 'back' };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`go_back error: ${msg}`);\n return { action: '', error: msg };\n }\n },\n }),\n\n reload: tool<Record<string, never>, { action: string; mode?: string; error?: string }>({\n description: 'Reload the current page',\n inputSchema: z.object({}),\n execute: async () => {\n track('reload');\n log(`reload called`);\n if (config.readOnly) return { action: '', error: 'read_only_mode' };\n if (config.dryRun) return { action: 'reload', mode: 'dry-run' };\n if (!page?.reload) return { action: '', error: 'Page unavailable' };\n try {\n await page.reload();\n log(`reload succeeded`);\n return { action: 'reload' };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`reload error: ${msg}`);\n return { action: '', error: msg };\n }\n },\n }),\n\n // === Inspection Tools ===\n snapshot: tool<Record<string, never>, { tree?: unknown; html?: string; mode?: string; error?: string }>({\n description: 'Get the accessibility tree of the current page to understand its structure. ALWAYS call this first!',\n inputSchema: z.object({}),\n execute: async () => {\n track('snapshot');\n log(`snapshot called`);\n if (config.dryRun) return { tree: null, mode: 'dry-run' };\n if (!page) return { error: 'Page unavailable' };\n\n let tree: unknown;\n let html: string | undefined;\n\n // Try accessibility tree first\n try {\n if (page.accessibility && typeof page.accessibility.snapshot === 'function') {\n tree = await page.accessibility.snapshot();\n }\n } catch (err) {\n log(`accessibility snapshot error: ${err instanceof Error ? err.message : String(err)}`);\n }\n\n // If accessibility tree is empty, fall back to HTML content\n if (!tree && page.evaluate) {\n try {\n html = await page.evaluate('document.body.innerHTML') as string;\n if (html) {\n log(`snapshot using HTML fallback`);\n }\n } catch (err) {\n log(`HTML fallback error: ${err instanceof Error ? err.message : String(err)}`);\n }\n }\n\n if (!tree && !html) {\n return { error: 'Could not get page content' };\n }\n\n log(`snapshot succeeded`);\n const sanitizedTree = sanitizeUnknown(tree, maxPayloadBytes);\n const sanitizedHtml = html ? truncate(sanitizeString(html), maxPayloadBytes) : undefined;\n return { tree: sanitizedTree, html: sanitizedHtml };\n },\n }),\n\n screenshot: tool<{ fullPage?: boolean }, { screenshot?: string; mode?: string; error?: string }>({\n description: 'Capture a screenshot of the current page',\n inputSchema: z.object({\n fullPage: z.boolean().optional().describe('Whether to capture the full page'),\n }),\n execute: async ({ fullPage }) => {\n track('screenshot');\n log(`screenshot called fullPage=${fullPage}`);\n if (config.dryRun) return { screenshot: undefined, mode: 'dry-run' };\n if (!page?.screenshot) return { error: 'Page unavailable' };\n try {\n const buffer = await page.screenshot({ fullPage: fullPage ?? false });\n const b = typeof buffer === 'string' ? Buffer.from(buffer) : Buffer.from(buffer);\n const base64 = b.toString('base64');\n // base64 length is ~4/3 of bytes; guard with both byte and payload caps\n if (b.byteLength > maxPayloadBytes || base64.length > maxPayloadBytes) {\n return { error: 'screenshot_too_large' };\n }\n const dataUrl = `data:image/png;base64,${truncate(base64, maxPayloadBytes)}`;\n log(`screenshot succeeded`);\n return { screenshot: dataUrl };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`screenshot error: ${msg}`);\n return { error: msg };\n }\n },\n }),\n\n get_text: tool<{ selector: string }, { text?: string | null; mode?: string; error?: string }>({\n description: 'Get the text content of an element',\n inputSchema: z.object({ selector: z.string().describe('CSS or Playwright selector') }),\n execute: async ({ selector }) => {\n track(`get_text:${selector}`);\n log(`get_text called with selector=\"${selector}\"`);\n if (config.dryRun) return { text: null, mode: 'dry-run' };\n if (!page?.locator) return { error: 'Page unavailable' };\n try {\n const el = page.locator(selector);\n const text = el?.textContent ? await el.textContent() : null;\n const sanitized = text ? truncate(sanitizeString(text), maxPayloadBytes) : text;\n log(`get_text succeeded: \"${sanitized}\"`);\n return { text: sanitized };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`get_text error: ${msg}`);\n return { error: msg };\n }\n },\n }),\n\n is_visible: tool<{ selector: string }, { visible?: boolean; mode?: string; error?: string }>({\n description: 'Check if an element is visible',\n inputSchema: z.object({ selector: z.string().describe('CSS or Playwright selector') }),\n execute: async ({ selector }) => {\n track(`is_visible:${selector}`);\n log(`is_visible called with selector=\"${selector}\"`);\n if (config.dryRun) return { visible: false, mode: 'dry-run' };\n if (!page?.locator) return { error: 'Page unavailable' };\n try {\n const el = page.locator(selector);\n const visible = el?.isVisible ? await el.isVisible() : false;\n log(`is_visible succeeded: ${visible}`);\n return { visible };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`is_visible error: ${msg}`);\n return { error: msg };\n }\n },\n }),\n\n evaluate: tool<{ script: string }, { result?: unknown; mode?: string; error?: string }>({\n description: 'Run JavaScript in the page context',\n inputSchema: z.object({ script: z.string().describe('JavaScript code to execute') }),\n execute: async ({ script }) => {\n track('evaluate');\n log(`evaluate called`);\n if (!config.allowEvaluate) return { error: 'evaluate_disabled' };\n if (violatesCodeDenylist(script)) return { error: 'evaluate_blocked' };\n if (config.dryRun) return { result: null, mode: 'dry-run' };\n if (!page?.evaluate) return { error: 'Page unavailable' };\n try {\n const scriptSafe = truncate(script, maxPayloadBytes);\n const result = await page.evaluate(scriptSafe);\n log(`evaluate succeeded`);\n return { result: sanitizeUnknown(result, maxPayloadBytes) };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`evaluate error: ${msg}`);\n return { error: msg };\n }\n },\n }),\n\n // === Power Tool: Arbitrary Playwright Code ===\n run_playwright_code: tool<{ code: string }, { executed: boolean; result?: unknown; mode?: string; error?: string }>({\n description:\n 'Execute arbitrary Playwright code. Has access to `page` object. Use for complex operations not covered by other tools. Example: `await page.getByRole(\"button\", { name: \"Submit\" }).click();`',\n inputSchema: z.object({\n code: z.string().describe('Playwright code to execute (has access to `page` object)'),\n }),\n execute: async ({ code }) => {\n track('run_playwright_code');\n log(`run_playwright_code called with code:\\n${code}`);\n if (!config.allowRunCode) return { executed: false, error: 'run_code_disabled' };\n if (violatesCodeDenylist(code)) return { executed: false, error: 'run_code_blocked' };\n if (config.dryRun) return { executed: false, mode: 'dry-run' };\n if (!page) return { executed: false, error: 'Page unavailable' };\n try {\n // Execute the code with page in scope\n const boundedCode = truncate(code, maxPayloadBytes);\n const fn = new Function('page', `return (async () => { ${boundedCode} })();`);\n const result = await fn(page);\n log(`run_playwright_code succeeded`);\n return { executed: true, result: sanitizeUnknown(result, maxPayloadBytes) };\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n log(`run_playwright_code error: ${msg}`);\n return { executed: false, error: msg };\n }\n },\n }),\n\n // === Completion Tool ===\n mark_complete: tool<{ reason: string; summary: string }, { complete: boolean; reason: string; summary: string; mode?: string }>({\n description:\n 'Call this when the browser is ready for the test to continue. You MUST call this when done healing!',\n inputSchema: z.object({\n reason: z.string().describe('Brief technical description of what was done'),\n summary: z.string().describe('One-line human-readable summary for the test report, e.g. \"Clicked \\'Send OTP\\' instead of \\'Submit\\'\" or \"Filled email field by accessible name\"'),\n }),\n execute: async ({ reason, summary }) => {\n track('mark_complete');\n log(`mark_complete called with reason=\"${reason}\" summary=\"${summary}\"`);\n // In dry-run mode, don't actually mark as complete - the original action should still fail\n if (!config.dryRun) {\n complete = true;\n completionReason = reason;\n completionSummary = summary;\n }\n return { complete: !config.dryRun, reason, summary, mode: config.dryRun ? 'dry-run' : undefined };\n },\n }),\n };\n\n return {\n tools,\n isComplete: () => complete,\n getCompletionReason: () => completionReason,\n getSummary: () => completionSummary,\n };\n}\n\n// === System Prompt ===\n\nfunction buildSystemPrompt(\n testContext: TestContext,\n config: ReturnType<typeof loadCanaryConfig>\n): string {\n const testInfo = testContext.testFile\n ? `\n## Test Context\nFile: ${testContext.testFile}\nTest: \"${testContext.testTitle ?? 'unknown'}\"\n`\n : '';\n\n const testSource = testContext.testSource\n ? `\n### Full Test Code:\n\\`\\`\\`typescript\n${testContext.testSource}\n\\`\\`\\`\n`\n : '';\n\n const expectedAfter = testContext.expectedAfter\n ? `\n### What the Test Expects After This Step:\n${testContext.expectedAfter}\n`\n : '';\n\n return `You are an AI test healer. A Playwright test step failed. Your job is to make the browser ready for the next step in the test **without violating the intent of the test**. Heal only the failing step; do not advance beyond it.\n\n## Your Goal\nGet the browser into the state the test expects, then call mark_complete.\n${testInfo}${testSource}\n### Current Step That Failed:\n${testContext.currentStep}\n${expectedAfter}\n## The Failure\nAction attempted: ${testContext.action}\nTarget: ${testContext.target ?? 'unknown'}\nError: ${testContext.errorMessage ?? 'unknown'}\n\n## Test Intent and Discipline\n- Adhere to the spirit of the test. If advancing would bypass a required validation or skip a missing field, you must not proceed—report failure instead.\n- Heal only this step. Avoid extra navigation or state changes unrelated to the intended action. If you take a detour, undo it before completion.\n- If you cannot confidently achieve the expected postcondition for this step, do not mark complete; let the test fail.\n\n## Tools Available\n\n### Inspection (use first!)\n- snapshot() - Get accessibility tree to see page structure\n- screenshot() - Capture visual screenshot\n- get_text(selector) - Get text content of an element\n- is_visible(selector) - Check if element is visible\n\n### Actions\n- click(text) - Click element by visible text\n- click_selector(selector) - Click by CSS/Playwright selector\n- fill_by_name(name, value) - PREFERRED: Fill textbox by accessible name from snapshot (e.g. textbox \"Enter 6-digit code\")\n- fill(selector, value) - Fill input by CSS selector (use placeholder attribute, NOT aria-label)\n- type(selector, text) - Type text character by character\n- press_key(key) - Press keyboard key (Enter, Tab, etc.)\n- hover(selector) - Hover over element\n- select_option(selector, value) - Select from dropdown\n\n### Navigation & Waiting\n- wait(ms) - Wait for a duration\n- wait_for_selector(selector, state?) - Wait for element state\n- navigate(url) - Navigate to URL\n- go_back() - Go back in history\n- reload() - Reload page\n\n### Power Tool\n- run_playwright_code(code) - Execute any Playwright code with \\`page\\` object\n Example: \\`await page.getByRole('button', { name: 'Submit' }).click();\\`\n\n### Completion (REQUIRED)\n- mark_complete(reason, summary) - Call when browser is ready for test to continue\n - reason: technical description of what was done\n - summary: one-line human-readable summary for test report (e.g. \"Clicked 'Send OTP' instead of 'Submit'\")\n\n## CRITICAL RULES - READ CAREFULLY\n\n### Rule 1: YOU MUST PERFORM THE ACTION YOURSELF\nThe test tried to ${testContext.action}(\"${testContext.target ?? ''}\") but it failed.\nYOU must perform the equivalent action on the correct element.\nDO NOT leave it for \"the next step\" - there is no next step until YOU complete the action.\n\n### Rule 2: The workflow is ALWAYS:\n1. snapshot() - understand the page\n2. click() / fill() / type() - PERFORM THE ACTION on the correct element\n3. mark_complete() - report what you did\n\n### Rule 3: NEVER call mark_complete without performing an action\nIf mark_complete is called without a click/fill/type, the test WILL FAIL.\n\n## Example 1: Test tried click(\"Submit\") but button says \"Send OTP\"\nCORRECT:\n1. snapshot() → see \"Send OTP\" button exists\n2. click(\"Send OTP\") → CLICK THE BUTTON\n3. mark_complete({reason: \"clicked Send OTP button\", summary: \"Clicked 'Send OTP' instead of 'Submit'\"})\n\nWRONG (this will cause test failure):\n1. snapshot() → see \"Send OTP\" button exists\n2. mark_complete(...) ← WRONG! No click performed!\n\n## Example 2: Test tried fill with wrong placeholder\nIf snapshot shows: textbox \"Enter 6-digit code\" [ref=e37]\nCORRECT:\n1. snapshot() → see textbox \"Enter 6-digit code\"\n2. fill_by_name(\"Enter 6-digit code\", \"111222\") → USE THE NAME FROM SNAPSHOT\n3. mark_complete({reason: \"filled by accessible name\", summary: \"Filled 'Enter 6-digit code' field\"})\n\n## If healing would break test intent\n- If the page is missing required UI (e.g., a field not present) or advancing would skip validation, do NOT hack around it. Report failure by not calling mark_complete.\n- If you temporarily navigate or change state, undo it before completion so the test can continue from the intended point.\n\n## Start now\n1. Call snapshot()\n2. Identify the correct element for the intended action\n3. PERFORM THE ACTION (click, fill, type, etc.)\n4. Call mark_complete with what you did\n\nMode: ${resolveMode(config)}.`;\n}\n\n// === Utilities ===\n\nfunction resolveMode(config: ReturnType<typeof loadCanaryConfig>): 'dry-run' | 'warn' | 'full' {\n if (config.dryRun) return 'dry-run';\n if (config.warnOnly) return 'warn';\n return 'full';\n}\n\nfunction baseOutcome({\n started,\n healed,\n shouldRetryOriginal,\n mode,\n reason,\n actionsRun = [],\n errors = [],\n}: {\n started: number;\n healed: boolean;\n shouldRetryOriginal: boolean;\n mode: 'dry-run' | 'warn' | 'full';\n reason?: string;\n actionsRun?: string[];\n errors?: string[];\n}): HealOutcome {\n return {\n healed,\n shouldRetryOriginal,\n actionsRun,\n timedOut: false,\n durationMs: Date.now() - started,\n errors,\n mode,\n reason,\n };\n}\n\ntype ToolStep = {\n finishReason?: string;\n toolCalls?: Array<{ toolName?: string; args?: unknown }>;\n toolResults?: Array<{ toolName?: string; result?: unknown; error?: unknown }>;\n};\n\nfunction logToolStep(step: ToolStep): void {\n const color = {\n magenta: '\\x1b[35m',\n cyan: '\\x1b[36m',\n yellow: '\\x1b[33m',\n green: '\\x1b[32m',\n reset: '\\x1b[0m',\n } as const;\n\n const calls = step.toolCalls ?? [];\n const results = step.toolResults ?? [];\n\n if (calls.length === 0 && results.length === 0) {\n // eslint-disable-next-line no-console\n console.log(\n `${color.magenta}[canary][tool] step${color.reset} (no tool calls/results; finish=${step.finishReason ?? 'unknown'})`\n );\n }\n\n for (const call of calls) {\n const name = call.toolName ?? 'unknown_tool';\n const args = call.args ? truncate(safeJson(call.args), 300) : '';\n // eslint-disable-next-line no-console\n console.log(\n `${color.magenta}[canary][tool] call${color.reset} ${color.cyan}${name}${color.reset}${args ? ` args=${args}` : ''}`\n );\n }\n\n for (const result of results) {\n const name = result.toolName ?? 'unknown_tool';\n const res = result.result ? truncate(safeJson(result.result), 300) : undefined;\n const err = result.error ? truncate(safeJson(result.error), 300) : undefined;\n const statusColor = err ? color.yellow : color.green;\n // eslint-disable-next-line no-console\n console.log(\n `${color.magenta}[canary][tool] result${color.reset} ${color.cyan}${name}${color.reset}` +\n (res ? ` ${statusColor}->${color.reset} ${res}` : '') +\n (err ? ` ${color.yellow}error=${err}${color.reset}` : '')\n );\n }\n}\n\nfunction buildInitialMessages(\n testContext: TestContext,\n failure: FailureContext,\n initial: { snapshot?: unknown; html?: string; screenshot?: string | undefined }\n): Array<{ role: 'user'; content: string }> {\n const parts: string[] = [];\n parts.push(\n `Please heal this test step. Start by reviewing the snapshot below. Failing step: ${failure.action ?? 'unknown'}(${failure.target ?? 'unknown'}). Error: ${failure.errorMessage ?? 'unknown'}.`\n );\n\n if (testContext.testSource) {\n parts.push(\n `Full test source (truncated/redacted):\\n\\`\\`\\`typescript\\n${truncate(testContext.testSource, 15000)}\\n\\`\\`\\``\n );\n }\n\n parts.push(`Test file: ${testContext.testFile ?? 'unknown'}`);\n parts.push(`Test title: ${testContext.testTitle ?? 'unknown'}`);\n\n if (initial.snapshot) {\n parts.push(`Initial accessibility snapshot (sanitized):\\n\\`\\`\\`json\\n${truncate(safeJson(initial.snapshot), 8000)}\\n\\`\\`\\``);\n }\n if (initial.html) {\n parts.push(`HTML fallback (sanitized):\\n\\`\\`\\`html\\n${truncate(initial.html, 4000)}\\n\\`\\`\\``);\n }\n if (initial.screenshot) {\n parts.push(`Screenshot (base64 data URL, truncated): ${truncate(initial.screenshot, 12000)}`);\n }\n\n parts.push(`Remember: perform only the intended step and mark_complete only when the postcondition for this step is truly met.`);\n\n return [{ role: 'user', content: parts.join('\\n\\n') }];\n}\n\nasync function buildInitialPageContext(\n page: PageLike | undefined,\n config: ReturnType<typeof loadCanaryConfig>\n): Promise<{ snapshot?: unknown; html?: string; screenshot?: string }> {\n const maxPayloadBytes = Math.max(1000, config.maxPayloadBytes || 60000);\n const result: { snapshot?: unknown; html?: string; screenshot?: string } = {};\n if (!page) return result;\n\n // Try AX snapshot\n try {\n if (page.accessibility?.snapshot) {\n const tree = await page.accessibility.snapshot();\n if (tree) {\n result.snapshot = sanitizeUnknown(tree, maxPayloadBytes);\n }\n }\n } catch {\n // ignore\n }\n\n // HTML fallback\n if (!result.snapshot && page.evaluate) {\n try {\n const html = (await page.evaluate('document.body.innerHTML')) as string;\n if (html) {\n result.html = truncate(sanitizeString(html), maxPayloadBytes);\n }\n } catch {\n // ignore\n }\n }\n\n // Optional screenshot (only if vision enabled)\n if (config.visionEnabled && page.screenshot) {\n try {\n const buffer = await page.screenshot({ fullPage: false });\n const b = typeof buffer === 'string' ? Buffer.from(buffer) : Buffer.from(buffer);\n const base64 = b.toString('base64');\n if (b.byteLength <= maxPayloadBytes && base64.length <= maxPayloadBytes) {\n result.screenshot = `data:image/png;base64,${truncate(base64, maxPayloadBytes)}`;\n }\n } catch {\n // ignore\n }\n }\n\n return result;\n}\n\nfunction truncate(value: string, max: number): string {\n if (value.length <= max) return value;\n return value.slice(0, max) + '...';\n}\n\nfunction safeJson(value: unknown): string {\n try {\n return JSON.stringify(value);\n } catch {\n return String(value);\n }\n}\n\nexport function sanitizeString(value: string): string {\n let result = value;\n for (const { regex, replacement } of REDACTION_PATTERNS) {\n result = result.replace(regex, replacement);\n }\n return result;\n}\n\nexport function sanitizeUnknown(value: unknown, maxPayloadBytes: number, seen = new WeakSet()): unknown {\n if (value === null || value === undefined) return value;\n if (typeof value === 'string') {\n return truncate(sanitizeString(value), maxPayloadBytes);\n }\n if (typeof value === 'number' || typeof value === 'boolean') return value;\n if (typeof value === 'object') {\n if (seen.has(value as object)) return '[REDACTED_CYCLE]';\n seen.add(value as object);\n if (Array.isArray(value)) {\n return value.slice(0, 50).map((v) => sanitizeUnknown(v, maxPayloadBytes, seen));\n }\n const out: Record<string, unknown> = {};\n const entries = Object.entries(value as Record<string, unknown>).slice(0, 50);\n for (const [k, v] of entries) {\n out[k] = sanitizeUnknown(v, maxPayloadBytes, seen);\n }\n return out;\n }\n return '[REDACTED_UNKNOWN]';\n}\n\nexport function violatesCodeDenylist(code: string): boolean {\n return CODE_DENY_PATTERNS.some((p) => p.test(code));\n}\n","import { createOpenAI } from \"@ai-sdk/openai\";\nimport { createAnthropic } from \"@ai-sdk/anthropic\";\nimport { createAzure } from \"@ai-sdk/azure\";\nimport type { LanguageModel } from \"ai\";\nimport { loadCanaryConfig, type CanaryConfig } from \"./config\";\n\ntype ResolvedModel = {\n model: LanguageModel | null;\n modelId?: string;\n reason?: string;\n};\n\nexport function resolveHealerModel(config: CanaryConfig = loadCanaryConfig()): ResolvedModel {\n if (!config.enabled) {\n return { model: null, reason: \"healing_disabled\" };\n }\n\n if (!config.apiKey) {\n return { model: null, reason: \"missing_api_key\" };\n }\n\n const provider = (config.aiProvider ?? \"openai\").toLowerCase();\n const modelId = config.aiModel || (provider === \"anthropic\" ? \"claude-3-5-haiku-20241022\" : \"gpt-4o-mini\");\n\n try {\n switch (provider) {\n case \"openai\": {\n const client = createOpenAI({ apiKey: config.apiKey, baseURL: process.env.AI_BASE_URL });\n return { model: client(modelId), modelId };\n }\n case \"anthropic\": {\n const client = createAnthropic({ apiKey: config.apiKey, baseURL: process.env.AI_BASE_URL });\n return { model: client(modelId), modelId };\n }\n case \"azure\": {\n const resourceName = process.env.AZURE_OPENAI_RESOURCE_NAME;\n const apiVersion = process.env.AZURE_OPENAI_API_VERSION;\n if (!resourceName) {\n return { model: null, reason: \"missing_azure_resource\" };\n }\n const client = createAzure({\n apiKey: config.apiKey,\n resourceName,\n apiVersion,\n });\n return { model: client(modelId), modelId };\n }\n default:\n return { model: null, reason: \"unsupported_provider\" };\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n return { model: null, reason: `model_error:${message}` };\n }\n}\n","import fs from \"node:fs\";\nimport path from \"node:path\";\nimport { loadCanaryConfig } from \"./config\";\nimport { loadCanaryEnv } from \"./env\";\n\nexport type HealingEvent = {\n kind: \"locator\" | \"page\" | \"expect\" | \"unknown\";\n action?: string;\n target?: string;\n errorMessage?: string;\n healed: boolean;\n workerId?: string;\n timestamp?: string;\n decision?: string;\n reason?: string;\n strategy?: string;\n actions?: Array<{ type: string; detail?: string }>;\n durationMs?: number;\n mode?: \"dry-run\" | \"warn\" | \"full\";\n modelId?: string;\n // Report fields\n summary?: string; // Human-readable one-line healing summary from agent\n testFile?: string; // Test file path for correlation\n testTitle?: string; // Test title for correlation\n};\n\ndeclare global {\n // eslint-disable-next-line no-var\n var __CANARY_EVENTS: HealingEvent[] | undefined;\n // eslint-disable-next-line no-var\n var __CANARY_PATCHED: boolean | undefined;\n // eslint-disable-next-line no-var\n var __CANARY_EVENT_LOG_PATH: string | undefined;\n}\n\nexport function getEventLog(): HealingEvent[] {\n if (!globalThis.__CANARY_EVENTS) {\n globalThis.__CANARY_EVENTS = [];\n }\n return globalThis.__CANARY_EVENTS;\n}\n\nexport function setEventLogPath(path: string): void {\n globalThis.__CANARY_EVENT_LOG_PATH = path;\n}\n\nexport function getEventLogPath(): string | undefined {\n return globalThis.__CANARY_EVENT_LOG_PATH;\n}\n\nexport function recordHealingEvent(event: HealingEvent): void {\n loadCanaryEnv();\n const log = getEventLog();\n const entry: HealingEvent = {\n ...event,\n workerId: process.env.TEST_WORKER_INDEX,\n timestamp: new Date().toISOString(),\n };\n log.push(entry);\n appendEvent(entry);\n}\n\nexport function markPatched(): void {\n globalThis.__CANARY_PATCHED = true;\n}\n\nexport function alreadyPatched(): boolean {\n return globalThis.__CANARY_PATCHED === true;\n}\n\nfunction appendEvent(event: HealingEvent): void {\n const logPath = getEventLogPath();\n if (!logPath) return;\n const config = loadCanaryConfig();\n if (!config.eventLoggingEnabled) return;\n\n try {\n fs.mkdirSync(path.dirname(logPath), { recursive: true });\n fs.appendFileSync(logPath, JSON.stringify(event) + \"\\n\");\n } catch {\n // Swallow logging errors to avoid impacting tests.\n }\n}\n","import { installInstrumentation } from \"./instrumentation\";\nimport { getEventLog } from \"./state\";\n\n// Ensure globals are initialized and patch Playwright classes in each worker process.\ngetEventLog();\ninstallInstrumentation();\n"],"mappings":";;;;;;;;;AAAA,OAAO,YAAY;AACnB,SAAS,qBAAqB;;;ACD9B,OAAOA,WAAU;;;ACAjB,OAAO,QAAQ;AACf,OAAO,UAAU;AAEjB,IAAI,SAAS;AAEN,SAAS,gBAAsB;AACpC,MAAI,OAAQ;AACZ,WAAS;AAET,QAAM,eAAe,QAAQ,IAAI;AACjC,QAAM,cAAc,KAAK,KAAK,QAAQ,IAAI,GAAG,MAAM;AACnD,QAAM,UAAU,gBAAgB;AAEhC,MAAI,GAAG,WAAW,OAAO,GAAG;AAC1B,aAAS,OAAO;AAAA,EAClB;AACF;AAEA,SAAS,SAAS,UAAwB;AACxC,MAAI;AACF,UAAM,MAAM,GAAG,aAAa,UAAU,OAAO;AAC7C,eAAW,QAAQ,IAAI,MAAM,IAAI,GAAG;AAClC,YAAM,UAAU,KAAK,KAAK;AAC1B,UAAI,CAAC,WAAW,QAAQ,WAAW,GAAG,EAAG;AACzC,YAAM,MAAM,QAAQ,QAAQ,GAAG;AAC/B,UAAI,QAAQ,GAAI;AAChB,YAAM,MAAM,QAAQ,MAAM,GAAG,GAAG,EAAE,KAAK;AACvC,YAAM,QAAQ,QAAQ,MAAM,MAAM,CAAC,EAAE,KAAK;AAC1C,UAAI,CAAC,IAAK;AACV,UAAI,QAAQ,IAAI,GAAG,MAAM,QAAW;AAClC,gBAAQ,IAAI,GAAG,IAAI,YAAY,KAAK;AAAA,MACtC;AAAA,IACF;AAAA,EACF,QAAQ;AAAA,EAER;AACF;AAEA,SAAS,YAAY,OAAuB;AAC1C,MAAK,MAAM,WAAW,GAAI,KAAK,MAAM,SAAS,GAAI,KAAO,MAAM,WAAW,GAAG,KAAK,MAAM,SAAS,GAAG,GAAI;AACtG,WAAO,MAAM,MAAM,GAAG,EAAE;AAAA,EAC1B;AACA,SAAO;AACT;;;ADpBO,SAAS,mBAAiC;AAC/C,gBAAc;AAEd,QAAM,gBAAgB,MAAM;AAC1B,UAAM,WAAW,QAAQ,IAAI;AAC7B,QAAI,SAAU,QAAO;AACrB,UAAM,SAAS,QAAQ,IAAI,qBAAqB;AAChD,WAAOC,MAAK,KAAK,QAAQ,IAAI,GAAG,gBAAgB,aAAa,iBAAiB,MAAM,QAAQ;AAAA,EAC9F,GAAG;AAEH,QAAM,aAA2B;AAAA,IAC/B,SAAS;AAAA,IACT,0BAA0B,QAAQ,IAAI;AAAA,IACtC,YAAY,QAAQ,IAAI;AAAA,IACxB,SAAS,QAAQ,IAAI;AAAA,IACrB,QAAQ,QAAQ,IAAI;AAAA,IACpB,eAAe,OAAO,QAAQ,IAAI,iBAAiB,IAAM;AAAA;AAAA,IACzD,YAAY,OAAO,QAAQ,IAAI,sBAAsB,EAAE;AAAA;AAAA,IACvD,QAAQ,QAAQ,IAAI,mBAAmB;AAAA,IACvC,UAAU,QAAQ,IAAI,qBAAqB;AAAA,IAC3C,eAAe,QAAQ,IAAI,kBAAkB,OAAO,QAAQ,IAAI,cAAc;AAAA,IAC9E,OAAO,QAAQ,IAAI,iBAAiB;AAAA,IACpC,UAAU,QAAQ,IAAI,qBAAqB;AAAA,IAC3C,cAAc,QAAQ,IAAI,0BAA0B;AAAA,IACpD,eAAe,QAAQ,IAAI,0BAA0B;AAAA,IACrD,iBAAiB,OAAO,QAAQ,IAAI,4BAA4B,GAAK;AAAA;AAAA,IACrE;AAAA,IACA,qBAAqB,QAAQ,IAAI,qBAAqB;AAAA,EACxD;AAEA,QAAM,WACJ,QAAQ,IAAI,mBAAmB,OAC/B,QAAQ,IAAI,oBAAoB,OAChC,QAAQ,IAAI,eAAe;AAE7B,SAAO;AAAA,IACL,GAAG;AAAA,IACH,SAAS,CAAC,YAAY,WAAW;AAAA,EACnC;AACF;;;AE9DA,SAAS,aAAa,YAAY,YAAuB;AACzD,SAAS,SAAS;;;ACDlB,SAAS,oBAAoB;AAC7B,SAAS,uBAAuB;AAChC,SAAS,mBAAmB;AAUrB,SAAS,mBAAmB,SAAuB,iBAAiB,GAAkB;AAC3F,MAAI,CAAC,OAAO,SAAS;AACnB,WAAO,EAAE,OAAO,MAAM,QAAQ,mBAAmB;AAAA,EACnD;AAEA,MAAI,CAAC,OAAO,QAAQ;AAClB,WAAO,EAAE,OAAO,MAAM,QAAQ,kBAAkB;AAAA,EAClD;AAEA,QAAM,YAAY,OAAO,cAAc,UAAU,YAAY;AAC7D,QAAM,UAAU,OAAO,YAAY,aAAa,cAAc,8BAA8B;AAE5F,MAAI;AACF,YAAQ,UAAU;AAAA,MAChB,KAAK,UAAU;AACb,cAAM,SAAS,aAAa,EAAE,QAAQ,OAAO,QAAQ,SAAS,QAAQ,IAAI,YAAY,CAAC;AACvF,eAAO,EAAE,OAAO,OAAO,OAAO,GAAG,QAAQ;AAAA,MAC3C;AAAA,MACA,KAAK,aAAa;AAChB,cAAM,SAAS,gBAAgB,EAAE,QAAQ,OAAO,QAAQ,SAAS,QAAQ,IAAI,YAAY,CAAC;AAC1F,eAAO,EAAE,OAAO,OAAO,OAAO,GAAG,QAAQ;AAAA,MAC3C;AAAA,MACA,KAAK,SAAS;AACZ,cAAM,eAAe,QAAQ,IAAI;AACjC,cAAM,aAAa,QAAQ,IAAI;AAC/B,YAAI,CAAC,cAAc;AACjB,iBAAO,EAAE,OAAO,MAAM,QAAQ,yBAAyB;AAAA,QACzD;AACA,cAAM,SAAS,YAAY;AAAA,UACzB,QAAQ,OAAO;AAAA,UACf;AAAA,UACA;AAAA,QACF,CAAC;AACD,eAAO,EAAE,OAAO,OAAO,OAAO,GAAG,QAAQ;AAAA,MAC3C;AAAA,MACA;AACE,eAAO,EAAE,OAAO,MAAM,QAAQ,uBAAuB;AAAA,IACzD;AAAA,EACF,SAAS,OAAO;AACd,UAAM,UAAU,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AACrE,WAAO,EAAE,OAAO,MAAM,QAAQ,eAAe,OAAO,GAAG;AAAA,EACzD;AACF;;;ADjDA,IAAM,qBAAoE;AAAA,EACxE,EAAE,OAAO,2BAA2B,aAAa,oBAAoB;AAAA,EACrE,EAAE,OAAO,qCAAqC,aAAa,qBAAqB;AAAA,EAChF,EAAE,OAAO,gCAAgC,aAAa,oBAAoB;AAAA,EAC1E,EAAE,OAAO,mDAAmD,aAAa,mBAAmB;AAAA,EAC5F,EAAE,OAAO,eAAe,aAAa,oBAAoB;AAC3D;AACA,IAAM,qBAAqB,CAAC,iBAAiB,kBAAkB,mBAAmB,gBAAgB;AAgF3F,SAAS,gBAAgB,SAAuC;AACrE,QAAM,WAAW,QAAQ,gBAAgB,IAAI,YAAY;AAGzD,MACE,QAAQ,SAAS,QAAQ,KACzB,QAAQ,SAAS,iDAAiD,GAClE;AACA,WAAO,EAAE,UAAU,OAAO,QAAQ,iBAAiB;AAAA,EACrD;AAGA,MAAI,QAAQ,SAAS,mBAAmB,KAAK,QAAQ,SAAS,OAAO,GAAG;AACtE,WAAO,EAAE,UAAU,OAAO,QAAQ,oBAAoB;AAAA,EACxD;AAGA,SAAO,EAAE,UAAU,MAAM,QAAQ,gBAAgB;AACnD;AAIA,eAAsB,mBACpB,UACA,SACA,SACsB;AACtB,QAAM,SAAS,iBAAiB;AAChC,QAAM,UAAU,KAAK,IAAI;AACzB,QAAM,OAAO,YAAY,MAAM;AAC/B,QAAM,aAAuB,CAAC;AAC9B,QAAM,SAAmB,CAAC;AAC1B,QAAM,WAAW,OAAO,SAAS,QAAQ,IAAI,oBAAoB;AACjE,QAAM,iBAAiB,MAAM,wBAAwB,QAAQ,MAAM,MAAM;AAEzE,MAAI,CAAC,SAAS,UAAU;AACtB,WAAO,YAAY;AAAA,MACjB;AAAA,MACA;AAAA,MACA,QAAQ;AAAA,MACR,qBAAqB;AAAA,MACrB,QAAQ,SAAS;AAAA,MACjB;AAAA,MACA;AAAA,IACF,CAAC;AAAA,EACH;AAEA,QAAM,EAAE,OAAO,SAAS,QAAQ,YAAY,IAAI,mBAAmB,MAAM;AACzE,MAAI,CAAC,OAAO;AACV,WAAO,YAAY;AAAA,MACjB;AAAA,MACA;AAAA,MACA,QAAQ;AAAA,MACR,qBAAqB;AAAA,MACrB,QAAQ,eAAe;AAAA,MACvB;AAAA,MACA;AAAA,IACF,CAAC;AAAA,EACH;AAGA,QAAM,cAA2B;AAAA,IAC/B,UAAU,QAAQ,aAAa;AAAA,IAC/B,WAAW,QAAQ,aAAa;AAAA,IAChC,YAAY,QAAQ,aAAa,aAAa,eAAe,QAAQ,YAAY,UAAU,IAAI;AAAA,IAC/F,aAAa,GAAG,QAAQ,UAAU,SAAS,IAAI,QAAQ,UAAU,SAAS;AAAA,IAC1E,eAAe,QAAQ,aAAa;AAAA,IACpC,QAAQ,QAAQ,UAAU;AAAA,IAC1B,QAAQ,QAAQ;AAAA,IAChB,cAAc,QAAQ;AAAA,EACxB;AAGA,QAAM,UAAU,mBAAmB;AAAA,IACjC,MAAM,QAAQ;AAAA,IACd;AAAA,IACA;AAAA,IACA,OAAO,OAAO;AAAA,EAChB,CAAC;AAED,MAAI;AAEJ,MAAI;AACF,UAAM,QAAQ,YAAY,QAAQ,OAAO,aAAa;AACtD,UAAM,SAAS,MAAM,WAAW;AAAA,MAC9B;AAAA,MACA,QAAQ,kBAAkB,aAAa,MAAM;AAAA,MAC7C,UAAU,qBAAqB,aAAa,SAAS,cAAc;AAAA,MACnE,OAAO,QAAQ;AAAA,MACf,UAAU,YAAY,KAAK,IAAI,GAAG,OAAO,UAAU,CAAC;AAAA,MACpD,aAAa;AAAA,MACb,YAAY;AAAA,MACZ,cAAc,CAAC,SAAS;AACtB,YAAI,OAAO,OAAO;AAEhB,kBAAQ;AAAA,YACN,6CAA6C,KAAK,YAAY,eAAe,KAAK,WAAW,UAAU,CAAC,iBAAiB,KAAK,aAAa,UAAU,CAAC;AAAA,UACxJ;AAAA,QACF;AACA,YAAI,UAAU;AACZ,sBAAY,IAAI;AAAA,QAClB;AAAA,MACF;AAAA,IACF,CAAC;AAGD,qBAAiB,KAAK,OAAO,YAAY;AACvC,UAAI,QAAQ,WAAW,GAAG;AACxB,2BAAmB,QAAQ,oBAAoB;AAC/C,YAAI,OAAO,OAAO;AAEhB,kBAAQ,IAAI,0CAA0C,gBAAgB,EAAE;AAAA,QAC1E;AACA;AAAA,MACF;AAAA,IACF;AAAA,EACF,SAAS,OAAO;AACd,UAAM,UAAU,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AACrE,WAAO,KAAK,OAAO;AACnB,QAAI,OAAO,OAAO;AAEhB,cAAQ,IAAI,gCAAgC,OAAO,EAAE;AAAA,IACvD;AAAA,EACF;AAEA,QAAM,SAAS,QAAQ,WAAW;AAElC,SAAO;AAAA,IACL;AAAA,IACA,qBAAqB;AAAA;AAAA,IACrB;AAAA,IACA,UAAU,KAAK,IAAI,IAAI,UAAU,OAAO;AAAA,IACxC,YAAY,KAAK,IAAI,IAAI;AAAA,IACzB;AAAA,IACA;AAAA,IACA,QAAQ,qBAAqB,SAAS,iBAAiB;AAAA,IACvD;AAAA,IACA,SAAS,QAAQ,WAAW;AAAA,EAC9B;AACF;AAIO,SAAS,mBAAmB,QAWjC;AACA,QAAM,EAAE,MAAM,QAAQ,YAAY,MAAM,IAAI;AAC5C,MAAI,WAAW;AACf,MAAI;AACJ,MAAI;AACJ,QAAM,kBAAkB,KAAK,IAAI,KAAM,OAAO,mBAAmB,GAAK;AAEtE,QAAM,MAAM,CAAC,QAAgB;AAC3B,QAAI,OAAO;AAET,cAAQ,IAAI,mBAAmB,GAAG,EAAE;AAAA,IACtC;AAAA,EACF;AAEA,QAAM,QAAQ,CAAC,WAAmB;AAChC,eAAW,KAAK,MAAM;AAAA,EACxB;AAGA,QAAM,QAAwC;AAAA;AAAA,IAE5C,OAAO,KAA2E;AAAA,MAChF,aAAa;AAAA,MACb,aAAa,EAAE,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,SAAS,2BAA2B,EAAE,CAAC;AAAA,MAChF,SAAS,OAAO,EAAE,KAAK,MAAM;AAC3B,cAAM,SAAS,IAAI,EAAE;AACrB,YAAI,2BAA2B,IAAI,GAAG;AACtC,YAAI,OAAO,SAAU,QAAO,EAAE,SAAS,IAAI,OAAO,iBAAiB;AACnE,YAAI,OAAO,OAAQ,QAAO,EAAE,SAAS,MAAM,MAAM,UAAU;AAC3D,YAAI,CAAC,MAAM,UAAW,QAAO,EAAE,SAAS,IAAI,OAAO,mBAAmB;AACtE,YAAI;AACF,gBAAM,KAAK,KAAK,UAAU,IAAI;AAC9B,cAAI,IAAI,wBAAwB;AAC9B,kBAAM,GAAG,uBAAuB,EAAE,SAAS,IAAK,CAAC;AAAA,UACnD;AACA,gBAAM,YAAY;AAClB,cAAI,WAAW,OAAO;AACpB,kBAAM,UAAU,MAAM,EAAE,SAAS,IAAK,CAAC;AAAA,UACzC;AACA,cAAI,wBAAwB,IAAI,GAAG;AACnC,iBAAO,EAAE,SAAS,KAAK;AAAA,QACzB,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,gBAAgB,GAAG,EAAE;AACzB,iBAAO,EAAE,SAAS,IAAI,OAAO,IAAI;AAAA,QACnC;AAAA,MACF;AAAA,IACF,CAAC;AAAA,IAED,gBAAgB,KAA+E;AAAA,MAC7F,aAAa;AAAA,MACb,aAAa,EAAE,OAAO,EAAE,UAAU,EAAE,OAAO,EAAE,SAAS,4BAA4B,EAAE,CAAC;AAAA,MACrF,SAAS,OAAO,EAAE,SAAS,MAAM;AAC/B,cAAM,kBAAkB,QAAQ,EAAE;AAClC,YAAI,wCAAwC,QAAQ,GAAG;AACvD,YAAI,OAAO,SAAU,QAAO,EAAE,SAAS,IAAI,OAAO,iBAAiB;AACnE,YAAI,OAAO,OAAQ,QAAO,EAAE,SAAS,UAAU,MAAM,UAAU;AAC/D,YAAI,CAAC,MAAM,QAAS,QAAO,EAAE,SAAS,IAAI,OAAO,mBAAmB;AACpE,YAAI;AACF,gBAAM,KAAK,KAAK,QAAQ,QAAQ;AAChC,cAAI,IAAI,wBAAwB;AAC9B,kBAAM,GAAG,uBAAuB,EAAE,SAAS,IAAK,CAAC;AAAA,UACnD;AACA,cAAI,IAAI,OAAO;AACb,kBAAM,GAAG,MAAM,EAAE,SAAS,IAAK,CAAC;AAAA,UAClC;AACA,cAAI,iCAAiC,QAAQ,GAAG;AAChD,iBAAO,EAAE,SAAS,SAAS;AAAA,QAC7B,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,yBAAyB,GAAG,EAAE;AAClC,iBAAO,EAAE,SAAS,IAAI,OAAO,IAAI;AAAA,QACnC;AAAA,MACF;AAAA,IACF,CAAC;AAAA,IAED,MAAM,KAA6G;AAAA,MACjH,aAAa;AAAA,MACb,aAAa,EAAE,OAAO;AAAA,QACpB,UAAU,EAAE,OAAO,EAAE,SAAS,4DAA4D;AAAA,QAC1F,OAAO,EAAE,OAAO,EAAE,SAAS,eAAe;AAAA,MAC5C,CAAC;AAAA,MACD,SAAS,OAAO,EAAE,UAAU,MAAM,MAAM;AACtC,cAAM,QAAQ,QAAQ,EAAE;AACxB,YAAI,8BAA8B,QAAQ,YAAY,KAAK,GAAG;AAC9D,YAAI,OAAO,SAAU,QAAO,EAAE,QAAQ,IAAI,OAAO,iBAAiB;AAClE,YAAI,OAAO,OAAQ,QAAO,EAAE,QAAQ,UAAU,MAAM,UAAU;AAC9D,YAAI,CAAC,MAAM,QAAS,QAAO,EAAE,QAAQ,IAAI,OAAO,mBAAmB;AACnE,YAAI;AACF,gBAAM,KAAK,KAAK,QAAQ,QAAQ;AAChC,cAAI,IAAI,MAAM;AACZ,kBAAM,GAAG,KAAK,OAAO,EAAE,SAAS,IAAK,CAAC;AAAA,UACxC;AACA,cAAI,gBAAgB;AACpB,iBAAO,EAAE,QAAQ,UAAU,MAAM;AAAA,QACnC,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,eAAe,GAAG,EAAE;AACxB,iBAAO,EAAE,QAAQ,IAAI,OAAO,IAAI;AAAA,QAClC;AAAA,MACF;AAAA,IACF,CAAC;AAAA,IAED,cAAc,KAAyG;AAAA,MACrH,aAAa;AAAA,MACb,aAAa,EAAE,OAAO;AAAA,QACpB,MAAM,EAAE,OAAO,EAAE,SAAS,+EAA+E;AAAA,QACzG,OAAO,EAAE,OAAO,EAAE,SAAS,eAAe;AAAA,MAC5C,CAAC;AAAA,MACD,SAAS,OAAO,EAAE,MAAM,MAAM,MAAM;AAClC,cAAM,gBAAgB,IAAI,EAAE;AAC5B,YAAI,kCAAkC,IAAI,YAAY,KAAK,GAAG;AAC9D,YAAI,OAAO,SAAU,QAAO,EAAE,QAAQ,IAAI,OAAO,iBAAiB;AAClE,YAAI,OAAO,OAAQ,QAAO,EAAE,QAAQ,MAAM,MAAM,UAAU;AAC1D,YAAI,CAAC,MAAM,QAAS,QAAO,EAAE,QAAQ,IAAI,OAAO,mBAAmB;AACnE,YAAI;AAEF,gBAAM,KAAK,KAAK,QAAQ,sBAAsB,IAAI,IAAI;AACtD,cAAI,IAAI,MAAM;AACZ,kBAAM,GAAG,KAAK,OAAO,EAAE,SAAS,IAAK,CAAC;AAAA,UACxC;AACA,cAAI,wBAAwB;AAC5B,iBAAO,EAAE,QAAQ,MAAM,MAAM;AAAA,QAC/B,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,uBAAuB,GAAG,EAAE;AAChC,iBAAO,EAAE,QAAQ,IAAI,OAAO,IAAI;AAAA,QAClC;AAAA,MACF;AAAA,IACF,CAAC;AAAA,IAED,MAAM,KAA0G;AAAA,MAC9G,aAAa;AAAA,MACb,aAAa,EAAE,OAAO;AAAA,QACpB,UAAU,EAAE,OAAO,EAAE,SAAS,4BAA4B;AAAA,QAC1D,MAAM,EAAE,OAAO,EAAE,SAAS,cAAc;AAAA,MAC1C,CAAC;AAAA,MACD,SAAS,OAAO,EAAE,UAAU,KAAK,MAAM;AACrC,cAAM,QAAQ,QAAQ,EAAE;AACxB,YAAI,8BAA8B,QAAQ,WAAW,IAAI,GAAG;AAC5D,YAAI,OAAO,SAAU,QAAO,EAAE,OAAO,IAAI,OAAO,iBAAiB;AACjE,YAAI,OAAO,OAAQ,QAAO,EAAE,OAAO,MAAM,MAAM,UAAU;AACzD,YAAI,CAAC,MAAM,QAAS,QAAO,EAAE,OAAO,IAAI,OAAO,mBAAmB;AAClE,YAAI;AACF,gBAAM,KAAK,KAAK,QAAQ,QAAQ;AAChC,cAAI,IAAI,mBAAmB;AACzB,kBAAM,GAAG,kBAAkB,IAAI;AAAA,UACjC;AACA,cAAI,gBAAgB;AACpB,iBAAO,EAAE,OAAO,MAAM,MAAM,SAAS;AAAA,QACvC,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,eAAe,GAAG,EAAE;AACxB,iBAAO,EAAE,OAAO,IAAI,OAAO,IAAI;AAAA,QACjC;AAAA,MACF;AAAA,IACF,CAAC;AAAA,IAED,WAAW,KAA0E;AAAA,MACnF,aAAa;AAAA,MACb,aAAa,EAAE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,SAAS,cAAc,EAAE,CAAC;AAAA,MAClE,SAAS,OAAO,EAAE,IAAI,MAAM;AAC1B,cAAM,aAAa,GAAG,EAAE;AACxB,YAAI,8BAA8B,GAAG,GAAG;AACxC,YAAI,OAAO,SAAU,QAAO,EAAE,SAAS,IAAI,OAAO,iBAAiB;AACnE,YAAI,OAAO,OAAQ,QAAO,EAAE,SAAS,KAAK,MAAM,UAAU;AAC1D,YAAI,CAAC,MAAM,UAAU,MAAO,QAAO,EAAE,SAAS,IAAI,OAAO,mBAAmB;AAC5E,YAAI;AACF,gBAAM,KAAK,SAAS,MAAM,GAAG;AAC7B,cAAI,qBAAqB;AACzB,iBAAO,EAAE,SAAS,IAAI;AAAA,QACxB,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,oBAAoB,GAAG,EAAE;AAC7B,iBAAO,EAAE,SAAS,IAAI,OAAO,IAAI;AAAA,QACnC;AAAA,MACF;AAAA,IACF,CAAC;AAAA,IAED,OAAO,KAA+E;AAAA,MACpF,aAAa;AAAA,MACb,aAAa,EAAE,OAAO,EAAE,UAAU,EAAE,OAAO,EAAE,SAAS,4BAA4B,EAAE,CAAC;AAAA,MACrF,SAAS,OAAO,EAAE,SAAS,MAAM;AAC/B,cAAM,SAAS,QAAQ,EAAE;AACzB,YAAI,+BAA+B,QAAQ,GAAG;AAC9C,YAAI,OAAO,SAAU,QAAO,EAAE,SAAS,IAAI,OAAO,iBAAiB;AACnE,YAAI,OAAO,OAAQ,QAAO,EAAE,SAAS,UAAU,MAAM,UAAU;AAC/D,YAAI,CAAC,MAAM,QAAS,QAAO,EAAE,SAAS,IAAI,OAAO,mBAAmB;AACpE,YAAI;AACF,gBAAM,KAAK,KAAK,QAAQ,QAAQ;AAChC,cAAI,IAAI,OAAO;AACb,kBAAM,GAAG,MAAM;AAAA,UACjB;AACA,cAAI,iBAAiB;AACrB,iBAAO,EAAE,SAAS,SAAS;AAAA,QAC7B,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,gBAAgB,GAAG,EAAE;AACzB,iBAAO,EAAE,SAAS,IAAI,OAAO,IAAI;AAAA,QACnC;AAAA,MACF;AAAA,IACF,CAAC;AAAA,IAED,eAAe,KAA8G;AAAA,MAC3H,aAAa;AAAA,MACb,aAAa,EAAE,OAAO;AAAA,QACpB,UAAU,EAAE,OAAO,EAAE,SAAS,mDAAmD;AAAA,QACjF,OAAO,EAAE,OAAO,EAAE,SAAS,0BAA0B;AAAA,MACvD,CAAC;AAAA,MACD,SAAS,OAAO,EAAE,UAAU,MAAM,MAAM;AACtC,cAAM,iBAAiB,QAAQ,EAAE;AACjC,YAAI,uCAAuC,QAAQ,YAAY,KAAK,GAAG;AACvE,YAAI,OAAO,SAAU,QAAO,EAAE,UAAU,IAAI,OAAO,iBAAiB;AACpE,YAAI,OAAO,OAAQ,QAAO,EAAE,UAAU,OAAO,MAAM,UAAU;AAC7D,YAAI,CAAC,MAAM,QAAS,QAAO,EAAE,UAAU,IAAI,OAAO,mBAAmB;AACrE,YAAI;AACF,gBAAM,KAAK,KAAK,QAAQ,QAAQ;AAChC,cAAI,IAAI,cAAc;AACpB,kBAAM,GAAG,aAAa,KAAK;AAAA,UAC7B;AACA,cAAI,yBAAyB;AAC7B,iBAAO,EAAE,UAAU,OAAO,MAAM,SAAS;AAAA,QAC3C,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,wBAAwB,GAAG,EAAE;AACjC,iBAAO,EAAE,UAAU,IAAI,OAAO,IAAI;AAAA,QACpC;AAAA,MACF;AAAA,IACF,CAAC;AAAA;AAAA,IAGD,MAAM,KAAwD;AAAA,MAC5D,aAAa;AAAA,MACb,aAAa,EAAE,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,GAAG,EAAE,IAAI,GAAK,EAAE,SAAS,sBAAsB,EAAE,CAAC;AAAA,MAC7F,SAAS,OAAO,EAAE,GAAG,MAAM;AACzB,cAAM,QAAQ,EAAE,EAAE;AAClB,YAAI,uBAAuB,EAAE,EAAE;AAC/B,YAAI,OAAO,OAAQ,QAAO,EAAE,QAAQ,IAAI,MAAM,UAAU;AACxD,YAAI,MAAM,gBAAgB;AACxB,gBAAM,KAAK,eAAe,EAAE;AAAA,QAC9B,OAAO;AACL,gBAAM,IAAI,QAAQ,CAAC,YAAY,WAAW,SAAS,EAAE,CAAC;AAAA,QACxD;AACA,YAAI,gBAAgB;AACpB,eAAO,EAAE,QAAQ,GAAG;AAAA,MACtB;AAAA,IACF,CAAC;AAAA,IAED,mBAAmB,KAAqJ;AAAA,MACtK,aAAa;AAAA,MACb,aAAa,EAAE,OAAO;AAAA,QACpB,UAAU,EAAE,OAAO,EAAE,SAAS,4BAA4B;AAAA,QAC1D,OAAO,EAAE,KAAK,CAAC,WAAW,UAAU,YAAY,UAAU,CAAC,EAAE,SAAS,EAAE,SAAS,mBAAmB;AAAA,MACtG,CAAC;AAAA,MACD,SAAS,OAAO,EAAE,UAAU,MAAM,MAAM;AACtC,cAAM,qBAAqB,QAAQ,EAAE;AACrC,YAAI,2CAA2C,QAAQ,YAAY,SAAS,SAAS,GAAG;AACxF,YAAI,OAAO,OAAQ,QAAO,EAAE,OAAO,UAAU,MAAM,UAAU;AAC7D,YAAI,CAAC,MAAM,QAAS,QAAO,EAAE,OAAO,IAAI,OAAO,mBAAmB;AAClE,YAAI;AACF,gBAAM,KAAK,KAAK,QAAQ,QAAQ;AAChC,cAAI,IAAI,SAAS;AACf,kBAAM,GAAG,QAAQ,EAAE,OAAO,SAAS,UAAU,CAAC;AAAA,UAChD;AACA,cAAI,6BAA6B;AACjC,iBAAO,EAAE,OAAO,UAAU,OAAO,SAAS,UAAU;AAAA,QACtD,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,4BAA4B,GAAG,EAAE;AACrC,iBAAO,EAAE,OAAO,IAAI,OAAO,IAAI;AAAA,QACjC;AAAA,MACF;AAAA,IACF,CAAC;AAAA,IAED,UAAU,KAA4E;AAAA,MACpF,aAAa;AAAA,MACb,aAAa,EAAE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,SAAS,oBAAoB,EAAE,CAAC;AAAA,MACxE,SAAS,OAAO,EAAE,IAAI,MAAM;AAC1B,cAAM,YAAY,GAAG,EAAE;AACvB,YAAI,6BAA6B,GAAG,GAAG;AACvC,YAAI,OAAO,SAAU,QAAO,EAAE,WAAW,IAAI,OAAO,iBAAiB;AACrE,YAAI,OAAO,OAAQ,QAAO,EAAE,WAAW,KAAK,MAAM,UAAU;AAC5D,YAAI,CAAC,MAAM,KAAM,QAAO,EAAE,WAAW,IAAI,OAAO,mBAAmB;AACnE,YAAI;AACF,gBAAM,KAAK,KAAK,GAAG;AACnB,cAAI,oBAAoB;AACxB,iBAAO,EAAE,WAAW,IAAI;AAAA,QAC1B,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,mBAAmB,GAAG,EAAE;AAC5B,iBAAO,EAAE,WAAW,IAAI,OAAO,IAAI;AAAA,QACrC;AAAA,MACF;AAAA,IACF,CAAC;AAAA,IAED,SAAS,KAA+E;AAAA,MACtF,aAAa;AAAA,MACb,aAAa,EAAE,OAAO,CAAC,CAAC;AAAA,MACxB,SAAS,YAAY;AACnB,cAAM,SAAS;AACf,YAAI,gBAAgB;AACpB,YAAI,OAAO,SAAU,QAAO,EAAE,QAAQ,IAAI,OAAO,iBAAiB;AAClE,YAAI,OAAO,OAAQ,QAAO,EAAE,QAAQ,QAAQ,MAAM,UAAU;AAC5D,YAAI,CAAC,MAAM,OAAQ,QAAO,EAAE,QAAQ,IAAI,OAAO,mBAAmB;AAClE,YAAI;AACF,gBAAM,KAAK,OAAO;AAClB,cAAI,mBAAmB;AACvB,iBAAO,EAAE,QAAQ,OAAO;AAAA,QAC1B,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,kBAAkB,GAAG,EAAE;AAC3B,iBAAO,EAAE,QAAQ,IAAI,OAAO,IAAI;AAAA,QAClC;AAAA,MACF;AAAA,IACF,CAAC;AAAA,IAED,QAAQ,KAA+E;AAAA,MACrF,aAAa;AAAA,MACb,aAAa,EAAE,OAAO,CAAC,CAAC;AAAA,MACxB,SAAS,YAAY;AACnB,cAAM,QAAQ;AACd,YAAI,eAAe;AACnB,YAAI,OAAO,SAAU,QAAO,EAAE,QAAQ,IAAI,OAAO,iBAAiB;AAClE,YAAI,OAAO,OAAQ,QAAO,EAAE,QAAQ,UAAU,MAAM,UAAU;AAC9D,YAAI,CAAC,MAAM,OAAQ,QAAO,EAAE,QAAQ,IAAI,OAAO,mBAAmB;AAClE,YAAI;AACF,gBAAM,KAAK,OAAO;AAClB,cAAI,kBAAkB;AACtB,iBAAO,EAAE,QAAQ,SAAS;AAAA,QAC5B,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,iBAAiB,GAAG,EAAE;AAC1B,iBAAO,EAAE,QAAQ,IAAI,OAAO,IAAI;AAAA,QAClC;AAAA,MACF;AAAA,IACF,CAAC;AAAA;AAAA,IAGD,UAAU,KAA8F;AAAA,MACtG,aAAa;AAAA,MACb,aAAa,EAAE,OAAO,CAAC,CAAC;AAAA,MACxB,SAAS,YAAY;AACnB,cAAM,UAAU;AAChB,YAAI,iBAAiB;AACrB,YAAI,OAAO,OAAQ,QAAO,EAAE,MAAM,MAAM,MAAM,UAAU;AACxD,YAAI,CAAC,KAAM,QAAO,EAAE,OAAO,mBAAmB;AAE9C,YAAI;AACJ,YAAI;AAGJ,YAAI;AACF,cAAI,KAAK,iBAAiB,OAAO,KAAK,cAAc,aAAa,YAAY;AAC3E,mBAAO,MAAM,KAAK,cAAc,SAAS;AAAA,UAC3C;AAAA,QACF,SAAS,KAAK;AACZ,cAAI,iCAAiC,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC,EAAE;AAAA,QACzF;AAGA,YAAI,CAAC,QAAQ,KAAK,UAAU;AAC1B,cAAI;AACF,mBAAO,MAAM,KAAK,SAAS,yBAAyB;AACpD,gBAAI,MAAM;AACR,kBAAI,8BAA8B;AAAA,YACpC;AAAA,UACF,SAAS,KAAK;AACZ,gBAAI,wBAAwB,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC,EAAE;AAAA,UAChF;AAAA,QACF;AAEA,YAAI,CAAC,QAAQ,CAAC,MAAM;AAClB,iBAAO,EAAE,OAAO,6BAA6B;AAAA,QAC/C;AAEA,YAAI,oBAAoB;AACxB,cAAM,gBAAgB,gBAAgB,MAAM,eAAe;AAC3D,cAAM,gBAAgB,OAAO,SAAS,eAAe,IAAI,GAAG,eAAe,IAAI;AAC/E,eAAO,EAAE,MAAM,eAAe,MAAM,cAAc;AAAA,MACpD;AAAA,IACF,CAAC;AAAA,IAED,YAAY,KAAqF;AAAA,MAC/F,aAAa;AAAA,MACb,aAAa,EAAE,OAAO;AAAA,QACpB,UAAU,EAAE,QAAQ,EAAE,SAAS,EAAE,SAAS,kCAAkC;AAAA,MAC9E,CAAC;AAAA,MACD,SAAS,OAAO,EAAE,SAAS,MAAM;AAC/B,cAAM,YAAY;AAClB,YAAI,8BAA8B,QAAQ,EAAE;AAC5C,YAAI,OAAO,OAAQ,QAAO,EAAE,YAAY,QAAW,MAAM,UAAU;AACnE,YAAI,CAAC,MAAM,WAAY,QAAO,EAAE,OAAO,mBAAmB;AAC1D,YAAI;AACF,gBAAM,SAAS,MAAM,KAAK,WAAW,EAAE,UAAU,YAAY,MAAM,CAAC;AACpE,gBAAM,IAAI,OAAO,WAAW,WAAW,OAAO,KAAK,MAAM,IAAI,OAAO,KAAK,MAAM;AAC/E,gBAAM,SAAS,EAAE,SAAS,QAAQ;AAElC,cAAI,EAAE,aAAa,mBAAmB,OAAO,SAAS,iBAAiB;AACrE,mBAAO,EAAE,OAAO,uBAAuB;AAAA,UACzC;AACA,gBAAM,UAAU,yBAAyB,SAAS,QAAQ,eAAe,CAAC;AAC1E,cAAI,sBAAsB;AAC1B,iBAAO,EAAE,YAAY,QAAQ;AAAA,QAC/B,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,qBAAqB,GAAG,EAAE;AAC9B,iBAAO,EAAE,OAAO,IAAI;AAAA,QACtB;AAAA,MACF;AAAA,IACF,CAAC;AAAA,IAED,UAAU,KAAoF;AAAA,MAC5F,aAAa;AAAA,MACb,aAAa,EAAE,OAAO,EAAE,UAAU,EAAE,OAAO,EAAE,SAAS,4BAA4B,EAAE,CAAC;AAAA,MACrF,SAAS,OAAO,EAAE,SAAS,MAAM;AAC/B,cAAM,YAAY,QAAQ,EAAE;AAC5B,YAAI,kCAAkC,QAAQ,GAAG;AACjD,YAAI,OAAO,OAAQ,QAAO,EAAE,MAAM,MAAM,MAAM,UAAU;AACxD,YAAI,CAAC,MAAM,QAAS,QAAO,EAAE,OAAO,mBAAmB;AACvD,YAAI;AACF,gBAAM,KAAK,KAAK,QAAQ,QAAQ;AAChC,gBAAM,OAAO,IAAI,cAAc,MAAM,GAAG,YAAY,IAAI;AACxD,gBAAM,YAAY,OAAO,SAAS,eAAe,IAAI,GAAG,eAAe,IAAI;AAC3E,cAAI,wBAAwB,SAAS,GAAG;AACxC,iBAAO,EAAE,MAAM,UAAU;AAAA,QAC3B,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,mBAAmB,GAAG,EAAE;AAC5B,iBAAO,EAAE,OAAO,IAAI;AAAA,QACtB;AAAA,MACF;AAAA,IACF,CAAC;AAAA,IAED,YAAY,KAAiF;AAAA,MAC3F,aAAa;AAAA,MACb,aAAa,EAAE,OAAO,EAAE,UAAU,EAAE,OAAO,EAAE,SAAS,4BAA4B,EAAE,CAAC;AAAA,MACrF,SAAS,OAAO,EAAE,SAAS,MAAM;AAC/B,cAAM,cAAc,QAAQ,EAAE;AAC9B,YAAI,oCAAoC,QAAQ,GAAG;AACnD,YAAI,OAAO,OAAQ,QAAO,EAAE,SAAS,OAAO,MAAM,UAAU;AAC5D,YAAI,CAAC,MAAM,QAAS,QAAO,EAAE,OAAO,mBAAmB;AACvD,YAAI;AACF,gBAAM,KAAK,KAAK,QAAQ,QAAQ;AAChC,gBAAM,UAAU,IAAI,YAAY,MAAM,GAAG,UAAU,IAAI;AACvD,cAAI,yBAAyB,OAAO,EAAE;AACtC,iBAAO,EAAE,QAAQ;AAAA,QACnB,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,qBAAqB,GAAG,EAAE;AAC9B,iBAAO,EAAE,OAAO,IAAI;AAAA,QACtB;AAAA,MACF;AAAA,IACF,CAAC;AAAA,IAED,UAAU,KAA8E;AAAA,MACtF,aAAa;AAAA,MACb,aAAa,EAAE,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,SAAS,4BAA4B,EAAE,CAAC;AAAA,MACnF,SAAS,OAAO,EAAE,OAAO,MAAM;AAC7B,cAAM,UAAU;AAChB,YAAI,iBAAiB;AACrB,YAAI,CAAC,OAAO,cAAe,QAAO,EAAE,OAAO,oBAAoB;AAC/D,YAAI,qBAAqB,MAAM,EAAG,QAAO,EAAE,OAAO,mBAAmB;AACrE,YAAI,OAAO,OAAQ,QAAO,EAAE,QAAQ,MAAM,MAAM,UAAU;AAC1D,YAAI,CAAC,MAAM,SAAU,QAAO,EAAE,OAAO,mBAAmB;AACxD,YAAI;AACF,gBAAM,aAAa,SAAS,QAAQ,eAAe;AACnD,gBAAM,SAAS,MAAM,KAAK,SAAS,UAAU;AAC7C,cAAI,oBAAoB;AACxB,iBAAO,EAAE,QAAQ,gBAAgB,QAAQ,eAAe,EAAE;AAAA,QAC5D,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,mBAAmB,GAAG,EAAE;AAC5B,iBAAO,EAAE,OAAO,IAAI;AAAA,QACtB;AAAA,MACF;AAAA,IACF,CAAC;AAAA;AAAA,IAGD,qBAAqB,KAA+F;AAAA,MAClH,aACE;AAAA,MACF,aAAa,EAAE,OAAO;AAAA,QACpB,MAAM,EAAE,OAAO,EAAE,SAAS,0DAA0D;AAAA,MACtF,CAAC;AAAA,MACD,SAAS,OAAO,EAAE,KAAK,MAAM;AAC3B,cAAM,qBAAqB;AAC3B,YAAI;AAAA,EAA0C,IAAI,EAAE;AACpD,YAAI,CAAC,OAAO,aAAc,QAAO,EAAE,UAAU,OAAO,OAAO,oBAAoB;AAC/E,YAAI,qBAAqB,IAAI,EAAG,QAAO,EAAE,UAAU,OAAO,OAAO,mBAAmB;AACpF,YAAI,OAAO,OAAQ,QAAO,EAAE,UAAU,OAAO,MAAM,UAAU;AAC7D,YAAI,CAAC,KAAM,QAAO,EAAE,UAAU,OAAO,OAAO,mBAAmB;AAC/D,YAAI;AAEF,gBAAM,cAAc,SAAS,MAAM,eAAe;AAClD,gBAAM,KAAK,IAAI,SAAS,QAAQ,yBAAyB,WAAW,QAAQ;AAC5E,gBAAM,SAAS,MAAM,GAAG,IAAI;AAC5B,cAAI,+BAA+B;AACnC,iBAAO,EAAE,UAAU,MAAM,QAAQ,gBAAgB,QAAQ,eAAe,EAAE;AAAA,QAC5E,SAAS,KAAK;AACZ,gBAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,cAAI,8BAA8B,GAAG,EAAE;AACvC,iBAAO,EAAE,UAAU,OAAO,OAAO,IAAI;AAAA,QACvC;AAAA,MACF;AAAA,IACF,CAAC;AAAA;AAAA,IAGD,eAAe,KAAiH;AAAA,MAC9H,aACE;AAAA,MACF,aAAa,EAAE,OAAO;AAAA,QACpB,QAAQ,EAAE,OAAO,EAAE,SAAS,8CAA8C;AAAA,QAC1E,SAAS,EAAE,OAAO,EAAE,SAAS,+IAAmJ;AAAA,MAClL,CAAC;AAAA,MACD,SAAS,OAAO,EAAE,QAAQ,QAAQ,MAAM;AACtC,cAAM,eAAe;AACrB,YAAI,qCAAqC,MAAM,cAAc,OAAO,GAAG;AAEvE,YAAI,CAAC,OAAO,QAAQ;AAClB,qBAAW;AACX,6BAAmB;AACnB,8BAAoB;AAAA,QACtB;AACA,eAAO,EAAE,UAAU,CAAC,OAAO,QAAQ,QAAQ,SAAS,MAAM,OAAO,SAAS,YAAY,OAAU;AAAA,MAClG;AAAA,IACF,CAAC;AAAA,EACH;AAEA,SAAO;AAAA,IACL;AAAA,IACA,YAAY,MAAM;AAAA,IAClB,qBAAqB,MAAM;AAAA,IAC3B,YAAY,MAAM;AAAA,EACpB;AACF;AAIA,SAAS,kBACP,aACA,QACQ;AACR,QAAM,WAAW,YAAY,WACzB;AAAA;AAAA,QAEE,YAAY,QAAQ;AAAA,SACnB,YAAY,aAAa,SAAS;AAAA,IAErC;AAEJ,QAAM,aAAa,YAAY,aAC3B;AAAA;AAAA;AAAA,EAGJ,YAAY,UAAU;AAAA;AAAA,IAGlB;AAEJ,QAAM,gBAAgB,YAAY,gBAC9B;AAAA;AAAA,EAEJ,YAAY,aAAa;AAAA,IAErB;AAEJ,SAAO;AAAA;AAAA;AAAA;AAAA,EAIP,QAAQ,GAAG,UAAU;AAAA;AAAA,EAErB,YAAY,WAAW;AAAA,EACvB,aAAa;AAAA;AAAA,oBAEK,YAAY,MAAM;AAAA,UAC5B,YAAY,UAAU,SAAS;AAAA,SAChC,YAAY,gBAAgB,SAAS;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,oBA4C1B,YAAY,MAAM,KAAK,YAAY,UAAU,EAAE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,QAuC3D,YAAY,MAAM,CAAC;AAC3B;AAIA,SAAS,YAAY,QAA0E;AAC7F,MAAI,OAAO,OAAQ,QAAO;AAC1B,MAAI,OAAO,SAAU,QAAO;AAC5B,SAAO;AACT;AAEA,SAAS,YAAY;AAAA,EACnB;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA,aAAa,CAAC;AAAA,EACd,SAAS,CAAC;AACZ,GAQgB;AACd,SAAO;AAAA,IACL;AAAA,IACA;AAAA,IACA;AAAA,IACA,UAAU;AAAA,IACV,YAAY,KAAK,IAAI,IAAI;AAAA,IACzB;AAAA,IACA;AAAA,IACA;AAAA,EACF;AACF;AAQA,SAAS,YAAY,MAAsB;AACzC,QAAM,QAAQ;AAAA,IACZ,SAAS;AAAA,IACT,MAAM;AAAA,IACN,QAAQ;AAAA,IACR,OAAO;AAAA,IACP,OAAO;AAAA,EACT;AAEA,QAAM,QAAQ,KAAK,aAAa,CAAC;AACjC,QAAM,UAAU,KAAK,eAAe,CAAC;AAErC,MAAI,MAAM,WAAW,KAAK,QAAQ,WAAW,GAAG;AAE9C,YAAQ;AAAA,MACN,GAAG,MAAM,OAAO,sBAAsB,MAAM,KAAK,mCAAmC,KAAK,gBAAgB,SAAS;AAAA,IACpH;AAAA,EACF;AAEA,aAAW,QAAQ,OAAO;AACxB,UAAM,OAAO,KAAK,YAAY;AAC9B,UAAM,OAAO,KAAK,OAAO,SAAS,SAAS,KAAK,IAAI,GAAG,GAAG,IAAI;AAE9D,YAAQ;AAAA,MACN,GAAG,MAAM,OAAO,sBAAsB,MAAM,KAAK,IAAI,MAAM,IAAI,GAAG,IAAI,GAAG,MAAM,KAAK,GAAG,OAAO,SAAS,IAAI,KAAK,EAAE;AAAA,IACpH;AAAA,EACF;AAEA,aAAW,UAAU,SAAS;AAC5B,UAAM,OAAO,OAAO,YAAY;AAChC,UAAM,MAAM,OAAO,SAAS,SAAS,SAAS,OAAO,MAAM,GAAG,GAAG,IAAI;AACrE,UAAM,MAAM,OAAO,QAAQ,SAAS,SAAS,OAAO,KAAK,GAAG,GAAG,IAAI;AACnE,UAAM,cAAc,MAAM,MAAM,SAAS,MAAM;AAE/C,YAAQ;AAAA,MACN,GAAG,MAAM,OAAO,wBAAwB,MAAM,KAAK,IAAI,MAAM,IAAI,GAAG,IAAI,GAAG,MAAM,KAAK,MACnF,MAAM,IAAI,WAAW,KAAK,MAAM,KAAK,IAAI,GAAG,KAAK,OACjD,MAAM,IAAI,MAAM,MAAM,SAAS,GAAG,GAAG,MAAM,KAAK,KAAK;AAAA,IAC1D;AAAA,EACF;AACF;AAEA,SAAS,qBACP,aACA,SACA,SAC0C;AAC1C,QAAM,QAAkB,CAAC;AACzB,QAAM;AAAA,IACJ,oFAAoF,QAAQ,UAAU,SAAS,IAAI,QAAQ,UAAU,SAAS,aAAa,QAAQ,gBAAgB,SAAS;AAAA,EAC9L;AAEA,MAAI,YAAY,YAAY;AAC1B,UAAM;AAAA,MACJ;AAAA;AAAA,EAA6D,SAAS,YAAY,YAAY,IAAK,CAAC;AAAA;AAAA,IACtG;AAAA,EACF;AAEA,QAAM,KAAK,cAAc,YAAY,YAAY,SAAS,EAAE;AAC5D,QAAM,KAAK,eAAe,YAAY,aAAa,SAAS,EAAE;AAE9D,MAAI,QAAQ,UAAU;AACpB,UAAM,KAAK;AAAA;AAAA,EAA4D,SAAS,SAAS,QAAQ,QAAQ,GAAG,GAAI,CAAC;AAAA,OAAU;AAAA,EAC7H;AACA,MAAI,QAAQ,MAAM;AAChB,UAAM,KAAK;AAAA;AAAA,EAA2C,SAAS,QAAQ,MAAM,GAAI,CAAC;AAAA,OAAU;AAAA,EAC9F;AACA,MAAI,QAAQ,YAAY;AACtB,UAAM,KAAK,4CAA4C,SAAS,QAAQ,YAAY,IAAK,CAAC,EAAE;AAAA,EAC9F;AAEA,QAAM,KAAK,oHAAoH;AAE/H,SAAO,CAAC,EAAE,MAAM,QAAQ,SAAS,MAAM,KAAK,MAAM,EAAE,CAAC;AACvD;AAEA,eAAe,wBACb,MACA,QACqE;AACrE,QAAM,kBAAkB,KAAK,IAAI,KAAM,OAAO,mBAAmB,GAAK;AACtE,QAAM,SAAqE,CAAC;AAC5E,MAAI,CAAC,KAAM,QAAO;AAGlB,MAAI;AACF,QAAI,KAAK,eAAe,UAAU;AAChC,YAAM,OAAO,MAAM,KAAK,cAAc,SAAS;AAC/C,UAAI,MAAM;AACR,eAAO,WAAW,gBAAgB,MAAM,eAAe;AAAA,MACzD;AAAA,IACF;AAAA,EACF,QAAQ;AAAA,EAER;AAGA,MAAI,CAAC,OAAO,YAAY,KAAK,UAAU;AACrC,QAAI;AACF,YAAM,OAAQ,MAAM,KAAK,SAAS,yBAAyB;AAC3D,UAAI,MAAM;AACR,eAAO,OAAO,SAAS,eAAe,IAAI,GAAG,eAAe;AAAA,MAC9D;AAAA,IACF,QAAQ;AAAA,IAER;AAAA,EACF;AAGA,MAAI,OAAO,iBAAiB,KAAK,YAAY;AAC3C,QAAI;AACF,YAAM,SAAS,MAAM,KAAK,WAAW,EAAE,UAAU,MAAM,CAAC;AACxD,YAAM,IAAI,OAAO,WAAW,WAAW,OAAO,KAAK,MAAM,IAAI,OAAO,KAAK,MAAM;AAC/E,YAAM,SAAS,EAAE,SAAS,QAAQ;AAClC,UAAI,EAAE,cAAc,mBAAmB,OAAO,UAAU,iBAAiB;AACvE,eAAO,aAAa,yBAAyB,SAAS,QAAQ,eAAe,CAAC;AAAA,MAChF;AAAA,IACF,QAAQ;AAAA,IAER;AAAA,EACF;AAEA,SAAO;AACT;AAEA,SAAS,SAAS,OAAe,KAAqB;AACpD,MAAI,MAAM,UAAU,IAAK,QAAO;AAChC,SAAO,MAAM,MAAM,GAAG,GAAG,IAAI;AAC/B;AAEA,SAAS,SAAS,OAAwB;AACxC,MAAI;AACF,WAAO,KAAK,UAAU,KAAK;AAAA,EAC7B,QAAQ;AACN,WAAO,OAAO,KAAK;AAAA,EACrB;AACF;AAEO,SAAS,eAAe,OAAuB;AACpD,MAAI,SAAS;AACb,aAAW,EAAE,OAAO,YAAY,KAAK,oBAAoB;AACvD,aAAS,OAAO,QAAQ,OAAO,WAAW;AAAA,EAC5C;AACA,SAAO;AACT;AAEO,SAAS,gBAAgB,OAAgB,iBAAyB,OAAO,oBAAI,QAAQ,GAAY;AACtG,MAAI,UAAU,QAAQ,UAAU,OAAW,QAAO;AAClD,MAAI,OAAO,UAAU,UAAU;AAC7B,WAAO,SAAS,eAAe,KAAK,GAAG,eAAe;AAAA,EACxD;AACA,MAAI,OAAO,UAAU,YAAY,OAAO,UAAU,UAAW,QAAO;AACpE,MAAI,OAAO,UAAU,UAAU;AAC7B,QAAI,KAAK,IAAI,KAAe,EAAG,QAAO;AACtC,SAAK,IAAI,KAAe;AACxB,QAAI,MAAM,QAAQ,KAAK,GAAG;AACxB,aAAO,MAAM,MAAM,GAAG,EAAE,EAAE,IAAI,CAAC,MAAM,gBAAgB,GAAG,iBAAiB,IAAI,CAAC;AAAA,IAChF;AACA,UAAM,MAA+B,CAAC;AACtC,UAAM,UAAU,OAAO,QAAQ,KAAgC,EAAE,MAAM,GAAG,EAAE;AAC5E,eAAW,CAAC,GAAG,CAAC,KAAK,SAAS;AAC5B,UAAI,CAAC,IAAI,gBAAgB,GAAG,iBAAiB,IAAI;AAAA,IACnD;AACA,WAAO;AAAA,EACT;AACA,SAAO;AACT;AAEO,SAAS,qBAAqB,MAAuB;AAC1D,SAAO,mBAAmB,KAAK,CAAC,MAAM,EAAE,KAAK,IAAI,CAAC;AACpD;;;AElmCA,OAAOC,SAAQ;AACf,OAAOC,WAAU;AAkCV,SAAS,cAA8B;AAC5C,MAAI,CAAC,WAAW,iBAAiB;AAC/B,eAAW,kBAAkB,CAAC;AAAA,EAChC;AACA,SAAO,WAAW;AACpB;AAEO,SAAS,gBAAgBC,OAAoB;AAClD,aAAW,0BAA0BA;AACvC;AAEO,SAAS,kBAAsC;AACpD,SAAO,WAAW;AACpB;AAEO,SAAS,mBAAmB,OAA2B;AAC5D,gBAAc;AACd,QAAM,MAAM,YAAY;AACxB,QAAM,QAAsB;AAAA,IAC1B,GAAG;AAAA,IACH,UAAU,QAAQ,IAAI;AAAA,IACtB,YAAW,oBAAI,KAAK,GAAE,YAAY;AAAA,EACpC;AACA,MAAI,KAAK,KAAK;AACd,cAAY,KAAK;AACnB;AAEO,SAAS,cAAoB;AAClC,aAAW,mBAAmB;AAChC;AAEO,SAAS,iBAA0B;AACxC,SAAO,WAAW,qBAAqB;AACzC;AAEA,SAAS,YAAY,OAA2B;AAC9C,QAAM,UAAU,gBAAgB;AAChC,MAAI,CAAC,QAAS;AACd,QAAM,SAAS,iBAAiB;AAChC,MAAI,CAAC,OAAO,oBAAqB;AAEjC,MAAI;AACF,IAAAC,IAAG,UAAUD,MAAK,QAAQ,OAAO,GAAG,EAAE,WAAW,KAAK,CAAC;AACvD,IAAAC,IAAG,eAAe,SAAS,KAAK,UAAU,KAAK,IAAI,IAAI;AAAA,EACzD,QAAQ;AAAA,EAER;AACF;;;AL3EA,IAAI,QAAQ;AAGZ,IAAM,YAAiB,OAAO,cAAY,cAAc,YAAU,cAAc,YAAY,GAAG;AAE/F,IAAM,kBAAkB,oBAAI,IAAI;AAAA,EAC9B;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,CAAC;AAED,IAAM,eAAe,oBAAI,IAAI;AAAA,EAC3B;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,CAAC;AAED,IAAM,oBAAoB,oBAAI,IAAI;AAAA,EAChC;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,CAAC;AAED,IAAM,wBAAwB,oBAAI,IAAI;AAAA,EACpC;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,CAAC;AAEM,SAAS,yBAA+B;AAC7C,MAAI,eAAe,EAAG;AAEtB,QAAM,SAAS,iBAAiB;AAChC,UAAQ,CAAC,CAAC,OAAO;AACjB,MAAI,CAAC,OAAO,SAAS;AACnB,gBAAY;AACZ;AAAA,EACF;AAEA,kBAAgB,OAAO,YAAY;AACnC,MAAI,CAAC,wBAAwB,OAAO,wBAAwB,GAAG;AAC7D,uBAAmB;AAAA,MACjB,MAAM;AAAA,MACN,QAAQ;AAAA,MACR,cAAc;AAAA,MACd,QAAQ;AAAA,IACV,CAAC;AACD,gBAAY;AACZ;AAAA,EACF;AACA,2BAAyB;AAEzB,MAAI,OAAO;AAET,YAAQ,IAAI,uDAAuD,OAAO,YAAY,GAAG;AAAA,EAC3F;AAEA,qBAAmB;AAAA,IACjB,MAAM;AAAA,IACN,QAAQ;AAAA,IACR,QAAQ;AAAA,EACV,CAAC;AACD,cAAY;AACd;AAEA,SAAS,2BAAiC;AAGxC,QAAM,MAAM;AACZ,QAAM,eAAiC,IAAI,uBAAuB,IAAI;AACtE,MAAI,CAAC,IAAI,qBAAqB;AAC5B,QAAI,sBAAsB;AAAA,EAC5B;AAGA,MAAI,QAAQ,SAAS,YAAY,SAAiB,QAAa,QAAiB;AAC9E,QAAI,YAAY,sBAAsB,YAAY,mBAAmB;AACnE,UAAI,OAAO;AAET,gBAAQ,IAAI,uDAAuD,OAAO,GAAG;AAAA,MAC/E;AACA,YAAM,OAAO,aAAa,KAAK,MAAM,SAAS,QAAQ,MAAM;AAC5D,aAAO,wBAAwB,IAAI;AAAA,IACrC;AACA,WAAO,aAAa,KAAK,MAAM,SAAS,QAAQ,MAAM;AAAA,EACxD;AACF;AAQA,SAAS,wBAAwB,MAA4C;AAC3E,MAAK,KAAuC,gBAAiB,QAAO;AAEpE,QAAM,WAAW,KAAK,KAAK,OAAO;AAAA,IAChC,MAAM,OAAO,EAAE,KAAK,GAAG,QAAQ;AAC7B,yBAAmB;AAAA,QACjB,MAAM;AAAA,QACN,QAAQ;AAAA,QACR,QAAQ;AAAA,MACV,CAAC;AACD,YAAM,cAAc,SAAS,IAAI;AACjC,YAAM,IAAI,WAAW;AAAA,IACvB;AAAA,EACF,CAAC;AAED,QAAM,UAA6B;AAAA,IACjC,GAAG;AAAA,IACH,MAAM;AAAA,IACN,QAAQ,WAAW,KAAK,MAAM;AAAA,EAChC;AAEA,EAAC,QAA0C,kBAAkB;AAC7D,MAAI,OAAO;AAET,YAAQ,IAAI,oCAAoC;AAAA,EAClD;AACA,SAAO;AACT;AAEA,SAAS,wBAAwB,UAA4B;AAC3D,MAAI,CAAC,SAAU,QAAO;AACtB,MAAI;AACF,UAAM,MAAM,UAAU,yBAAyB;AAC/C,QAAI,IAAI,WAAW,IAAI,QAAQ,WAAW,QAAQ,GAAG;AACnD,aAAO;AAAA,IACT;AAEA,YAAQ,KAAK,kDAAkD,QAAQ,WAAW,IAAI,WAAW,SAAS,EAAE;AAC5G,WAAO;AAAA,EACT,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAEA,SAAS,WAAW,YAAsE;AACxF,QAAM,eAAe,CAAC,aAAsB,SAA0B;AACpE,WAAO,IAAI,MAAM,eAAe,CAAC,GAAG;AAAA,MAClC,IAAI,QAAQ,MAAM,UAAU;AAC1B,cAAM,QAAQ,QAAQ,IAAI,QAAQ,MAAM,QAAQ;AAChD,YAAI,OAAO,SAAS,YAAY,OAAO,UAAU,YAAY;AAC3D,iBAAO,IAAI,SAAoB,sBAAsB,EAAE,SAAS,OAA0C,aAAa,MAAM,MAAM,cAAc,QAAQ,KAAK,CAAC;AAAA,QACjK;AACA,eAAO;AAAA,MACT;AAAA,IACF,CAAC;AAAA,EACH;AAEA,QAAM,QAAQ,IAAI,MAAM,YAAY;AAAA,IAClC,MAAM,QAAQ,SAAS,UAAU;AAC/B,YAAM,cAAe,OAA2C,MAAM,SAAS,QAAqB;AACpG,aAAO,aAAa,aAAa,MAAM;AAAA,IACzC;AAAA,IACA,IAAI,QAAQ,MAAM,UAAU;AAC1B,YAAM,QAAQ,QAAQ,IAAI,QAAQ,MAAM,QAAQ;AAChD,UAAI,SAAS,UAAU,OAAO,UAAU,YAAY;AAClD,eAAO,IAAI,SAAoB,aAAc,MAA0C,MAAM,QAAQ,IAAI,GAAG,MAAM;AAAA,MACpH;AACA,aAAO;AAAA,IACT;AAAA,EACF,CAAC;AAED,SAAO;AACT;AAEA,SAAS,SAAkC,MAA0B;AACnE,SAAO,IAAI,MAAM,MAAM;AAAA,IACrB,IAAI,QAAQ,MAAM,UAAU;AAC1B,YAAM,QAAQ,QAAQ,IAAI,QAAQ,MAAM,QAAQ;AAEhD,UAAI,OAAO,SAAS,UAAU;AAC5B,YAAI,kBAAkB,IAAI,IAAI,KAAK,OAAO,UAAU,YAAY;AAC9D,iBAAO,IAAI,SAAoB;AAC7B,kBAAM,UAAW,MAA0C,MAAM,QAAQ,IAAI;AAC7E,mBAAO,YAAY,OAAO;AAAA,UAC5B;AAAA,QACF;AAEA,YAAI,aAAa,IAAI,IAAI,KAAK,OAAO,UAAU,YAAY;AACzD,iBAAO,UAAU,SAAoB;AACnC,mBAAO,mBAAmB;AAAA,cACxB,MAAM;AAAA,cACN,QAAQ;AAAA,cACR,QAAQ,iBAAiB,MAAM;AAAA,cAC/B,SAAS;AAAA,cACT,MAAM;AAAA,cACN,QAAQ,MAAO,MAA0C,MAAM,QAAQ,IAAI;AAAA,YAC7E,CAAC;AAAA,UACH;AAAA,QACF;AAAA,MACF;AAEA,aAAO;AAAA,IACT;AAAA,EACF,CAAC;AACH;AAEA,SAAS,YAAY,SAA2B;AAC9C,MAAI,CAAC,WAAW,OAAO,YAAY,SAAU,QAAO;AAEpD,SAAO,IAAI,MAAM,SAAS;AAAA,IACxB,IAAI,QAAQ,MAAM,UAAU;AAC1B,YAAM,QAAQ,QAAQ,IAAI,QAAQ,MAAM,QAAQ;AAEhD,UAAI,OAAO,SAAS,YAAY,gBAAgB,IAAI,IAAI,KAAK,OAAO,UAAU,YAAY;AACxF,eAAO,UAAU,SAAoB;AACnC,iBAAO,mBAAmB;AAAA,YACxB,MAAM;AAAA,YACN,QAAQ;AAAA,YACR,QAAQ,iBAAiB,MAAM;AAAA,YAC/B,SAAS;AAAA,YACT,MAAM;AAAA,YACN,QAAQ,MAAO,MAA0C,MAAM,QAAQ,IAAI;AAAA,UAC7E,CAAC;AAAA,QACH;AAAA,MACF;AAEA,UAAI,OAAO,SAAS,YAAY,sBAAsB,IAAI,IAAI,KAAK,OAAO,UAAU,YAAY;AAC9F,eAAO,IAAI,SAAoB,YAAa,MAA0C,MAAM,QAAQ,IAAI,CAAC;AAAA,MAC3G;AAEA,aAAO;AAAA,IACT;AAAA,EACF,CAAC;AACH;AAWA,eAAe,mBAAsB,KAAiC;AACpE,MAAI;AACF,WAAO,MAAM,QAAQ,QAAQ,IAAI,OAAO,CAAC;AAAA,EAC3C,SAAS,OAAO;AACd,UAAM,UAAU,oBAAoB,IAAI,MAAM,IAAI,QAAQ,IAAI,QAAQ,KAAK;AAC3E,uBAAmB,EAAE,GAAG,SAAS,QAAQ,MAAM,CAAC;AAChD,UAAM,WAAW,gBAAgB,OAAO;AACxC,QAAI,OAAO;AAET,cAAQ,IAAI,6CAA6C,IAAI,IAAI,WAAW,IAAI,MAAM,WAAW,SAAS,WAAW,SAAS,UAAU,aAAa,SAAS,MAAM,EAAE;AAAA,IACxK;AACA,QAAI,CAAC,SAAS,UAAU;AACtB,YAAM;AAAA,IACR;AAEA,UAAM,UAAU,MAAM,mBAAmB,UAAU,SAAS;AAAA,MAC1D,MAAM,IAAI;AAAA,MACV,QAAQ,IAAI;AAAA,MACZ,QAAQ,IAAI;AAAA,MACZ,SAAS,cAAc,IAAI,OAAO,IAAI,IAAI,UAAU;AAAA,MACpD,MAAM,WAAW,IAAI,IAAI,IAAI,IAAI,OAAO;AAAA,MACxC,aAAa;AAAA,QACX,UAAU,aAAa;AAAA,QACvB,WAAW,aAAa;AAAA,QACxB,YAAY,eAAe,aAAa,QAAQ;AAAA,MAClD;AAAA,IACF,CAAC;AAGD,UAAM,cAAc,eAAe;AAEnC,UAAM,cAAc;AAAA,MAClB,GAAG;AAAA,MACH,QAAQ,QAAQ;AAAA,MAChB,UAAU;AAAA,MACV,QAAQ,QAAQ,UAAU,SAAS;AAAA,MACnC,SAAS,oBAAoB,QAAQ,UAAU;AAAA,MAC/C,YAAY,QAAQ;AAAA,MACpB,MAAM,QAAQ;AAAA,MACd,UAAU,SAAS;AAAA,MACnB,SAAS,QAAQ;AAAA,MACjB,SAAS,QAAQ;AAAA,MACjB,UAAU,aAAa;AAAA,MACvB,WAAW,aAAa;AAAA,IAC1B;AAEA,QAAI,QAAQ,UAAU,CAAC,QAAQ,qBAAqB;AAClD,yBAAmB,EAAE,GAAG,aAAa,QAAQ,KAAK,CAAC;AACnD,aAAO;AAAA,IACT;AAEA,QAAI,QAAQ,qBAAqB;AAC/B,UAAI;AACF,cAAM,UAAU,MAAM,QAAQ,QAAQ,IAAI,OAAO,CAAC;AAClD,YAAI,OAAO;AAET,kBAAQ,IAAI,uDAAuD,IAAI,MAAM,EAAE;AAAA,QACjF;AACA,2BAAmB,EAAE,GAAG,aAAa,QAAQ,KAAK,CAAC;AACnD,eAAO;AAAA,MACT,SAAS,YAAY;AACnB,cAAM,YAAY,UAAU,UAAU;AACtC,2BAAmB,EAAE,GAAG,aAAa,QAAQ,OAAO,cAAc,UAAU,QAAQ,CAAC;AACrF,YAAI,OAAO;AAET,kBAAQ,IAAI,oDAAoD,IAAI,MAAM,KAAK,UAAU,WAAW,UAAU,EAAE;AAAA,QAClH;AACA,cAAM;AAAA,MACR;AAAA,IACF;AAEA,uBAAmB,WAAW;AAC9B,UAAM;AAAA,EACR;AACF;AAEA,eAAe,sBAAsB,QAMhB;AACnB,QAAM,EAAE,SAAS,aAAa,MAAM,cAAc,KAAK,IAAI;AAE3D,QAAM,SAAS,MAAM,QAAQ,MAAM,cAAc,IAAI;AAGrD,MAAI,SAAS,QAAQ;AACnB,WAAO,QAAQ,QAAQ,OAAO,CAAC;AAAA,EACjC;AAEA,MAAI;AACF,WAAO,MAAM,QAAQ,QAAQ,OAAO,CAAC;AAAA,EACvC,SAAS,OAAO;AACd,UAAM,SAAS,gBAAgB,OAAO,CAAC,CAAC;AACxC,UAAM,UAAU,oBAAoB,UAAU,aAAa,QAAQ,KAAK;AACxE,uBAAmB,EAAE,GAAG,SAAS,QAAQ,MAAM,CAAC;AAChD,UAAM,WAAW,gBAAgB,OAAO;AACxC,QAAI,CAAC,SAAS,UAAU;AACtB,YAAM;AAAA,IACR;AAEA,UAAM,UAAU,MAAM,mBAAmB,UAAU,SAAS;AAAA,MAC1D,MAAM;AAAA,MACN,QAAQ;AAAA,MACR;AAAA,MACA,SAAS,cAAc,OAAO,CAAC,CAAC,IAAI,KAAK,CAAC,IAAI;AAAA,IAChD,CAAC;AAGD,UAAM,cAAc,eAAe;AAEnC,UAAM,cAAc;AAAA,MAClB,GAAG;AAAA,MACH,QAAQ;AAAA,MACR,UAAU;AAAA,MACV,QAAQ,QAAQ,UAAU,SAAS;AAAA,MACnC,SAAS,oBAAoB,QAAQ,UAAU;AAAA,MAC/C,YAAY,QAAQ;AAAA,MACpB,MAAM,QAAQ;AAAA,MACd,UAAU,SAAS;AAAA,MACnB,SAAS,QAAQ;AAAA,MACjB,SAAS,QAAQ;AAAA,MACjB,UAAU,aAAa;AAAA,MACvB,WAAW,aAAa;AAAA,IAC1B;AAEA,QAAI,QAAQ,qBAAqB;AAC/B,UAAI;AACF,cAAM,UAAU,MAAM,QAAQ,QAAQ,OAAO,CAAC;AAC9C,2BAAmB,EAAE,GAAG,aAAa,QAAQ,KAAK,CAAC;AACnD,eAAO;AAAA,MACT,SAAS,YAAY;AACnB,cAAM,YAAY,UAAU,UAAU;AACtC,2BAAmB,EAAE,GAAG,aAAa,QAAQ,OAAO,cAAc,UAAU,QAAQ,CAAC;AACrF,cAAM;AAAA,MACR;AAAA,IACF;AAEA,uBAAmB,WAAW;AAC9B,UAAM;AAAA,EACR;AACF;AAEA,SAAS,oBACP,MACA,QACA,QACA,OACgB;AAChB,QAAM,OAAO,UAAU,KAAK;AAC5B,SAAO;AAAA,IACL;AAAA,IACA;AAAA,IACA;AAAA,IACA,cAAc,KAAK;AAAA,IACnB,WAAW,KAAK;AAAA,IAChB,OAAO,KAAK;AAAA,EACd;AACF;AAEA,SAAS,oBAAoB,SAA6D;AACxF,SAAO,QAAQ,IAAI,CAAC,YAAY,EAAE,MAAM,OAAO,EAAE;AACnD;AAEA,SAAS,UAAU,OAAqE;AACtF,MAAI,iBAAiB,OAAO;AAC1B,WAAO,EAAE,SAAS,MAAM,SAAS,MAAM,MAAM,MAAM,OAAO,MAAM,MAAM;AAAA,EACxE;AACA,SAAO,EAAE,SAAS,OAAO,UAAU,WAAW,QAAQ,KAAK,UAAU,KAAK,EAAE;AAC9E;AAEA,SAAS,cAAc,WAAqF;AAC1G,SAAO,QAAQ,aAAa,OAAO,cAAc,YAAY,4BAA6B,SAAqC;AACjI;AAEA,SAAS,WAAW,WAAuF;AACzG,SAAO,QAAQ,aAAa,OAAO,cAAc,YAAY,oBAAqB,SAAqC;AACzH;AAEA,SAAS,iBAAiB,QAAqC;AAC7D,MAAI;AACF,QAAI,OAAO,WAAW,YAAY,WAAW,QAAQ,cAAc,QAAQ;AACzE,YAAM,IAAI,OAAQ,OAAsC,SAAS,CAAC;AAClE,UAAI,KAAK,MAAM,kBAAmB,QAAO;AAAA,IAC3C;AAAA,EACF,QAAQ;AAAA,EAER;AACA,SAAO;AACT;AAEA,SAAS,eAAe,UAAuC;AAC7D,MAAI,CAAC,SAAU,QAAO;AACtB,MAAI;AACF,WAAO,UAAU,IAAI,EAAE,aAAa,UAAU,OAAO;AAAA,EACvD,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAEA,SAAS,gBAAgB,WAAwC;AAC/D,MAAI,CAAC,UAAW,QAAO;AACvB,MAAI,OAAO,cAAc,SAAU,QAAO;AAC1C,MAAI,OAAO,cAAc,UAAU;AACjC,QAAI,cAAe,aAAyC,OAAQ,UAAsC,aAAa,UAAU;AAC/H,aAAO,OAAQ,UAAsC,QAAQ;AAAA,IAC/D;AAAA,EACF;AACA,SAAO,iBAAiB,SAAS;AACnC;AAMA,SAAS,iBAAwE;AAC/E,MAAI;AAEF,UAAM,aAAa,UAAU,kBAAkB;AAC/C,QAAI,YAAY,MAAM,MAAM;AAC1B,YAAM,OAAO,WAAW,KAAK,KAAK;AAClC,UAAI,MAAM;AACR,eAAO;AAAA,UACL,UAAU,KAAK;AAAA,UACf,WAAW,KAAK;AAAA,QAClB;AAAA,MACF;AAAA,IACF;AAAA,EACF,QAAQ;AAAA,EAER;AACA,SAAO;AACT;;;AM1fA,YAAY;AACZ,uBAAuB;","names":["path","path","fs","path","path","fs"]}