lobster-cli 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +389 -0
- package/dist/agent/core.js +1013 -0
- package/dist/agent/core.js.map +1 -0
- package/dist/agent/index.js +1027 -0
- package/dist/agent/index.js.map +1 -0
- package/dist/brain/index.js +60 -0
- package/dist/brain/index.js.map +1 -0
- package/dist/browser/dom/index.js +1096 -0
- package/dist/browser/dom/index.js.map +1 -0
- package/dist/browser/index.js +2034 -0
- package/dist/browser/index.js.map +1 -0
- package/dist/browser/manager.js +86 -0
- package/dist/browser/manager.js.map +1 -0
- package/dist/browser/page-adapter.js +1345 -0
- package/dist/browser/page-adapter.js.map +1 -0
- package/dist/cascade/index.js +138 -0
- package/dist/cascade/index.js.map +1 -0
- package/dist/config/index.js +110 -0
- package/dist/config/index.js.map +1 -0
- package/dist/config/schema.js +66 -0
- package/dist/config/schema.js.map +1 -0
- package/dist/discover/index.js +545 -0
- package/dist/discover/index.js.map +1 -0
- package/dist/index.js +5529 -0
- package/dist/index.js.map +1 -0
- package/dist/lib.js +4206 -0
- package/dist/lib.js.map +1 -0
- package/dist/llm/client.js +379 -0
- package/dist/llm/client.js.map +1 -0
- package/dist/llm/index.js +397 -0
- package/dist/llm/index.js.map +1 -0
- package/dist/llm/openai-client.js +214 -0
- package/dist/llm/openai-client.js.map +1 -0
- package/dist/output/index.js +93 -0
- package/dist/output/index.js.map +1 -0
- package/dist/pipeline/index.js +802 -0
- package/dist/pipeline/index.js.map +1 -0
- package/dist/router/decision.js +80 -0
- package/dist/router/decision.js.map +1 -0
- package/dist/router/index.js +3443 -0
- package/dist/router/index.js.map +1 -0
- package/dist/types/index.js +23 -0
- package/dist/types/index.js.map +1 -0
- package/logo.svg +11 -0
- package/package.json +65 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/browser/manager.ts","../../src/utils/logger.ts","../../src/browser/dom/flat-tree.ts","../../src/browser/dom/snapshot.ts","../../src/browser/dom/semantic-tree.ts","../../src/browser/dom/markdown.ts","../../src/browser/dom/form-state.ts","../../src/browser/interceptor.ts","../../src/browser/page-adapter.ts","../../src/browser/dom/interactive.ts","../../src/browser/wait.ts","../../src/browser/lightpanda.ts"],"sourcesContent":["import puppeteer, { type Browser, type Page } from 'puppeteer-core';\nimport { existsSync } from 'node:fs';\nimport { log } from '../utils/logger.js';\n\nexport interface BrowserManagerConfig {\n executablePath?: string;\n headless?: boolean;\n cdpEndpoint?: string;\n connectTimeout?: number;\n}\n\nexport class BrowserManager {\n private browser: Browser | null = null;\n private config: BrowserManagerConfig;\n\n constructor(config: BrowserManagerConfig = {}) {\n this.config = config;\n }\n\n async connect(): Promise<Browser> {\n if (this.browser?.connected) return this.browser;\n\n if (this.config.cdpEndpoint) {\n log.debug(`Connecting to CDP endpoint: ${this.config.cdpEndpoint}`);\n this.browser = await puppeteer.connect({\n browserWSEndpoint: this.config.cdpEndpoint,\n });\n return this.browser;\n }\n\n const executablePath = this.config.executablePath || findChrome();\n if (!executablePath) {\n throw new Error(\n 'Chrome/Chromium not found. Set LOBSTER_BROWSER_PATH or config browser.executablePath'\n );\n }\n\n log.debug(`Launching Chrome: ${executablePath}`);\n this.browser = await puppeteer.launch({\n executablePath,\n headless: this.config.headless ?? true,\n args: [\n '--no-sandbox',\n '--disable-setuid-sandbox',\n '--disable-dev-shm-usage',\n '--disable-gpu',\n ],\n });\n\n return this.browser;\n }\n\n async newPage(): Promise<Page> {\n const browser = await this.connect();\n return browser.newPage();\n }\n\n async close(): Promise<void> {\n if (this.browser) {\n await this.browser.close().catch(() => {});\n this.browser = null;\n }\n }\n}\n\nfunction findChrome(): string | undefined {\n const paths =\n process.platform === 'darwin'\n ? [\n '/Applications/Google Chrome.app/Contents/MacOS/Google Chrome',\n '/Applications/Chromium.app/Contents/MacOS/Chromium',\n '/Applications/Google Chrome Canary.app/Contents/MacOS/Google Chrome Canary',\n ]\n : process.platform === 'win32'\n ? [\n 'C:\\\\Program Files\\\\Google\\\\Chrome\\\\Application\\\\chrome.exe',\n 'C:\\\\Program Files (x86)\\\\Google\\\\Chrome\\\\Application\\\\chrome.exe',\n ]\n : [\n '/usr/bin/google-chrome',\n '/usr/bin/google-chrome-stable',\n '/usr/bin/chromium-browser',\n '/usr/bin/chromium',\n '/snap/bin/chromium',\n ];\n\n return paths.find((p) => existsSync(p));\n}\n","import chalk from 'chalk';\n\nexport const log = {\n info: (msg: string) => console.log(chalk.blue('ℹ'), msg),\n success: (msg: string) => console.log(chalk.green('✓'), msg),\n warn: (msg: string) => console.log(chalk.yellow('⚠'), msg),\n error: (msg: string) => console.error(chalk.red('✗'), msg),\n debug: (msg: string) => {\n if (process.env.LOBSTER_DEBUG) console.log(chalk.gray('⋯'), msg);\n },\n step: (n: number, msg: string) => console.log(chalk.cyan(`[${n}]`), msg),\n dim: (msg: string) => console.log(chalk.dim(msg)),\n};\n","/**\n * Script that runs inside the browser to extract a flat DOM tree\n * with indexed interactive elements — the format the AI agent uses.\n *\n * Based on Page Agent's DOM extraction approach.\n */\nexport const FLAT_TREE_SCRIPT = `\n(() => {\n const INTERACTIVE_TAGS = new Set([\n 'a', 'button', 'input', 'select', 'textarea', 'details', 'summary',\n 'label', 'option', 'fieldset', 'legend',\n ]);\n\n const INTERACTIVE_ROLES = new Set([\n 'button', 'link', 'textbox', 'checkbox', 'radio', 'combobox',\n 'listbox', 'menu', 'menuitem', 'tab', 'switch', 'slider',\n 'searchbox', 'spinbutton', 'option', 'menuitemcheckbox', 'menuitemradio',\n ]);\n\n const ATTR_WHITELIST = [\n 'type', 'role', 'aria-label', 'aria-expanded', 'aria-selected',\n 'aria-checked', 'aria-disabled', 'placeholder', 'title', 'href',\n 'value', 'name', 'alt', 'src',\n ];\n\n let highlightIndex = 0;\n const nodes = {};\n const selectorMap = {};\n\n function isVisible(el) {\n if (el.offsetWidth === 0 && el.offsetHeight === 0) return false;\n const style = getComputedStyle(el);\n if (style.display === 'none' || style.visibility === 'hidden' || style.opacity === '0') return false;\n return true;\n }\n\n function isInteractive(el) {\n const tag = el.tagName.toLowerCase();\n if (INTERACTIVE_TAGS.has(tag)) return true;\n const role = el.getAttribute('role');\n if (role && INTERACTIVE_ROLES.has(role)) return true;\n if (el.getAttribute('contenteditable') === 'true') return true;\n if (el.getAttribute('tabindex') !== null && parseInt(el.getAttribute('tabindex')) >= 0) return true;\n if (el.onclick || el.getAttribute('onclick')) return true;\n return false;\n }\n\n function getAttributes(el) {\n const attrs = {};\n for (const attr of ATTR_WHITELIST) {\n const val = el.getAttribute(attr);\n if (val !== null && val !== '') attrs[attr] = val;\n }\n return attrs;\n }\n\n function getScrollable(el) {\n const style = getComputedStyle(el);\n const overflowY = style.overflowY;\n const overflowX = style.overflowX;\n const isScrollableY = (overflowY === 'auto' || overflowY === 'scroll') && el.scrollHeight > el.clientHeight;\n const isScrollableX = (overflowX === 'auto' || overflowX === 'scroll') && el.scrollWidth > el.clientWidth;\n if (!isScrollableY && !isScrollableX) return null;\n return {\n left: el.scrollLeft,\n top: el.scrollTop,\n right: el.scrollWidth - el.clientWidth - el.scrollLeft,\n bottom: el.scrollHeight - el.clientHeight - el.scrollTop,\n };\n }\n\n function walk(el, parentId) {\n if (!el || el.nodeType === 8) return; // skip comments\n\n if (el.nodeType === 3) { // text node\n const text = el.textContent.trim();\n if (!text) return;\n const id = 'text_' + Math.random().toString(36).slice(2, 8);\n nodes[id] = { id, tagName: '#text', text, parentId };\n if (parentId && nodes[parentId]) {\n nodes[parentId].children = nodes[parentId].children || [];\n nodes[parentId].children.push(id);\n }\n return;\n }\n\n if (el.nodeType !== 1) return; // only elements\n\n const tag = el.tagName.toLowerCase();\n if (['script', 'style', 'noscript', 'svg', 'path'].includes(tag)) return;\n if (!isVisible(el)) return;\n\n const id = tag + '_' + Math.random().toString(36).slice(2, 8);\n const interactive = isInteractive(el);\n const node = {\n id,\n tagName: tag,\n attributes: getAttributes(el),\n parentId,\n children: [],\n isInteractive: interactive,\n };\n\n if (interactive) {\n node.highlightIndex = highlightIndex;\n selectorMap[highlightIndex] = id;\n highlightIndex++;\n }\n\n const scrollable = getScrollable(el);\n if (scrollable) node.scrollable = scrollable;\n\n const text = [];\n for (const child of el.childNodes) {\n if (child.nodeType === 3 && child.textContent.trim()) {\n text.push(child.textContent.trim());\n }\n }\n if (text.length > 0) node.text = text.join(' ').slice(0, 200);\n\n nodes[id] = node;\n\n if (parentId && nodes[parentId]) {\n nodes[parentId].children.push(id);\n }\n\n for (const child of el.children) {\n walk(child, id);\n }\n }\n\n const rootId = 'root';\n nodes[rootId] = { id: rootId, tagName: 'body', children: [], attributes: {} };\n for (const child of document.body.children) {\n walk(child, rootId);\n }\n\n return { rootId, map: nodes, selectorMap };\n})()\n`;\n\n/**\n * Convert a FlatDomTree into the indexed text format that the LLM agent reads.\n * Example output:\n * [0]<button type=submit>Search</>\n * [1]<input type=text placeholder=\"Enter query\" />\n */\nexport function flatTreeToString(tree: { rootId: string; map: Record<string, any> }): string {\n const lines: string[] = [];\n\n function walk(nodeId: string, depth: number) {\n const node = tree.map[nodeId];\n if (!node) return;\n\n const indent = '\\t'.repeat(depth);\n\n if (node.tagName === '#text') {\n if (node.text) lines.push(`${indent}${node.text}`);\n return;\n }\n\n const attrs = node.attributes || {};\n const attrStr = Object.entries(attrs)\n .map(([k, v]) => (v === '' ? k : `${k}=\"${v}\"`))\n .join(' ');\n\n const prefix = node.highlightIndex !== undefined ? `[${node.highlightIndex}]` : '';\n const scrollInfo = node.scrollable\n ? ` |scroll: ${Math.round(node.scrollable.top)}px up, ${Math.round(node.scrollable.bottom)}px down|`\n : '';\n\n const text = node.text || '';\n const tag = node.tagName;\n\n if (prefix || text || node.children?.length > 0) {\n const opening = `${indent}${prefix}<${tag}${attrStr ? ' ' + attrStr : ''}${scrollInfo}>`;\n\n if (!node.children?.length || (node.children.length === 0 && text)) {\n lines.push(`${opening}${text}</>`);\n } else {\n lines.push(`${opening}${text}`);\n for (const childId of node.children || []) {\n walk(childId, depth + 1);\n }\n }\n } else {\n for (const childId of node.children || []) {\n walk(childId, depth);\n }\n }\n }\n\n walk(tree.rootId, 0);\n return lines.join('\\n');\n}\n","/**\n * Advanced DOM snapshot script — runs inside the browser.\n * Multi-stage pruning pipeline producing LLM-optimized output.\n *\n * Stages:\n * 1. Walk DOM, collect visibility + layout + interactivity signals\n * 2. Prune invisible, zero-area, non-content elements\n * 3. SVG & decoration collapse\n * 4. Shadow DOM traversal\n * 5. Same-origin iframe extraction\n * 6. Bounding-box parent-child dedup (link/button wrapping)\n * 7. Paint-order occlusion detection (overlay/modal coverage)\n * 8. Attribute whitelist filtering\n * 9. Ad/tracker filtering\n * 10. Scroll position info\n * 11. data-ref annotation for targeting\n * 12. Token-efficient serialization with interactive indices\n */\n/**\n * Build snapshot script with optional previous hashes for diff marking.\n * Elements new since last snapshot get a `*` prefix on their index.\n */\nexport function buildSnapshotScript(previousHashes?: string[]): string {\n return SNAPSHOT_SCRIPT_FN(previousHashes || []);\n}\n\nfunction SNAPSHOT_SCRIPT_FN(prevHashes: string[]): string {\n return `\n(() => {\n let idx = 0;\n const __prevHashes = new Set(${JSON.stringify(prevHashes)});\n const __currentHashes = [];\n`;\n}\n\nexport const SNAPSHOT_SCRIPT = `\n(() => {\n let idx = 0;\n const __prevHashes = (window.__lobster_prev_hashes) ? new Set(window.__lobster_prev_hashes) : null;\n const __currentHashes = [];\n\n const SKIP_TAGS = new Set([\n 'script','style','noscript','svg','path','meta','link','head',\n 'template','slot','colgroup','col',\n ]);\n\n const INTERACTIVE_TAGS = new Set([\n 'a','button','input','select','textarea','details','summary','label',\n ]);\n\n const INTERACTIVE_ROLES = new Set([\n 'button','link','textbox','checkbox','radio','combobox','listbox',\n 'menu','menuitem','tab','switch','slider','searchbox','spinbutton',\n 'option','menuitemcheckbox','menuitemradio','treeitem',\n ]);\n\n const ATTR_WHITELIST = [\n 'type','role','aria-label','aria-expanded','aria-selected','aria-checked',\n 'aria-disabled','aria-haspopup','aria-pressed','placeholder','title',\n 'href','value','name','alt','src','action','method','for',\n 'data-testid','data-id','contenteditable','tabindex',\n ];\n\n const AD_PATTERNS = /ad[-_]?banner|ad[-_]?container|google[-_]?ad|doubleclick|adsbygoogle|sponsored|^ad$/i;\n\n // ── Stage 1: Visibility check ──\n function isVisible(el) {\n if (el.offsetWidth === 0 && el.offsetHeight === 0 && el.tagName !== 'INPUT') return false;\n const s = getComputedStyle(el);\n if (s.display === 'none') return false;\n if (s.visibility === 'hidden' || s.visibility === 'collapse') return false;\n if (s.opacity === '0') return false;\n if (s.clipPath === 'inset(100%)') return false;\n // Check for offscreen positioning\n const rect = el.getBoundingClientRect();\n if (rect.right < 0 || rect.bottom < 0) return false;\n return true;\n }\n\n // ── Stage 2: Interactive detection ──\n function isInteractive(el) {\n const tag = el.tagName.toLowerCase();\n if (INTERACTIVE_TAGS.has(tag)) {\n // Skip disabled elements\n if (el.disabled) return false;\n // Skip hidden inputs\n if (tag === 'input' && el.type === 'hidden') return false;\n return true;\n }\n const role = el.getAttribute('role');\n if (role && INTERACTIVE_ROLES.has(role)) return true;\n if (el.contentEditable === 'true') return true;\n if (el.tabIndex >= 0 && el.getAttribute('tabindex') !== null) return true;\n if (el.onclick) return true;\n return false;\n }\n\n // ── Stage 8: Attribute filtering ──\n function getAttrs(el) {\n const parts = [];\n for (const name of ATTR_WHITELIST) {\n let v = el.getAttribute(name);\n if (v === null || v === '') continue;\n // Truncate long values\n if (v.length > 80) v = v.slice(0, 77) + '...';\n // Skip href=\"javascript:...\"\n if (name === 'href' && v.startsWith('javascript:')) continue;\n parts.push(name + '=' + v);\n }\n return parts.length ? ' ' + parts.join(' ') : '';\n }\n\n // ── Stage 9: Ad filtering ──\n function isAd(el) {\n const id = el.id || '';\n const cls = el.className || '';\n if (typeof cls === 'string' && AD_PATTERNS.test(cls)) return true;\n if (AD_PATTERNS.test(id)) return true;\n if (el.tagName === 'IFRAME' && AD_PATTERNS.test(el.src || '')) return true;\n return false;\n }\n\n // ── Stage 10: Scroll info ──\n function getScrollInfo(el) {\n const s = getComputedStyle(el);\n const overflowY = s.overflowY;\n const overflowX = s.overflowX;\n const scrollableY = (overflowY === 'auto' || overflowY === 'scroll') && el.scrollHeight > el.clientHeight;\n const scrollableX = (overflowX === 'auto' || overflowX === 'scroll') && el.scrollWidth > el.clientWidth;\n if (!scrollableY && !scrollableX) return '';\n\n const parts = [];\n if (scrollableY) {\n const up = Math.round(el.scrollTop);\n const down = Math.round(el.scrollHeight - el.clientHeight - el.scrollTop);\n if (up > 0) parts.push(up + 'px up');\n if (down > 0) parts.push(down + 'px down');\n }\n if (scrollableX) {\n const left = Math.round(el.scrollLeft);\n const right = Math.round(el.scrollWidth - el.clientWidth - el.scrollLeft);\n if (left > 0) parts.push(left + 'px left');\n if (right > 0) parts.push(right + 'px right');\n }\n return parts.length ? ' |scroll: ' + parts.join(', ') + '|' : '';\n }\n\n // ── Stage 6: Bounding-box dedup ──\n // If a parent and child are both interactive and have ~same bounding box,\n // skip the parent (e.g., <a><button>Click</button></a>)\n function isWrappingInteractive(el) {\n if (!isInteractive(el)) return false;\n const rect = el.getBoundingClientRect();\n if (rect.width === 0 || rect.height === 0) return false;\n for (const child of el.children) {\n if (!isInteractive(child)) continue;\n const cr = child.getBoundingClientRect();\n const overlapX = Math.min(rect.right, cr.right) - Math.max(rect.left, cr.left);\n const overlapY = Math.min(rect.bottom, cr.bottom) - Math.max(rect.top, cr.top);\n const overlapArea = Math.max(0, overlapX) * Math.max(0, overlapY);\n const parentArea = rect.width * rect.height;\n if (parentArea > 0 && overlapArea / parentArea > 0.85) return true;\n }\n return false;\n }\n\n // ── Stage 7: Occlusion detection ──\n function isOccluded(el) {\n const rect = el.getBoundingClientRect();\n if (rect.width === 0 || rect.height === 0) return false;\n const cx = rect.left + rect.width / 2;\n const cy = rect.top + rect.height / 2;\n const topEl = document.elementFromPoint(cx, cy);\n if (!topEl) return false;\n if (topEl === el || el.contains(topEl) || topEl.contains(el)) return false;\n // Check z-index — if top element is a modal/overlay, mark as occluded\n const topZ = parseInt(getComputedStyle(topEl).zIndex) || 0;\n const elZ = parseInt(getComputedStyle(el).zIndex) || 0;\n return topZ > elZ + 10;\n }\n\n // ── Stage 5: Iframe content extraction ──\n function getIframeContent(iframe, depth, maxDepth) {\n try {\n const doc = iframe.contentDocument;\n if (!doc || !doc.body) return '';\n return '\\\\n' + walkNode(doc.body, depth, maxDepth);\n } catch { return ''; }\n }\n\n // ── Stage 4: Shadow DOM traversal ──\n function getShadowContent(el, depth, maxDepth) {\n if (!el.shadowRoot) return '';\n let out = '';\n for (const child of el.shadowRoot.childNodes) {\n out += walkNode(child, depth, maxDepth);\n }\n return out;\n }\n\n // ── Input value hint ──\n function getInputHint(el) {\n const tag = el.tagName.toLowerCase();\n if (tag === 'input') {\n const type = el.type || 'text';\n const val = el.value || '';\n const checked = el.checked;\n if (type === 'checkbox' || type === 'radio') {\n return checked ? ' [checked]' : ' [unchecked]';\n }\n if (val) return ' value=\"' + val.slice(0, 50) + '\"';\n }\n if (tag === 'textarea' && el.value) {\n return ' value=\"' + el.value.slice(0, 50) + '\"';\n }\n if (tag === 'select' && el.selectedOptions?.length) {\n return ' selected=\"' + el.selectedOptions[0].text.slice(0, 40) + '\"';\n }\n return '';\n }\n\n const MAX_DEPTH = 25;\n const MAX_TEXT = 150;\n\n function walkNode(node, depth, maxDepth) {\n if (depth > maxDepth) return '';\n if (!node) return '';\n\n // Text node\n if (node.nodeType === 3) {\n const t = node.textContent.trim();\n if (!t) return '';\n const text = t.length > MAX_TEXT ? t.slice(0, MAX_TEXT) + '...' : t;\n return ' '.repeat(depth) + text + '\\\\n';\n }\n\n // Comment node — skip\n if (node.nodeType === 8) return '';\n\n // Only element nodes from here\n if (node.nodeType !== 1) return '';\n\n const el = node;\n const tag = el.tagName.toLowerCase();\n\n // ── Stage 3: Skip tags ──\n if (SKIP_TAGS.has(tag)) return '';\n\n // ── Stage 2: Visibility ──\n if (!isVisible(el)) return '';\n\n // ── Stage 9: Ad filtering ──\n if (isAd(el)) return '';\n\n // ── Stage 6: Bbox dedup — skip wrapping interactive parent ──\n const skipSelf = isWrappingInteractive(el);\n\n const indent = ' '.repeat(depth);\n const inter = !skipSelf && isInteractive(el);\n let prefix = '';\n if (inter) {\n const thisIdx = idx++;\n // Hash: tag + text + key attributes for diff tracking\n const hashText = tag + ':' + (el.textContent || '').trim().slice(0, 40) + ':' + (el.getAttribute('href') || '') + ':' + (el.getAttribute('aria-label') || '');\n __currentHashes.push(hashText);\n const isNew = __prevHashes && __prevHashes.size > 0 && !__prevHashes.has(hashText);\n prefix = isNew ? '*[' + thisIdx + ']' : '[' + thisIdx + ']';\n }\n\n // ── Stage 11: Annotate with data-ref ──\n if (inter) {\n try { el.dataset.ref = String(idx - 1); } catch {}\n }\n\n // ── Stage 7: Occlusion check for interactive elements ──\n if (inter && isOccluded(el)) {\n // Still include but mark as occluded\n // (agent needs to know element exists but may need to scroll/close modal)\n }\n\n const a = getAttrs(el);\n const scrollInfo = getScrollInfo(el);\n const inputHint = inter ? getInputHint(el) : '';\n\n // Leaf text extraction\n let leafText = '';\n if (el.childNodes.length === 1 && el.childNodes[0].nodeType === 3) {\n const t = el.childNodes[0].textContent.trim();\n if (t) leafText = t.length > MAX_TEXT ? t.slice(0, MAX_TEXT) + '...' : t;\n }\n\n // ── Stage 5: Iframe ──\n if (tag === 'iframe') {\n const iframeContent = getIframeContent(el, depth + 1, maxDepth);\n if (iframeContent) {\n return indent + prefix + '<iframe' + a + '>\\\\n' + iframeContent;\n }\n return '';\n }\n\n // Build output\n let out = '';\n\n if (skipSelf) {\n // Skip self but render children\n for (const c of el.childNodes) out += walkNode(c, depth, maxDepth);\n out += getShadowContent(el, depth, maxDepth);\n return out;\n }\n\n if (inter || leafText || el.children.length === 0) {\n if (leafText) {\n out = indent + prefix + '<' + tag + a + scrollInfo + inputHint + '>' + leafText + '</' + tag + '>\\\\n';\n } else {\n out = indent + prefix + '<' + tag + a + scrollInfo + inputHint + '>\\\\n';\n for (const c of el.childNodes) out += walkNode(c, depth + 1, maxDepth);\n out += getShadowContent(el, depth + 1, maxDepth);\n }\n } else {\n // Non-interactive container — flatten depth if no useful info\n if (scrollInfo) {\n out = indent + '<' + tag + scrollInfo + '>\\\\n';\n for (const c of el.childNodes) out += walkNode(c, depth + 1, maxDepth);\n out += getShadowContent(el, depth + 1, maxDepth);\n } else {\n for (const c of el.childNodes) out += walkNode(c, depth, maxDepth);\n out += getShadowContent(el, depth, maxDepth);\n }\n }\n\n return out;\n }\n\n // ── Page-level scroll info header ──\n const scrollY = window.scrollY;\n const scrollMax = document.documentElement.scrollHeight - window.innerHeight;\n const scrollPct = scrollMax > 0 ? Math.round((scrollY / scrollMax) * 100) : 0;\n const vpW = window.innerWidth;\n const vpH = window.innerHeight;\n const pageH = document.documentElement.scrollHeight;\n\n let header = '';\n header += 'viewport: ' + vpW + 'x' + vpH + ' | page_height: ' + pageH + 'px';\n header += ' | scroll: ' + scrollPct + '%';\n if (scrollY > 50) header += ' (' + Math.round(scrollY) + 'px from top)';\n if (scrollMax - scrollY > 50) header += ' (' + Math.round(scrollMax - scrollY) + 'px more below)';\n header += '\\\\n---\\\\n';\n\n // Store current hashes for next diff comparison\n window.__lobster_prev_hashes = __currentHashes;\n\n return header + walkNode(document.body, 0, MAX_DEPTH);\n})()\n`;\n","/**\n * Semantic tree — W3C accessible name algorithm, XPath, listener detection.\n *\n * Based on Lightpanda's SemanticTree.zig approach:\n * - Accessible name: aria-labelledby → aria-label → alt → title → placeholder → text content\n * - XPath generation for element location\n * - Interactive classification: native, aria, contenteditable, listener, focusable\n * - Disabled state with fieldset inheritance\n * - Input value, option, checked state extraction\n */\nexport const SEMANTIC_TREE_SCRIPT = `\n(() => {\n const SKIP = new Set(['script','style','noscript','svg','head','meta','link','template']);\n\n const ROLE_MAP = {\n a: 'link', button: 'button', input: 'textbox', select: 'combobox',\n textarea: 'textbox', h1: 'heading', h2: 'heading', h3: 'heading',\n h4: 'heading', h5: 'heading', h6: 'heading', nav: 'navigation',\n main: 'main', header: 'banner', footer: 'contentinfo', aside: 'complementary',\n form: 'form', table: 'table', img: 'img', ul: 'list', ol: 'list', li: 'listitem',\n section: 'region', article: 'article', dialog: 'dialog', details: 'group',\n summary: 'button', progress: 'progressbar', meter: 'meter', output: 'status',\n label: 'label', legend: 'legend', fieldset: 'group', option: 'option',\n tr: 'row', td: 'cell', th: 'columnheader', caption: 'caption',\n };\n\n const INTERACTIVE_ROLES = new Set([\n 'button','link','textbox','checkbox','radio','combobox','listbox',\n 'menu','menuitem','tab','switch','slider','searchbox','spinbutton',\n 'option','menuitemcheckbox','menuitemradio','treeitem',\n ]);\n\n // ── W3C Accessible Name Algorithm (simplified) ──\n function getAccessibleName(el) {\n // 1. aria-labelledby (highest priority)\n const labelledBy = el.getAttribute('aria-labelledby');\n if (labelledBy) {\n const ids = labelledBy.split(/\\\\s+/);\n const parts = ids.map(id => {\n const ref = document.getElementById(id);\n return ref ? ref.textContent.trim() : '';\n }).filter(Boolean);\n if (parts.length > 0) return parts.join(' ').slice(0, 120);\n }\n\n // 2. aria-label\n const ariaLabel = el.getAttribute('aria-label');\n if (ariaLabel) return ariaLabel.slice(0, 120);\n\n // 3. alt (for images)\n const alt = el.getAttribute('alt');\n if (alt) return alt.slice(0, 120);\n\n // 4. title\n const title = el.getAttribute('title');\n if (title) return title.slice(0, 120);\n\n // 5. placeholder (for inputs)\n const placeholder = el.getAttribute('placeholder');\n if (placeholder) return placeholder.slice(0, 120);\n\n // 6. value (for buttons)\n if (el.tagName === 'INPUT' && (el.type === 'submit' || el.type === 'button')) {\n const val = el.getAttribute('value');\n if (val) return val.slice(0, 120);\n }\n\n // 7. Associated label\n if (el.id) {\n const label = document.querySelector('label[for=\"' + el.id + '\"]');\n if (label) return label.textContent.trim().slice(0, 120);\n }\n\n // 8. Direct text content (only for leaf-ish elements)\n if (el.children.length <= 2) {\n const text = el.textContent.trim();\n if (text && text.length < 120) return text;\n }\n\n return '';\n }\n\n // ── XPath generation ──\n function getXPath(el) {\n const parts = [];\n let current = el;\n while (current && current.nodeType === 1) {\n let index = 1;\n let sibling = current.previousElementSibling;\n while (sibling) {\n if (sibling.tagName === current.tagName) index++;\n sibling = sibling.previousElementSibling;\n }\n const tag = current.tagName.toLowerCase();\n parts.unshift(tag + '[' + index + ']');\n current = current.parentElement;\n }\n return '/' + parts.join('/');\n }\n\n // ── Interactivity classification ──\n function classifyInteractivity(el) {\n const types = [];\n const tag = el.tagName.toLowerCase();\n\n // Native\n if (['a','button','input','select','textarea','details','summary'].includes(tag)) {\n if (tag === 'a' && !el.href) {} // anchor without href is not interactive\n else if (tag === 'input' && el.type === 'hidden') {} // hidden inputs\n else types.push('native');\n }\n\n // ARIA role\n const role = el.getAttribute('role');\n if (role && INTERACTIVE_ROLES.has(role)) types.push('aria');\n\n // Contenteditable\n if (el.contentEditable === 'true') types.push('contenteditable');\n\n // Focusable\n if (el.tabIndex >= 0 && el.getAttribute('tabindex') !== null) types.push('focusable');\n\n // Event listeners (check onclick and common inline handlers)\n if (el.onclick || el.onmousedown || el.onkeydown || el.onkeypress ||\n el.getAttribute('onclick') || el.getAttribute('onmousedown')) {\n types.push('listener');\n }\n\n return types;\n }\n\n // ── Disabled state with fieldset inheritance ──\n function isDisabled(el) {\n if (el.disabled) return true;\n // Check fieldset disabled inheritance\n let parent = el.parentElement;\n while (parent) {\n if (parent.tagName === 'FIELDSET' && parent.disabled) {\n // Exception: elements inside the first legend child are NOT disabled\n const firstLegend = parent.querySelector(':scope > legend');\n if (firstLegend && firstLegend.contains(el)) return false;\n return true;\n }\n parent = parent.parentElement;\n }\n return false;\n }\n\n // ── Walk the DOM ──\n function walk(el, depth, maxDepth) {\n if (!el || depth > maxDepth) return '';\n\n if (el.nodeType === 3) {\n const t = el.textContent.trim();\n return t ? ' '.repeat(depth) + 'text \"' + t.slice(0, 100) + '\"\\\\n' : '';\n }\n\n if (el.nodeType !== 1) return '';\n const tag = el.tagName.toLowerCase();\n if (SKIP.has(tag)) return '';\n\n const style = getComputedStyle(el);\n if (style.display === 'none' || style.visibility === 'hidden') return '';\n\n const indent = ' '.repeat(depth);\n const role = el.getAttribute('role') || ROLE_MAP[tag] || '';\n const name = getAccessibleName(el);\n const interTypes = classifyInteractivity(el);\n const interactive = interTypes.length > 0;\n const disabled = interactive && isDisabled(el);\n\n let line = indent;\n line += role || tag;\n\n if (name) line += ' \"' + name.slice(0, 80) + '\"';\n\n if (interactive) {\n line += ' [' + interTypes.join(',') + ']';\n if (disabled) line += ' {disabled}';\n line += ' xpath=' + getXPath(el);\n }\n\n // Input state\n if (tag === 'input') {\n const type = el.type || 'text';\n line += ' type=' + type;\n if (type === 'checkbox' || type === 'radio') {\n line += el.checked ? ' [checked]' : ' [unchecked]';\n } else if (el.value) {\n line += ' value=\"' + el.value.slice(0, 50) + '\"';\n }\n }\n if (tag === 'textarea' && el.value) {\n line += ' value=\"' + el.value.slice(0, 50) + '\"';\n }\n if (tag === 'select') {\n const opts = Array.from(el.options || []).map(o => ({\n text: o.text.slice(0, 30),\n value: o.value,\n selected: o.selected,\n }));\n const selected = opts.find(o => o.selected);\n if (selected) line += ' selected=\"' + selected.text + '\"';\n if (opts.length <= 10) {\n line += ' options=[' + opts.map(o => o.text).join('|') + ']';\n }\n }\n\n line += '\\\\n';\n\n let out = line;\n for (const c of el.childNodes) {\n out += walk(c, depth + 1, maxDepth);\n }\n\n // Shadow DOM\n if (el.shadowRoot) {\n for (const c of el.shadowRoot.childNodes) {\n out += walk(c, depth + 1, maxDepth);\n }\n }\n\n return out;\n }\n\n return walk(document.body, 0, 20);\n})()\n`;\n","/**\n * DOM-to-Markdown converter — runs inside the browser.\n *\n * Full-featured conversion based on Lightpanda's markdown.zig:\n * - Table support with header separator rows\n * - URL resolution (relative → absolute)\n * - Nested ordered/unordered lists with proper indentation\n * - Character escaping for Markdown special chars\n * - Strikethrough, code blocks, blockquotes\n * - Smart anchor handling (inline vs block)\n * - Whitespace collapsing\n */\nexport const MARKDOWN_SCRIPT = `\n(() => {\n const SKIP = new Set(['script','style','noscript','svg','head','template']);\n const baseUrl = location.href;\n\n // Resolve relative URLs to absolute\n function resolveUrl(href) {\n if (!href || href.startsWith('javascript:') || href.startsWith('#')) return href;\n try { return new URL(href, baseUrl).href; } catch { return href; }\n }\n\n // Escape Markdown special chars in text\n function escapeText(text) {\n return text\n .replace(/\\\\\\\\/g, '\\\\\\\\\\\\\\\\')\n .replace(/([*_~\\`\\\\[\\\\]|])/g, '\\\\\\\\$1');\n }\n\n // State tracking\n let listDepth = 0;\n let orderedCounters = [];\n let inPre = false;\n let inTable = false;\n\n function listIndent() { return ' '.repeat(listDepth); }\n\n function walk(el) {\n if (!el) return '';\n\n // Text node\n if (el.nodeType === 3) {\n const text = el.textContent || '';\n if (inPre) return text;\n // Collapse whitespace\n const collapsed = text.replace(/\\\\s+/g, ' ');\n return collapsed === ' ' && !el.previousSibling && !el.nextSibling ? '' : collapsed;\n }\n\n if (el.nodeType !== 1) return '';\n const tag = el.tagName.toLowerCase();\n if (SKIP.has(tag)) return '';\n\n // Visibility check\n try {\n const s = getComputedStyle(el);\n if (s.display === 'none' || s.visibility === 'hidden') return '';\n } catch {}\n\n // Get children content\n function childContent() {\n let out = '';\n for (const c of el.childNodes) out += walk(c);\n return out;\n }\n\n switch (tag) {\n // ── Headings ──\n case 'h1': return '\\\\n\\\\n# ' + childContent().trim() + '\\\\n\\\\n';\n case 'h2': return '\\\\n\\\\n## ' + childContent().trim() + '\\\\n\\\\n';\n case 'h3': return '\\\\n\\\\n### ' + childContent().trim() + '\\\\n\\\\n';\n case 'h4': return '\\\\n\\\\n#### ' + childContent().trim() + '\\\\n\\\\n';\n case 'h5': return '\\\\n\\\\n##### ' + childContent().trim() + '\\\\n\\\\n';\n case 'h6': return '\\\\n\\\\n###### ' + childContent().trim() + '\\\\n\\\\n';\n\n // ── Block elements ──\n case 'p': return '\\\\n\\\\n' + childContent().trim() + '\\\\n\\\\n';\n case 'br': return '\\\\n';\n case 'hr': return '\\\\n\\\\n---\\\\n\\\\n';\n\n // ── Inline formatting ──\n case 'strong': case 'b': {\n const inner = childContent().trim();\n return inner ? '**' + inner + '**' : '';\n }\n case 'em': case 'i': {\n const inner = childContent().trim();\n return inner ? '*' + inner + '*' : '';\n }\n case 's': case 'del': case 'strike': {\n const inner = childContent().trim();\n return inner ? '~~' + inner + '~~' : '';\n }\n case 'code': {\n if (inPre) return childContent();\n const inner = childContent();\n return inner ? '\\\\x60' + inner + '\\\\x60' : '';\n }\n\n // ── Code blocks ──\n case 'pre': {\n inPre = true;\n const inner = childContent();\n inPre = false;\n const lang = el.querySelector('code')?.className?.match(/language-(\\\\w+)/)?.[1] || '';\n return '\\\\n\\\\n\\\\x60\\\\x60\\\\x60' + lang + '\\\\n' + inner.trim() + '\\\\n\\\\x60\\\\x60\\\\x60\\\\n\\\\n';\n }\n\n // ── Links ──\n case 'a': {\n const href = resolveUrl(el.getAttribute('href') || '');\n const inner = childContent().trim();\n const name = inner || el.getAttribute('aria-label') || el.getAttribute('title') || '';\n if (!name) return '';\n if (!href || href === '#' || href.startsWith('javascript:')) return name;\n return '[' + name + '](' + href + ')';\n }\n\n // ── Images ──\n case 'img': {\n const alt = el.getAttribute('alt') || '';\n const src = resolveUrl(el.getAttribute('src') || '');\n return src ? '' : '';\n }\n\n // ── Lists ──\n case 'ul': {\n listDepth++;\n orderedCounters.push(0);\n const inner = childContent();\n listDepth--;\n orderedCounters.pop();\n return '\\\\n' + inner;\n }\n case 'ol': {\n listDepth++;\n orderedCounters.push(0);\n const inner = childContent();\n listDepth--;\n orderedCounters.pop();\n return '\\\\n' + inner;\n }\n case 'li': {\n const parent = el.parentElement?.tagName?.toLowerCase();\n const isOrdered = parent === 'ol';\n const inner = childContent().trim();\n if (!inner) return '';\n if (isOrdered) {\n const counter = orderedCounters.length > 0\n ? ++orderedCounters[orderedCounters.length - 1] : 1;\n return listIndent() + counter + '. ' + inner + '\\\\n';\n }\n return listIndent() + '- ' + inner + '\\\\n';\n }\n\n // ── Blockquote ──\n case 'blockquote': {\n const inner = childContent().trim();\n if (!inner) return '';\n return '\\\\n\\\\n' + inner.split('\\\\n').map(line => '> ' + line).join('\\\\n') + '\\\\n\\\\n';\n }\n\n // ── Tables ──\n case 'table': {\n inTable = true;\n let out = '\\\\n\\\\n';\n const rows = el.querySelectorAll('tr');\n let headerDone = false;\n\n for (let i = 0; i < rows.length; i++) {\n const cells = rows[i].querySelectorAll('th, td');\n const isHeader = rows[i].querySelector('th') !== null;\n const cellTexts = [];\n for (const cell of cells) {\n let cellText = '';\n for (const c of cell.childNodes) cellText += walk(c);\n cellTexts.push(cellText.trim().replace(/\\\\|/g, '\\\\\\\\|').replace(/\\\\n/g, ' '));\n }\n\n out += '| ' + cellTexts.join(' | ') + ' |\\\\n';\n\n if (isHeader && !headerDone) {\n out += '| ' + cellTexts.map(() => '---').join(' | ') + ' |\\\\n';\n headerDone = true;\n }\n\n // First data row without headers — synthesize separator\n if (i === 0 && !isHeader && !headerDone) {\n out += '| ' + cellTexts.map(() => '---').join(' | ') + ' |\\\\n';\n headerDone = true;\n }\n }\n\n inTable = false;\n return out + '\\\\n';\n }\n case 'thead': case 'tbody': case 'tfoot':\n return childContent();\n case 'tr': case 'td': case 'th':\n // Handled by table walker above; fallback for orphaned elements\n return childContent();\n\n // ── Definition lists ──\n case 'dl': return '\\\\n\\\\n' + childContent() + '\\\\n\\\\n';\n case 'dt': return '\\\\n**' + childContent().trim() + '**\\\\n';\n case 'dd': return ': ' + childContent().trim() + '\\\\n';\n\n // ── Figure ──\n case 'figure': return '\\\\n\\\\n' + childContent().trim() + '\\\\n\\\\n';\n case 'figcaption': return '\\\\n*' + childContent().trim() + '*\\\\n';\n\n // ── Details/Summary ──\n case 'details': return '\\\\n\\\\n' + childContent() + '\\\\n\\\\n';\n case 'summary': return '**' + childContent().trim() + '**\\\\n\\\\n';\n\n // ── Generic blocks ──\n case 'div': case 'section': case 'article': case 'main': case 'aside':\n case 'header': case 'footer': case 'nav':\n return '\\\\n' + childContent() + '\\\\n';\n\n case 'span': case 'small': case 'sub': case 'sup': case 'abbr':\n case 'time': case 'mark': case 'cite': case 'q':\n return childContent();\n\n default:\n return childContent();\n }\n }\n\n const raw = walk(document.body);\n // Clean up: collapse 3+ newlines to 2, trim\n return raw.replace(/\\\\n{3,}/g, '\\\\n\\\\n').replace(/^\\\\n+|\\\\n+$/g, '').trim();\n})()\n`;\n","/**\n * Form state extraction — runs inside the browser.\n *\n * Extracts all form fields (including orphan fields not in <form> tags),\n * their types, labels, values, required/disabled state.\n *\n * Based on OpenCLI's getFormStateJs() pattern.\n */\nexport const FORM_STATE_SCRIPT = `\n(() => {\n function extractField(el) {\n const tag = el.tagName.toLowerCase();\n const type = (el.getAttribute('type') || tag).toLowerCase();\n\n // Skip non-user-facing inputs\n if (['hidden', 'submit', 'button', 'reset', 'image'].includes(type)) return null;\n\n const name = el.name || el.id || '';\n\n // Find label via multiple strategies\n const label =\n el.getAttribute('aria-label') ||\n (el.id ? document.querySelector('label[for=\"' + el.id + '\"]')?.textContent?.trim() : null) ||\n el.closest('label')?.textContent?.trim() ||\n el.placeholder ||\n '';\n\n // Extract value based on type\n let value;\n if (tag === 'select') {\n const selected = el.options[el.selectedIndex];\n value = selected ? selected.textContent.trim() : '';\n } else if (type === 'checkbox' || type === 'radio') {\n value = el.checked;\n } else if (type === 'password') {\n value = el.value ? '••••' : '';\n } else if (el.isContentEditable) {\n value = el.textContent?.trim()?.slice(0, 200) || '';\n } else {\n value = el.value || '';\n }\n\n return {\n tag,\n type,\n name,\n label: label.slice(0, 80),\n value: typeof value === 'string' ? value.slice(0, 200) : value,\n required: !!el.required,\n disabled: !!el.disabled,\n ref: el.dataset?.ref || null,\n };\n }\n\n const result = { forms: [], orphanFields: [] };\n\n // Collect forms\n for (const form of document.forms) {\n const fields = [];\n for (const el of form.elements) {\n const field = extractField(el);\n if (field) fields.push(field);\n }\n result.forms.push({\n id: form.id || '',\n name: form.name || '',\n action: form.action || '',\n method: (form.method || 'get').toUpperCase(),\n fields,\n });\n }\n\n // Collect orphan fields (not in a <form>)\n const allInputs = document.querySelectorAll(\n 'input, textarea, select, [contenteditable=\"true\"]'\n );\n for (const el of allInputs) {\n if (!el.form) {\n const field = extractField(el);\n if (field) result.orphanFields.push(field);\n }\n }\n\n return result;\n})()\n`;\n","/**\n * Network interceptor script — patches fetch and XHR to capture responses.\n * Based on OpenCLI's interception approach.\n */\nexport function buildInterceptorScript(pattern: string): string {\n return `\n(() => {\n if (window.__lobster_interceptor__) return;\n window.__lobster_interceptor__ = { requests: [] };\n const store = window.__lobster_interceptor__;\n const pattern = ${JSON.stringify(pattern)};\n\n // Patch fetch\n const origFetch = window.fetch;\n window.fetch = async function(...args) {\n const url = typeof args[0] === 'string' ? args[0] : args[0]?.url || '';\n const resp = await origFetch.apply(this, args);\n if (url.includes(pattern)) {\n const clone = resp.clone();\n try {\n const body = await clone.json();\n store.requests.push({ url, method: 'GET', status: resp.status, body, timestamp: Date.now() });\n } catch {}\n }\n return resp;\n };\n\n // Patch XHR\n const origOpen = XMLHttpRequest.prototype.open;\n const origSend = XMLHttpRequest.prototype.send;\n XMLHttpRequest.prototype.open = function(method, url, ...rest) {\n this.__url = url;\n this.__method = method;\n return origOpen.call(this, method, url, ...rest);\n };\n XMLHttpRequest.prototype.send = function(...args) {\n this.addEventListener('load', function() {\n if (this.__url && this.__url.includes(pattern)) {\n try {\n const body = JSON.parse(this.responseText);\n store.requests.push({ url: this.__url, method: this.__method, status: this.status, body, timestamp: Date.now() });\n } catch {}\n }\n });\n return origSend.apply(this, args);\n };\n})()\n`;\n}\n\nexport const GET_INTERCEPTED_SCRIPT = `\n(() => {\n const store = window.__lobster_interceptor__;\n if (!store) return [];\n const reqs = [...store.requests];\n store.requests = [];\n return reqs;\n})()\n`;\n","import type { Page } from 'puppeteer-core';\nimport type {\n IPage, WaitCondition, Cookie, NetworkEntry, TabInfo,\n SnapshotOptions, SemanticTreeOptions, FlatDomTree, BrowserState, FormState,\n} from '../types/page.js';\nimport { FLAT_TREE_SCRIPT, flatTreeToString } from './dom/flat-tree.js';\nimport { SNAPSHOT_SCRIPT } from './dom/snapshot.js';\nimport { SEMANTIC_TREE_SCRIPT } from './dom/semantic-tree.js';\nimport { MARKDOWN_SCRIPT } from './dom/markdown.js';\nimport { FORM_STATE_SCRIPT } from './dom/form-state.js';\nimport { buildInterceptorScript, GET_INTERCEPTED_SCRIPT } from './interceptor.js';\n\nexport class PuppeteerPage implements IPage {\n private page: Page;\n\n constructor(page: Page) {\n this.page = page;\n }\n\n get raw(): Page { return this.page; }\n\n async goto(url: string, options?: { waitUntil?: WaitCondition; timeout?: number }): Promise<void> {\n await this.page.goto(url, {\n waitUntil: (options?.waitUntil as any) || 'networkidle2',\n timeout: options?.timeout || 30000,\n });\n }\n\n async goBack(): Promise<void> {\n await this.page.goBack({ waitUntil: 'networkidle2' });\n }\n\n async url(): Promise<string> {\n return this.page.url();\n }\n\n async title(): Promise<string> {\n return this.page.title();\n }\n\n async evaluate<T = unknown>(js: string): Promise<T> {\n return this.page.evaluate(js) as Promise<T>;\n }\n\n async snapshot(_opts?: SnapshotOptions): Promise<string> {\n return this.page.evaluate(SNAPSHOT_SCRIPT) as Promise<string>;\n }\n\n async semanticTree(_opts?: SemanticTreeOptions): Promise<string> {\n return this.page.evaluate(SEMANTIC_TREE_SCRIPT) as Promise<string>;\n }\n\n async flatTree(): Promise<FlatDomTree> {\n const raw = await this.page.evaluate(FLAT_TREE_SCRIPT);\n return raw as FlatDomTree;\n }\n\n async markdown(): Promise<string> {\n return this.page.evaluate(MARKDOWN_SCRIPT) as Promise<string>;\n }\n\n async browserState(): Promise<BrowserState> {\n const state = await this.page.evaluate(`\n (() => {\n const scrollY = window.scrollY;\n const scrollX = window.scrollX;\n const vpW = window.innerWidth;\n const vpH = window.innerHeight;\n const pageW = document.documentElement.scrollWidth;\n const pageH = document.documentElement.scrollHeight;\n const maxScrollY = pageH - vpH;\n return {\n url: location.href,\n title: document.title,\n viewportWidth: vpW,\n viewportHeight: vpH,\n pageWidth: pageW,\n pageHeight: pageH,\n scrollX: scrollX,\n scrollY: scrollY,\n scrollPercent: maxScrollY > 0 ? Math.round((scrollY / maxScrollY) * 100) : 0,\n pixelsAbove: Math.round(scrollY),\n pixelsBelow: Math.round(Math.max(0, maxScrollY - scrollY)),\n };\n })()\n `) as BrowserState;\n return state;\n }\n\n async formState(): Promise<FormState> {\n return this.page.evaluate(FORM_STATE_SCRIPT) as Promise<FormState>;\n }\n\n async click(ref: string | number): Promise<void> {\n if (typeof ref === 'number') {\n await this.page.evaluate((idx) => {\n const el = document.querySelector('[data-ref=\"' + idx + '\"]') as HTMLElement;\n if (!el) throw new Error('Element with index ' + idx + ' not found');\n\n // Blur previously focused element\n const prev = document.activeElement as HTMLElement | null;\n if (prev && prev !== el && prev !== document.body) {\n prev.blur();\n prev.dispatchEvent(new MouseEvent('mouseout', { bubbles: true, cancelable: true }));\n prev.dispatchEvent(new MouseEvent('mouseleave', { bubbles: false, cancelable: true }));\n }\n\n // Scroll into view\n if (typeof (el as any).scrollIntoViewIfNeeded === 'function') {\n (el as any).scrollIntoViewIfNeeded();\n } else {\n el.scrollIntoView({ behavior: 'auto', block: 'center', inline: 'nearest' });\n }\n\n // Full mouse event sequence — required for React, analytics, custom handlers\n el.dispatchEvent(new MouseEvent('mouseenter', { bubbles: true, cancelable: true }));\n el.dispatchEvent(new MouseEvent('mouseover', { bubbles: true, cancelable: true }));\n el.dispatchEvent(new MouseEvent('mousedown', { bubbles: true, cancelable: true }));\n el.focus();\n el.dispatchEvent(new MouseEvent('mouseup', { bubbles: true, cancelable: true }));\n el.dispatchEvent(new MouseEvent('click', { bubbles: true, cancelable: true }));\n }, ref);\n // Wait for click processing (animations, state updates)\n await new Promise((r) => setTimeout(r, 200));\n } else {\n await this.page.click(ref);\n }\n }\n\n async typeText(ref: string | number, text: string): Promise<void> {\n if (typeof ref === 'number') {\n // First click the element (triggers full event sequence + focus)\n await this.click(ref);\n\n await this.page.evaluate((idx, txt) => {\n const el = document.querySelector('[data-ref=\"' + idx + '\"]') as HTMLElement;\n if (!el) throw new Error('Element with index ' + idx + ' not found');\n\n const isInput = el.tagName === 'INPUT' || el.tagName === 'TEXTAREA';\n const isContentEditable = el.isContentEditable;\n\n if (isContentEditable) {\n // ── Contenteditable: Plan A — synthetic InputEvents ──\n // Works for: React contenteditable, Quill\n // Clear existing content\n if (el.dispatchEvent(new InputEvent('beforeinput', {\n bubbles: true, cancelable: true, inputType: 'deleteContent',\n }))) {\n el.innerText = '';\n el.dispatchEvent(new InputEvent('input', {\n bubbles: true, inputType: 'deleteContent',\n }));\n }\n\n // Insert new text\n if (el.dispatchEvent(new InputEvent('beforeinput', {\n bubbles: true, cancelable: true, inputType: 'insertText', data: txt,\n }))) {\n el.innerText = txt;\n el.dispatchEvent(new InputEvent('input', {\n bubbles: true, inputType: 'insertText', data: txt,\n }));\n }\n\n // Verify Plan A worked\n const planAOk = el.innerText.trim() === txt.trim();\n\n if (!planAOk) {\n // ── Plan B — execCommand fallback ──\n // Works for: Slate.js, some rich-text editors\n el.focus();\n const doc = el.ownerDocument;\n const sel = (doc.defaultView || window).getSelection();\n const range = doc.createRange();\n range.selectNodeContents(el);\n sel?.removeAllRanges();\n sel?.addRange(range);\n doc.execCommand('delete', false);\n doc.execCommand('insertText', false, txt);\n }\n\n el.dispatchEvent(new Event('change', { bubbles: true }));\n el.blur();\n\n } else if (isInput) {\n // ── Input/Textarea: use native value setter to bypass React/Vue ──\n const inputEl = el as HTMLInputElement | HTMLTextAreaElement;\n const proto = Object.getPrototypeOf(inputEl);\n const descriptor =\n Object.getOwnPropertyDescriptor(proto, 'value') ||\n Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, 'value') ||\n Object.getOwnPropertyDescriptor(HTMLTextAreaElement.prototype, 'value');\n\n if (descriptor?.set) {\n descriptor.set.call(inputEl, txt);\n } else {\n inputEl.value = txt;\n }\n\n inputEl.dispatchEvent(new Event('input', { bubbles: true }));\n inputEl.dispatchEvent(new Event('change', { bubbles: true }));\n } else {\n // Fallback: try setting value anyway\n (el as any).value = txt;\n el.dispatchEvent(new Event('input', { bubbles: true }));\n el.dispatchEvent(new Event('change', { bubbles: true }));\n }\n }, ref, text);\n } else {\n // CSS selector path — click to focus, then use keyboard\n await this.page.click(ref, { count: 3 });\n await this.page.keyboard.type(text);\n }\n }\n\n async pressKey(key: string): Promise<void> {\n await this.page.keyboard.press(key as any);\n }\n\n async selectOption(ref: string | number, value: string): Promise<void> {\n const selector = typeof ref === 'number' ? '[data-ref=\"' + ref + '\"]' : ref;\n await this.page.select(selector, value);\n }\n\n async scroll(direction: 'up' | 'down' | 'left' | 'right', amount?: number): Promise<void> {\n const distance = amount || 500;\n const isVertical = direction === 'up' || direction === 'down';\n const positive = direction === 'down' || direction === 'right';\n const delta = positive ? distance : -distance;\n\n await this.page.evaluate((dy, dx, isVert) => {\n // Helper: check if element is a valid scroll container\n const canScroll = (el) => {\n if (!el) return false;\n const s = getComputedStyle(el);\n if (isVert) {\n return /(auto|scroll|overlay)/.test(s.overflowY) &&\n el.scrollHeight > el.clientHeight &&\n el.clientHeight >= window.innerHeight * 0.3;\n } else {\n return /(auto|scroll|overlay)/.test(s.overflowX) &&\n el.scrollWidth > el.clientWidth &&\n el.clientWidth >= window.innerWidth * 0.3;\n }\n };\n\n // Walk from active element up to find a scrollable container\n let el = document.activeElement;\n while (el && !canScroll(el) && el !== document.body) {\n el = el.parentElement;\n }\n\n // If no scrollable ancestor, search the DOM\n if (!canScroll(el)) {\n el = Array.from(document.querySelectorAll('*')).find(canScroll) || null;\n }\n\n const isPageLevel = !el || el === document.body ||\n el === document.documentElement || el === document.scrollingElement;\n\n if (isPageLevel) {\n // Page-level scroll\n if (isVert) {\n window.scrollBy(0, dy);\n } else {\n window.scrollBy(dx, 0);\n }\n } else {\n // Container scroll\n if (isVert) {\n el.scrollBy({ top: dy, behavior: 'smooth' });\n } else {\n el.scrollBy({ left: dx, behavior: 'smooth' });\n }\n }\n }, isVertical ? delta : 0, isVertical ? 0 : delta, isVertical);\n\n // Wait for smooth scroll to settle\n await new Promise((r) => setTimeout(r, 150));\n }\n\n async scrollToElement(ref: string | number): Promise<void> {\n const selector = typeof ref === 'number' ? '[data-ref=\"' + ref + '\"]' : ref;\n await this.page.evaluate((sel) => {\n const el = document.querySelector(sel);\n if (!el) return;\n if (typeof (el as any).scrollIntoViewIfNeeded === 'function') {\n (el as any).scrollIntoViewIfNeeded();\n } else {\n el.scrollIntoView({ behavior: 'auto', block: 'center', inline: 'nearest' });\n }\n }, selector);\n }\n\n async getCookies(opts?: { domain?: string }): Promise<Cookie[]> {\n const cookies = await this.page.cookies();\n const filtered = opts?.domain\n ? cookies.filter((c) => c.domain.includes(opts.domain!))\n : cookies;\n return filtered.map((c) => ({\n name: c.name,\n value: c.value,\n domain: c.domain,\n path: c.path,\n expires: c.expires,\n httpOnly: c.httpOnly,\n secure: c.secure,\n sameSite: c.sameSite as Cookie['sameSite'],\n }));\n }\n\n async wait(options: number | { text?: string; time?: number; timeout?: number }): Promise<void> {\n if (typeof options === 'number') {\n await new Promise((r) => setTimeout(r, options * 1000));\n return;\n }\n if (options.time) {\n await new Promise((r) => setTimeout(r, options.time! * 1000));\n }\n if (options.text) {\n await this.page.waitForFunction(\n (t) => document.body.innerText.includes(t),\n { timeout: options.timeout || 30000 },\n options.text\n );\n }\n }\n\n async networkRequests(includeStatic?: boolean): Promise<NetworkEntry[]> {\n // Use Performance API to extract network requests from the browser\n const entries = await this.page.evaluate(`\n (() => {\n const entries = performance.getEntriesByType('resource');\n const staticTypes = new Set(['img', 'font', 'css', 'script', 'link']);\n const includeStatic = ${!!includeStatic};\n\n return entries\n .filter(e => includeStatic || !staticTypes.has(e.initiatorType))\n .map(e => ({\n url: e.name,\n method: 'GET',\n status: 200,\n type: e.initiatorType || 'other',\n size: e.transferSize || e.encodedBodySize || 0,\n duration: Math.round(e.duration),\n }));\n })()\n `) as NetworkEntry[];\n return entries || [];\n }\n\n async installInterceptor(pattern: string): Promise<void> {\n await this.page.evaluate(buildInterceptorScript(pattern));\n }\n\n async getInterceptedRequests(): Promise<unknown[]> {\n return this.page.evaluate(GET_INTERCEPTED_SCRIPT) as Promise<unknown[]>;\n }\n\n async screenshot(opts?: { format?: 'png' | 'jpeg'; fullPage?: boolean }): Promise<Buffer> {\n const result = await this.page.screenshot({\n type: opts?.format || 'png',\n fullPage: opts?.fullPage ?? false,\n });\n return Buffer.from(result);\n }\n\n async tabs(): Promise<TabInfo[]> {\n const browser = this.page.browser();\n const pages = await browser.pages();\n return pages.map((p, i) => ({\n id: i,\n url: p.url(),\n title: '',\n active: p === this.page,\n }));\n }\n\n async close(): Promise<void> {\n await this.page.close();\n }\n}\n","/**\n * Interactive element classification — identifies what's clickable/editable.\n * Based on Lightpanda's InteractivityType approach.\n */\nexport const INTERACTIVE_ELEMENTS_SCRIPT = `\n(() => {\n const results = [];\n\n function classify(el) {\n const tag = el.tagName.toLowerCase();\n const role = el.getAttribute('role');\n const types = [];\n\n // Native interactive\n if (['a', 'button', 'input', 'select', 'textarea', 'details', 'summary'].includes(tag)) {\n types.push('native');\n }\n\n // ARIA role interactive\n if (role && ['button', 'link', 'textbox', 'checkbox', 'radio', 'combobox', 'tab', 'switch', 'menuitem', 'slider'].includes(role)) {\n types.push('aria');\n }\n\n // Contenteditable\n if (el.contentEditable === 'true') types.push('contenteditable');\n\n // Focusable\n if (el.tabIndex >= 0 && el.getAttribute('tabindex') !== null) types.push('focusable');\n\n // Has click listener (approximate)\n if (el.onclick) types.push('listener');\n\n return types;\n }\n\n let idx = 0;\n const walker = document.createTreeWalker(document.body, NodeFilter.SHOW_ELEMENT);\n let node;\n while (node = walker.nextNode()) {\n const types = classify(node);\n if (types.length === 0) continue;\n\n const style = getComputedStyle(node);\n if (style.display === 'none' || style.visibility === 'hidden') continue;\n\n const rect = node.getBoundingClientRect();\n results.push({\n index: idx++,\n tag: node.tagName.toLowerCase(),\n role: node.getAttribute('role') || '',\n text: (node.textContent || '').trim().slice(0, 100),\n types,\n ariaLabel: node.getAttribute('aria-label') || '',\n rect: { x: rect.x, y: rect.y, width: rect.width, height: rect.height },\n });\n }\n\n return results;\n})()\n`;\n","import type { Page } from 'puppeteer-core';\nimport type { WaitCondition } from '../types/page.js';\n\nexport async function waitForCondition(\n page: Page,\n condition: WaitCondition,\n timeout: number = 30000\n): Promise<void> {\n switch (condition) {\n case 'load':\n await page.waitForNavigation({ waitUntil: 'load', timeout }).catch(() => {});\n break;\n case 'domcontentloaded':\n await page.waitForNavigation({ waitUntil: 'domcontentloaded', timeout }).catch(() => {});\n break;\n case 'networkidle0':\n await page.waitForNavigation({ waitUntil: 'networkidle0', timeout }).catch(() => {});\n break;\n case 'networkidle2':\n await page.waitForNavigation({ waitUntil: 'networkidle2', timeout }).catch(() => {});\n break;\n }\n}\n","/**\n * LobsterEngine — In-house lightweight HTML parser + content extractor.\n *\n * No Chrome, no external binary. Pure Node.js.\n * Fetches HTML via HTTP, parses into a DOM tree, extracts content\n * as markdown, text, links, or structured snapshot.\n *\n * Inspired by Lightpanda's approach but 100% our code.\n * Works for server-rendered pages. For JS-heavy SPAs, use Chrome engine.\n */\n\n// ── HTML Node types ──\n\nexport interface HtmlNode {\n type: 'element' | 'text' | 'comment';\n tag?: string;\n attributes?: Record<string, string>;\n children?: HtmlNode[];\n text?: string;\n selfClosing?: boolean;\n}\n\n// ── HTML Parser ──\n\nconst SELF_CLOSING = new Set([\n 'area', 'base', 'br', 'col', 'embed', 'hr', 'img', 'input',\n 'link', 'meta', 'param', 'source', 'track', 'wbr',\n]);\n\nconst RAWTEXT_TAGS = new Set(['script', 'style', 'textarea', 'title']);\n\n/**\n * Parse HTML string into a tree of HtmlNodes.\n * Handles: tags, attributes, self-closing, rawtext, comments, entities.\n */\nexport function parseHtml(html: string): HtmlNode[] {\n const root: HtmlNode[] = [];\n const stack: { node: HtmlNode; children: HtmlNode[] }[] = [{ node: { type: 'element', tag: 'root', children: root }, children: root }];\n let pos = 0;\n\n function current() { return stack[stack.length - 1]; }\n\n function addText(text: string) {\n if (!text) return;\n const decoded = decodeEntities(text);\n if (decoded.trim() || decoded.includes('\\n')) {\n current().children.push({ type: 'text', text: decoded });\n }\n }\n\n while (pos < html.length) {\n const nextTag = html.indexOf('<', pos);\n\n if (nextTag === -1) {\n addText(html.slice(pos));\n break;\n }\n\n if (nextTag > pos) {\n addText(html.slice(pos, nextTag));\n }\n\n // Comment\n if (html.startsWith('<!--', nextTag)) {\n const endComment = html.indexOf('-->', nextTag + 4);\n pos = endComment === -1 ? html.length : endComment + 3;\n continue;\n }\n\n // Doctype\n if (html.startsWith('<!', nextTag) || html.startsWith('<?', nextTag)) {\n const endDoctype = html.indexOf('>', nextTag);\n pos = endDoctype === -1 ? html.length : endDoctype + 1;\n continue;\n }\n\n // Closing tag\n if (html[nextTag + 1] === '/') {\n const endClose = html.indexOf('>', nextTag);\n if (endClose === -1) { pos = html.length; break; }\n const closeTag = html.slice(nextTag + 2, endClose).trim().toLowerCase();\n pos = endClose + 1;\n\n // Pop stack until we find matching tag\n for (let i = stack.length - 1; i > 0; i--) {\n if (stack[i].node.tag === closeTag) {\n stack.length = i;\n break;\n }\n }\n continue;\n }\n\n // Opening tag\n const tagEnd = html.indexOf('>', nextTag);\n if (tagEnd === -1) { pos = html.length; break; }\n\n const tagContent = html.slice(nextTag + 1, tagEnd);\n const selfClose = tagContent.endsWith('/');\n const cleanContent = selfClose ? tagContent.slice(0, -1).trim() : tagContent.trim();\n\n // Parse tag name and attributes\n const spaceIdx = cleanContent.search(/[\\s/]/);\n const tagName = (spaceIdx === -1 ? cleanContent : cleanContent.slice(0, spaceIdx)).toLowerCase();\n const attrStr = spaceIdx === -1 ? '' : cleanContent.slice(spaceIdx);\n\n if (!tagName || tagName.startsWith('!')) {\n pos = tagEnd + 1;\n continue;\n }\n\n const attributes = parseAttributes(attrStr);\n const isSelfClosing = selfClose || SELF_CLOSING.has(tagName);\n\n const node: HtmlNode = {\n type: 'element',\n tag: tagName,\n attributes,\n children: isSelfClosing ? undefined : [],\n selfClosing: isSelfClosing,\n };\n\n current().children.push(node);\n pos = tagEnd + 1;\n\n if (isSelfClosing) continue;\n\n // Raw text tags (script, style) — consume until closing tag\n if (RAWTEXT_TAGS.has(tagName)) {\n const endRaw = html.toLowerCase().indexOf(`</${tagName}`, pos);\n if (endRaw !== -1) {\n const rawText = html.slice(pos, endRaw);\n if (rawText.trim()) {\n node.children!.push({ type: 'text', text: rawText });\n }\n pos = html.indexOf('>', endRaw) + 1;\n }\n continue;\n }\n\n // Push onto stack for children\n stack.push({ node, children: node.children! });\n\n // Auto-close certain tags\n if (tagName === 'p' || tagName === 'li' || tagName === 'td' || tagName === 'th' || tagName === 'dt' || tagName === 'dd') {\n // Check if parent is same tag — auto-close\n if (stack.length >= 3 && stack[stack.length - 2].node.tag === tagName) {\n stack.splice(stack.length - 2, 1);\n }\n }\n }\n\n return root;\n}\n\nfunction parseAttributes(str: string): Record<string, string> {\n const attrs: Record<string, string> = {};\n const re = /(\\w[\\w-]*)(?:\\s*=\\s*(?:\"([^\"]*)\"|'([^']*)'|(\\S+)))?/g;\n let m;\n while ((m = re.exec(str)) !== null) {\n attrs[m[1].toLowerCase()] = decodeEntities(m[2] ?? m[3] ?? m[4] ?? '');\n }\n return attrs;\n}\n\nfunction decodeEntities(text: string): string {\n return text\n .replace(/&/g, '&')\n .replace(/</g, '<')\n .replace(/>/g, '>')\n .replace(/"/g, '\"')\n .replace(/'/g, \"'\")\n .replace(/'/g, \"'\")\n .replace(/ /g, ' ')\n .replace(/&#(\\d+);/g, (_, n) => String.fromCharCode(parseInt(n)))\n .replace(/&#x([0-9a-fA-F]+);/g, (_, n) => String.fromCharCode(parseInt(n, 16)));\n}\n\n// ── Content Extractors ──\n\nconst SKIP_TAGS = new Set(['script', 'style', 'noscript', 'svg', 'head', 'template', 'iframe']);\nconst BLOCK_TAGS = new Set(['div', 'p', 'section', 'article', 'main', 'aside', 'blockquote', 'pre', 'ul', 'ol', 'li', 'table', 'tr', 'td', 'th', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'br', 'hr', 'figure', 'figcaption', 'details', 'summary', 'dl', 'dt', 'dd', 'header', 'footer', 'nav', 'form']);\nconst HEADING_LEVELS: Record<string, string> = { h1: '#', h2: '##', h3: '###', h4: '####', h5: '#####', h6: '######' };\n\nconst INTERACTIVE_TAGS = new Set(['a', 'button', 'input', 'select', 'textarea', 'details', 'summary']);\nconst INTERACTIVE_ROLES = new Set(['button', 'link', 'textbox', 'checkbox', 'radio', 'combobox', 'tab', 'switch', 'menuitem']);\n\n/**\n * Extract plain text from parsed HTML.\n */\nexport function extractText(nodes: HtmlNode[]): string {\n let out = '';\n for (const node of nodes) {\n if (node.type === 'text') {\n out += node.text;\n continue;\n }\n if (node.type !== 'element' || !node.tag) continue;\n if (SKIP_TAGS.has(node.tag)) continue;\n\n const inner = node.children ? extractText(node.children) : '';\n if (BLOCK_TAGS.has(node.tag)) {\n out += '\\n' + inner.trim() + '\\n';\n } else {\n out += inner;\n }\n }\n return out.replace(/\\n{3,}/g, '\\n\\n').trim();\n}\n\n/**\n * Extract Markdown from parsed HTML.\n */\nexport function extractMarkdown(nodes: HtmlNode[], baseUrl?: string): string {\n let listDepth = 0;\n let olCounter: number[] = [];\n\n function resolveUrl(href: string): string {\n if (!href || href.startsWith('javascript:') || href.startsWith('#')) return href;\n if (href.startsWith('http://') || href.startsWith('https://') || href.startsWith('//')) return href;\n if (baseUrl) {\n try { return new URL(href, baseUrl).href; } catch {}\n }\n return href;\n }\n\n function walk(nodes: HtmlNode[]): string {\n let out = '';\n for (const node of nodes) {\n if (node.type === 'text') {\n out += node.text?.replace(/\\s+/g, ' ') || '';\n continue;\n }\n if (node.type !== 'element' || !node.tag) continue;\n if (SKIP_TAGS.has(node.tag)) continue;\n\n const tag = node.tag;\n const children = node.children || [];\n const inner = walk(children).trim();\n\n // Headings\n if (HEADING_LEVELS[tag]) {\n out += `\\n\\n${HEADING_LEVELS[tag]} ${inner}\\n\\n`;\n continue;\n }\n\n switch (tag) {\n case 'p': out += `\\n\\n${inner}\\n\\n`; break;\n case 'br': out += '\\n'; break;\n case 'hr': out += '\\n\\n---\\n\\n'; break;\n case 'strong': case 'b': if (inner) out += `**${inner}**`; break;\n case 'em': case 'i': if (inner) out += `*${inner}*`; break;\n case 's': case 'del': case 'strike': if (inner) out += `~~${inner}~~`; break;\n case 'code': if (inner) out += `\\`${inner}\\``; break;\n case 'pre': {\n const lang = children.find(c => c.tag === 'code')?.attributes?.class?.match(/language-(\\w+)/)?.[1] || '';\n out += `\\n\\n\\`\\`\\`${lang}\\n${inner}\\n\\`\\`\\`\\n\\n`;\n break;\n }\n case 'a': {\n const href = resolveUrl(node.attributes?.href || '');\n const text = inner || node.attributes?.['aria-label'] || node.attributes?.title || '';\n if (!text) break;\n if (!href || href === '#' || href.startsWith('javascript:')) { out += text; break; }\n out += `[${text}](${href})`;\n break;\n }\n case 'img': {\n const alt = node.attributes?.alt || '';\n const src = resolveUrl(node.attributes?.src || '');\n if (src) out += ``;\n break;\n }\n case 'ul': listDepth++; olCounter.push(0); out += '\\n' + walk(children); listDepth--; olCounter.pop(); break;\n case 'ol': listDepth++; olCounter.push(0); out += '\\n' + walk(children); listDepth--; olCounter.pop(); break;\n case 'li': {\n const indent = ' '.repeat(Math.max(0, listDepth - 1));\n const isOrdered = olCounter.length > 0 && olCounter[olCounter.length - 1] >= 0;\n if (isOrdered && olCounter.length > 0) olCounter[olCounter.length - 1]++;\n const counter = isOrdered && olCounter.length > 0 ? olCounter[olCounter.length - 1] : 0;\n const bullet = isOrdered ? `${counter}. ` : '- ';\n out += `${indent}${bullet}${inner}\\n`;\n break;\n }\n case 'blockquote': {\n if (inner) out += '\\n\\n' + inner.split('\\n').map(l => `> ${l}`).join('\\n') + '\\n\\n';\n break;\n }\n case 'table': {\n // Collect rows\n const rows = collectTableRows(children);\n if (rows.length > 0) {\n out += '\\n\\n';\n for (let i = 0; i < rows.length; i++) {\n out += '| ' + rows[i].join(' | ') + ' |\\n';\n if (i === 0) out += '| ' + rows[i].map(() => '---').join(' | ') + ' |\\n';\n }\n out += '\\n';\n }\n break;\n }\n case 'dt': out += `\\n**${inner}**\\n`; break;\n case 'dd': out += `: ${inner}\\n`; break;\n case 'figcaption': out += `\\n*${inner}*\\n`; break;\n case 'summary': out += `**${inner}**\\n\\n`; break;\n default:\n if (BLOCK_TAGS.has(tag)) {\n out += '\\n' + walk(children) + '\\n';\n } else {\n out += walk(children);\n }\n }\n }\n return out;\n }\n\n function collectTableRows(nodes: HtmlNode[]): string[][] {\n const rows: string[][] = [];\n for (const node of nodes) {\n if (node.tag === 'tr') {\n const cells: string[] = [];\n for (const cell of node.children || []) {\n if (cell.tag === 'td' || cell.tag === 'th') {\n cells.push(walk(cell.children || []).trim().replace(/\\|/g, '\\\\|').replace(/\\n/g, ' '));\n }\n }\n if (cells.length > 0) rows.push(cells);\n } else if (node.tag === 'thead' || node.tag === 'tbody' || node.tag === 'tfoot') {\n rows.push(...collectTableRows(node.children || []));\n }\n }\n return rows;\n }\n\n const raw = walk(nodes);\n return raw.replace(/\\n{3,}/g, '\\n\\n').trim();\n}\n\n/**\n * Extract a snapshot with interactive element indices.\n */\nexport function extractSnapshot(nodes: HtmlNode[]): string {\n let idx = 0;\n const ATTR_WHITELIST = ['type', 'role', 'aria-label', 'placeholder', 'href', 'value', 'name', 'alt'];\n\n function isInteractive(node: HtmlNode): boolean {\n if (!node.tag) return false;\n if (INTERACTIVE_TAGS.has(node.tag)) return true;\n const role = node.attributes?.role;\n if (role && INTERACTIVE_ROLES.has(role)) return true;\n if (node.attributes?.contenteditable === 'true') return true;\n if (node.attributes?.tabindex && parseInt(node.attributes.tabindex) >= 0) return true;\n return false;\n }\n\n function getAttrs(node: HtmlNode): string {\n const parts: string[] = [];\n for (const name of ATTR_WHITELIST) {\n const v = node.attributes?.[name];\n if (v) parts.push(`${name}=${v.slice(0, 60)}`);\n }\n return parts.length ? ' ' + parts.join(' ') : '';\n }\n\n function walk(nodes: HtmlNode[], depth: number): string {\n let out = '';\n for (const node of nodes) {\n if (node.type === 'text') {\n const t = node.text?.trim();\n if (t) out += ' '.repeat(depth) + t.slice(0, 150) + '\\n';\n continue;\n }\n if (node.type !== 'element' || !node.tag) continue;\n if (SKIP_TAGS.has(node.tag)) continue;\n\n const indent = ' '.repeat(depth);\n const inter = isInteractive(node);\n const prefix = inter ? `[${idx++}]` : '';\n const attrs = getAttrs(node);\n\n // Leaf text\n const leafText = node.children?.length === 1 && node.children[0].type === 'text'\n ? (node.children[0].text?.trim().slice(0, 150) || '') : '';\n\n if (inter || leafText || !node.children?.length) {\n if (leafText) {\n out += `${indent}${prefix}<${node.tag}${attrs}>${leafText}</${node.tag}>\\n`;\n } else {\n out += `${indent}${prefix}<${node.tag}${attrs}>\\n`;\n if (node.children) out += walk(node.children, depth + 1);\n }\n } else {\n if (node.children) out += walk(node.children, depth);\n }\n }\n return out;\n }\n\n return walk(nodes, 0);\n}\n\n/**\n * Extract links from parsed HTML.\n */\nexport function extractLinks(nodes: HtmlNode[], baseUrl?: string): { text: string; href: string }[] {\n const links: { text: string; href: string }[] = [];\n\n function walk(nodes: HtmlNode[]) {\n for (const node of nodes) {\n if (node.type === 'element' && node.tag === 'a' && node.attributes?.href) {\n let href = node.attributes.href;\n if (baseUrl && !href.startsWith('http')) {\n try { href = new URL(href, baseUrl).href; } catch {}\n }\n const text = extractText(node.children || []).trim();\n if (text && href && !href.startsWith('javascript:')) {\n links.push({ text: text.slice(0, 200), href });\n }\n }\n if (node.children) walk(node.children);\n }\n }\n\n walk(nodes);\n return links;\n}\n\n// ── Main fetch function (replaces Lightpanda binary) ──\n\nexport interface LobsterFetchResult {\n url: string;\n finalUrl: string;\n status: number;\n title: string;\n content: string;\n links?: { text: string; href: string }[];\n duration: number;\n}\n\n/**\n * Fetch a URL and extract content — no Chrome, no external binary.\n * Uses our in-house HTML parser.\n */\nexport async function lobsterFetch(\n url: string,\n options?: {\n dump?: 'markdown' | 'text' | 'snapshot' | 'html' | 'links';\n timeout?: number;\n headers?: Record<string, string>;\n followRedirects?: boolean;\n },\n): Promise<LobsterFetchResult> {\n const timeout = options?.timeout || 30000;\n const dump = options?.dump || 'markdown';\n\n const start = Date.now();\n\n const resp = await fetch(url, {\n headers: {\n 'User-Agent': 'LobsterCLI/0.1 (+https://github.com/iexcalibur/lobster-cli)',\n 'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8',\n 'Accept-Language': 'en-US,en;q=0.5',\n ...(options?.headers || {}),\n },\n redirect: options?.followRedirects !== false ? 'follow' : 'manual',\n signal: AbortSignal.timeout(timeout),\n });\n\n if (!resp.ok) {\n throw new Error(`HTTP ${resp.status} ${resp.statusText}`);\n }\n\n const html = await resp.text();\n const duration = Date.now() - start;\n const finalUrl = resp.url || url;\n\n // Parse\n const nodes = parseHtml(html);\n\n // Extract title\n let title = '';\n function findTitle(nodes: HtmlNode[]): void {\n for (const node of nodes) {\n if (node.tag === 'title' && node.children?.[0]?.text) {\n title = node.children[0].text.trim();\n return;\n }\n if (node.children) findTitle(node.children);\n }\n }\n findTitle(nodes);\n\n // Extract content based on dump format\n let content: string;\n let links: { text: string; href: string }[] | undefined;\n\n switch (dump) {\n case 'markdown':\n content = extractMarkdown(nodes, finalUrl);\n break;\n case 'text':\n content = extractText(nodes);\n break;\n case 'snapshot':\n content = extractSnapshot(nodes);\n break;\n case 'html':\n content = html;\n break;\n case 'links':\n links = extractLinks(nodes, finalUrl);\n content = links.map((l, i) => `${i + 1}. [${l.text}](${l.href})`).join('\\n');\n break;\n default:\n content = extractMarkdown(nodes, finalUrl);\n }\n\n return { url, finalUrl, status: resp.status, title, content, links, duration };\n}\n\n// Keep these exports for backward compatibility\nexport function isLightpandaAvailable(): boolean { return true; } // Our engine is always available\nexport function getInstallInstructions(): string { return 'Built-in — no installation needed'; }\n"],"mappings":"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;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;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;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;;ACzBxB,IAAM,uBAAuB;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;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;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;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;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;;;ACE7B,IAAM,kBAAkB;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;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;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;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;AAA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","names":["el","nodes"]}
|
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
// src/browser/manager.ts
|
|
2
|
+
import puppeteer from "puppeteer-core";
|
|
3
|
+
import { existsSync } from "fs";
|
|
4
|
+
|
|
5
|
+
// src/utils/logger.ts
|
|
6
|
+
import chalk from "chalk";
|
|
7
|
+
var log = {
|
|
8
|
+
info: (msg) => console.log(chalk.blue("\u2139"), msg),
|
|
9
|
+
success: (msg) => console.log(chalk.green("\u2713"), msg),
|
|
10
|
+
warn: (msg) => console.log(chalk.yellow("\u26A0"), msg),
|
|
11
|
+
error: (msg) => console.error(chalk.red("\u2717"), msg),
|
|
12
|
+
debug: (msg) => {
|
|
13
|
+
if (process.env.LOBSTER_DEBUG) console.log(chalk.gray("\u22EF"), msg);
|
|
14
|
+
},
|
|
15
|
+
step: (n, msg) => console.log(chalk.cyan(`[${n}]`), msg),
|
|
16
|
+
dim: (msg) => console.log(chalk.dim(msg))
|
|
17
|
+
};
|
|
18
|
+
|
|
19
|
+
// src/browser/manager.ts
|
|
20
|
+
var BrowserManager = class {
|
|
21
|
+
browser = null;
|
|
22
|
+
config;
|
|
23
|
+
constructor(config = {}) {
|
|
24
|
+
this.config = config;
|
|
25
|
+
}
|
|
26
|
+
async connect() {
|
|
27
|
+
if (this.browser?.connected) return this.browser;
|
|
28
|
+
if (this.config.cdpEndpoint) {
|
|
29
|
+
log.debug(`Connecting to CDP endpoint: ${this.config.cdpEndpoint}`);
|
|
30
|
+
this.browser = await puppeteer.connect({
|
|
31
|
+
browserWSEndpoint: this.config.cdpEndpoint
|
|
32
|
+
});
|
|
33
|
+
return this.browser;
|
|
34
|
+
}
|
|
35
|
+
const executablePath = this.config.executablePath || findChrome();
|
|
36
|
+
if (!executablePath) {
|
|
37
|
+
throw new Error(
|
|
38
|
+
"Chrome/Chromium not found. Set LOBSTER_BROWSER_PATH or config browser.executablePath"
|
|
39
|
+
);
|
|
40
|
+
}
|
|
41
|
+
log.debug(`Launching Chrome: ${executablePath}`);
|
|
42
|
+
this.browser = await puppeteer.launch({
|
|
43
|
+
executablePath,
|
|
44
|
+
headless: this.config.headless ?? true,
|
|
45
|
+
args: [
|
|
46
|
+
"--no-sandbox",
|
|
47
|
+
"--disable-setuid-sandbox",
|
|
48
|
+
"--disable-dev-shm-usage",
|
|
49
|
+
"--disable-gpu"
|
|
50
|
+
]
|
|
51
|
+
});
|
|
52
|
+
return this.browser;
|
|
53
|
+
}
|
|
54
|
+
async newPage() {
|
|
55
|
+
const browser = await this.connect();
|
|
56
|
+
return browser.newPage();
|
|
57
|
+
}
|
|
58
|
+
async close() {
|
|
59
|
+
if (this.browser) {
|
|
60
|
+
await this.browser.close().catch(() => {
|
|
61
|
+
});
|
|
62
|
+
this.browser = null;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
};
|
|
66
|
+
function findChrome() {
|
|
67
|
+
const paths = process.platform === "darwin" ? [
|
|
68
|
+
"/Applications/Google Chrome.app/Contents/MacOS/Google Chrome",
|
|
69
|
+
"/Applications/Chromium.app/Contents/MacOS/Chromium",
|
|
70
|
+
"/Applications/Google Chrome Canary.app/Contents/MacOS/Google Chrome Canary"
|
|
71
|
+
] : process.platform === "win32" ? [
|
|
72
|
+
"C:\\Program Files\\Google\\Chrome\\Application\\chrome.exe",
|
|
73
|
+
"C:\\Program Files (x86)\\Google\\Chrome\\Application\\chrome.exe"
|
|
74
|
+
] : [
|
|
75
|
+
"/usr/bin/google-chrome",
|
|
76
|
+
"/usr/bin/google-chrome-stable",
|
|
77
|
+
"/usr/bin/chromium-browser",
|
|
78
|
+
"/usr/bin/chromium",
|
|
79
|
+
"/snap/bin/chromium"
|
|
80
|
+
];
|
|
81
|
+
return paths.find((p) => existsSync(p));
|
|
82
|
+
}
|
|
83
|
+
export {
|
|
84
|
+
BrowserManager
|
|
85
|
+
};
|
|
86
|
+
//# sourceMappingURL=manager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/browser/manager.ts","../../src/utils/logger.ts"],"sourcesContent":["import puppeteer, { type Browser, type Page } from 'puppeteer-core';\nimport { existsSync } from 'node:fs';\nimport { log } from '../utils/logger.js';\n\nexport interface BrowserManagerConfig {\n executablePath?: string;\n headless?: boolean;\n cdpEndpoint?: string;\n connectTimeout?: number;\n}\n\nexport class BrowserManager {\n private browser: Browser | null = null;\n private config: BrowserManagerConfig;\n\n constructor(config: BrowserManagerConfig = {}) {\n this.config = config;\n }\n\n async connect(): Promise<Browser> {\n if (this.browser?.connected) return this.browser;\n\n if (this.config.cdpEndpoint) {\n log.debug(`Connecting to CDP endpoint: ${this.config.cdpEndpoint}`);\n this.browser = await puppeteer.connect({\n browserWSEndpoint: this.config.cdpEndpoint,\n });\n return this.browser;\n }\n\n const executablePath = this.config.executablePath || findChrome();\n if (!executablePath) {\n throw new Error(\n 'Chrome/Chromium not found. Set LOBSTER_BROWSER_PATH or config browser.executablePath'\n );\n }\n\n log.debug(`Launching Chrome: ${executablePath}`);\n this.browser = await puppeteer.launch({\n executablePath,\n headless: this.config.headless ?? true,\n args: [\n '--no-sandbox',\n '--disable-setuid-sandbox',\n '--disable-dev-shm-usage',\n '--disable-gpu',\n ],\n });\n\n return this.browser;\n }\n\n async newPage(): Promise<Page> {\n const browser = await this.connect();\n return browser.newPage();\n }\n\n async close(): Promise<void> {\n if (this.browser) {\n await this.browser.close().catch(() => {});\n this.browser = null;\n }\n }\n}\n\nfunction findChrome(): string | undefined {\n const paths =\n process.platform === 'darwin'\n ? [\n '/Applications/Google Chrome.app/Contents/MacOS/Google Chrome',\n '/Applications/Chromium.app/Contents/MacOS/Chromium',\n '/Applications/Google Chrome Canary.app/Contents/MacOS/Google Chrome Canary',\n ]\n : process.platform === 'win32'\n ? [\n 'C:\\\\Program Files\\\\Google\\\\Chrome\\\\Application\\\\chrome.exe',\n 'C:\\\\Program Files (x86)\\\\Google\\\\Chrome\\\\Application\\\\chrome.exe',\n ]\n : [\n '/usr/bin/google-chrome',\n '/usr/bin/google-chrome-stable',\n '/usr/bin/chromium-browser',\n '/usr/bin/chromium',\n '/snap/bin/chromium',\n ];\n\n return paths.find((p) => existsSync(p));\n}\n","import chalk from 'chalk';\n\nexport const log = {\n info: (msg: string) => console.log(chalk.blue('ℹ'), msg),\n success: (msg: string) => console.log(chalk.green('✓'), msg),\n warn: (msg: string) => console.log(chalk.yellow('⚠'), msg),\n error: (msg: string) => console.error(chalk.red('✗'), msg),\n debug: (msg: string) => {\n if (process.env.LOBSTER_DEBUG) console.log(chalk.gray('⋯'), msg);\n },\n step: (n: number, msg: string) => console.log(chalk.cyan(`[${n}]`), msg),\n dim: (msg: string) => console.log(chalk.dim(msg)),\n};\n"],"mappings":";AAAA,OAAO,eAA4C;AACnD,SAAS,kBAAkB;;;ACD3B,OAAO,WAAW;AAEX,IAAM,MAAM;AAAA,EACjB,MAAM,CAAC,QAAgB,QAAQ,IAAI,MAAM,KAAK,QAAG,GAAG,GAAG;AAAA,EACvD,SAAS,CAAC,QAAgB,QAAQ,IAAI,MAAM,MAAM,QAAG,GAAG,GAAG;AAAA,EAC3D,MAAM,CAAC,QAAgB,QAAQ,IAAI,MAAM,OAAO,QAAG,GAAG,GAAG;AAAA,EACzD,OAAO,CAAC,QAAgB,QAAQ,MAAM,MAAM,IAAI,QAAG,GAAG,GAAG;AAAA,EACzD,OAAO,CAAC,QAAgB;AACtB,QAAI,QAAQ,IAAI,cAAe,SAAQ,IAAI,MAAM,KAAK,QAAG,GAAG,GAAG;AAAA,EACjE;AAAA,EACA,MAAM,CAAC,GAAW,QAAgB,QAAQ,IAAI,MAAM,KAAK,IAAI,CAAC,GAAG,GAAG,GAAG;AAAA,EACvE,KAAK,CAAC,QAAgB,QAAQ,IAAI,MAAM,IAAI,GAAG,CAAC;AAClD;;;ADDO,IAAM,iBAAN,MAAqB;AAAA,EAClB,UAA0B;AAAA,EAC1B;AAAA,EAER,YAAY,SAA+B,CAAC,GAAG;AAC7C,SAAK,SAAS;AAAA,EAChB;AAAA,EAEA,MAAM,UAA4B;AAChC,QAAI,KAAK,SAAS,UAAW,QAAO,KAAK;AAEzC,QAAI,KAAK,OAAO,aAAa;AAC3B,UAAI,MAAM,+BAA+B,KAAK,OAAO,WAAW,EAAE;AAClE,WAAK,UAAU,MAAM,UAAU,QAAQ;AAAA,QACrC,mBAAmB,KAAK,OAAO;AAAA,MACjC,CAAC;AACD,aAAO,KAAK;AAAA,IACd;AAEA,UAAM,iBAAiB,KAAK,OAAO,kBAAkB,WAAW;AAChE,QAAI,CAAC,gBAAgB;AACnB,YAAM,IAAI;AAAA,QACR;AAAA,MACF;AAAA,IACF;AAEA,QAAI,MAAM,qBAAqB,cAAc,EAAE;AAC/C,SAAK,UAAU,MAAM,UAAU,OAAO;AAAA,MACpC;AAAA,MACA,UAAU,KAAK,OAAO,YAAY;AAAA,MAClC,MAAM;AAAA,QACJ;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,MACF;AAAA,IACF,CAAC;AAED,WAAO,KAAK;AAAA,EACd;AAAA,EAEA,MAAM,UAAyB;AAC7B,UAAM,UAAU,MAAM,KAAK,QAAQ;AACnC,WAAO,QAAQ,QAAQ;AAAA,EACzB;AAAA,EAEA,MAAM,QAAuB;AAC3B,QAAI,KAAK,SAAS;AAChB,YAAM,KAAK,QAAQ,MAAM,EAAE,MAAM,MAAM;AAAA,MAAC,CAAC;AACzC,WAAK,UAAU;AAAA,IACjB;AAAA,EACF;AACF;AAEA,SAAS,aAAiC;AACxC,QAAM,QACJ,QAAQ,aAAa,WACjB;AAAA,IACE;AAAA,IACA;AAAA,IACA;AAAA,EACF,IACA,QAAQ,aAAa,UACnB;AAAA,IACE;AAAA,IACA;AAAA,EACF,IACA;AAAA,IACE;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF;AAER,SAAO,MAAM,KAAK,CAAC,MAAM,WAAW,CAAC,CAAC;AACxC;","names":[]}
|