@fairfox/polly 0.5.2 → 0.6.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.
Files changed (48) hide show
  1. package/dist/cli/polly.js +9 -9
  2. package/dist/cli/polly.js.map +4 -4
  3. package/dist/cli/template-utils.js +3 -3
  4. package/dist/cli/template-utils.js.map +3 -3
  5. package/dist/src/shared/state/app-state.d.ts +8 -0
  6. package/dist/vendor/verify/src/cli.js +78 -71
  7. package/dist/vendor/verify/src/cli.js.map +10 -10
  8. package/dist/vendor/verify/src/public-api.d.ts +41 -0
  9. package/dist/vendor/verify/src/public-api.js +26 -0
  10. package/dist/vendor/verify/src/public-api.js.map +10 -0
  11. package/dist/vendor/visualize/src/cli.js +108 -104
  12. package/dist/vendor/visualize/src/cli.js.map +13 -13
  13. package/package.json +15 -2
  14. package/templates/pwa/build.ts.template +37 -37
  15. package/templates/pwa/server.ts.template +53 -53
  16. package/templates/pwa/src/service-worker.ts.template +131 -135
  17. package/templates/pwa/src/shared-worker.ts.template +114 -109
  18. package/dist/shared/state/app-state.d.ts +0 -8
  19. /package/dist/{background → src/background}/api-client.d.ts +0 -0
  20. /package/dist/{background → src/background}/context-menu.d.ts +0 -0
  21. /package/dist/{background → src/background}/index.d.ts +0 -0
  22. /package/dist/{background → src/background}/log-store.d.ts +0 -0
  23. /package/dist/{background → src/background}/message-router.d.ts +0 -0
  24. /package/dist/{background → src/background}/offscreen-manager.d.ts +0 -0
  25. /package/dist/{index.d.ts → src/index.d.ts} +0 -0
  26. /package/dist/{shared → src/shared}/adapters/chrome/context-menus.chrome.d.ts +0 -0
  27. /package/dist/{shared → src/shared}/adapters/chrome/offscreen.chrome.d.ts +0 -0
  28. /package/dist/{shared → src/shared}/adapters/chrome/runtime.chrome.d.ts +0 -0
  29. /package/dist/{shared → src/shared}/adapters/chrome/storage.chrome.d.ts +0 -0
  30. /package/dist/{shared → src/shared}/adapters/chrome/tabs.chrome.d.ts +0 -0
  31. /package/dist/{shared → src/shared}/adapters/chrome/window.chrome.d.ts +0 -0
  32. /package/dist/{shared → src/shared}/adapters/context-menus.adapter.d.ts +0 -0
  33. /package/dist/{shared → src/shared}/adapters/fetch.adapter.d.ts +0 -0
  34. /package/dist/{shared → src/shared}/adapters/index.d.ts +0 -0
  35. /package/dist/{shared → src/shared}/adapters/logger.adapter.d.ts +0 -0
  36. /package/dist/{shared → src/shared}/adapters/offscreen.adapter.d.ts +0 -0
  37. /package/dist/{shared → src/shared}/adapters/runtime.adapter.d.ts +0 -0
  38. /package/dist/{shared → src/shared}/adapters/storage.adapter.d.ts +0 -0
  39. /package/dist/{shared → src/shared}/adapters/tabs.adapter.d.ts +0 -0
  40. /package/dist/{shared → src/shared}/adapters/window.adapter.d.ts +0 -0
  41. /package/dist/{shared → src/shared}/lib/context-helpers.d.ts +0 -0
  42. /package/dist/{shared → src/shared}/lib/context-specific-helpers.d.ts +0 -0
  43. /package/dist/{shared → src/shared}/lib/errors.d.ts +0 -0
  44. /package/dist/{shared → src/shared}/lib/handler-execution-tracker.d.ts +0 -0
  45. /package/dist/{shared → src/shared}/lib/message-bus.d.ts +0 -0
  46. /package/dist/{shared → src/shared}/lib/state.d.ts +0 -0
  47. /package/dist/{shared → src/shared}/lib/test-helpers.d.ts +0 -0
  48. /package/dist/{shared → src/shared}/types/messages.d.ts +0 -0
package/dist/cli/polly.js CHANGED
@@ -41,7 +41,7 @@ function copyTemplateFiles(sourceDir, targetDir, projectName, relativePath = "")
41
41
  const targetPath = join(targetDir, relativePath, entry.name.replace(".template", ""));
42
42
  if (entry.isDirectory()) {
43
43
  mkdirSync(targetPath, { recursive: true });
44
- copyTemplateFiles(sourceDir, targetDir, projectName, join(relativePath, entry.name));
44
+ copyTemplateFiles(sourcePath, targetDir, projectName, join(relativePath, entry.name));
45
45
  } else if (entry.name.endsWith(".template")) {
46
46
  const content = readFileSync(sourcePath, "utf-8");
47
47
  const processed = processTemplate(content, projectName);
@@ -52,7 +52,7 @@ function copyTemplateFiles(sourceDir, targetDir, projectName, relativePath = "")
52
52
  }
53
53
  }
54
54
  function getTemplateDir(projectType, baseDir) {
55
- return join(baseDir, "..", "templates", projectType);
55
+ return join(baseDir, "../..", "templates", projectType);
56
56
  }
57
57
  function validateProjectName(name) {
58
58
  if (!name || name.trim().length === 0) {
@@ -95,7 +95,7 @@ async function loadConfig() {
95
95
  const config = await import(configPath);
96
96
  return config.default || config;
97
97
  } catch (error) {
98
- console.error(`\u274C Failed to load config: ${configPath}`);
98
+ console.log(`\u274C Failed to load config: ${configPath}`);
99
99
  throw error;
100
100
  }
101
101
  }
@@ -231,7 +231,7 @@ async function check() {
231
231
  if (optional) {
232
232
  continue;
233
233
  }
234
- console.error(`
234
+ console.log(`
235
235
  \x1B[31m\u2717 ${name} failed\x1B[0m
236
236
  `);
237
237
  process.exit(1);
@@ -244,13 +244,13 @@ async function init() {
244
244
  const projectType = typeArg;
245
245
  const validation = validateProjectName(projectName);
246
246
  if (!validation.valid) {
247
- console.error(`\x1B[31m\u2717 ${validation.error}\x1B[0m
247
+ console.log(`\x1B[31m\u2717 ${validation.error}\x1B[0m
248
248
  `);
249
249
  process.exit(1);
250
250
  }
251
251
  const projectPath = `${cwd}/${projectName}`;
252
252
  if (existsSync2(projectPath)) {
253
- console.error(`\x1B[31m\u2717 Directory '${projectName}' already exists\x1B[0m
253
+ console.log(`\x1B[31m\u2717 Directory '${projectName}' already exists\x1B[0m
254
254
  `);
255
255
  process.exit(1);
256
256
  }
@@ -303,17 +303,17 @@ async function main() {
303
303
  help();
304
304
  break;
305
305
  default:
306
- console.error(`\u274C Unknown command: ${command}
306
+ console.log(`\u274C Unknown command: ${command}
307
307
  `);
308
308
  help();
309
309
  process.exit(1);
310
310
  }
311
311
  } catch (error) {
312
- console.error(`
312
+ console.log(`
313
313
  \u274C Command failed:`, error);
314
314
  process.exit(1);
315
315
  }
316
316
  }
317
317
  main();
318
318
 
319
- //# debugId=498624C7345252DB64756E2164756E21
319
+ //# debugId=5CD7097C875BD18664756E2164756E21
@@ -2,10 +2,10 @@
2
2
  "version": 3,
3
3
  "sources": ["../cli/polly.ts", "../cli/template-utils.ts"],
4
4
  "sourcesContent": [
5
- "#!/usr/bin/env bun\n/**\n * Polly CLI\n *\n * Command-line tool for building multi-execution-context applications\n * with reactive state and cross-context messaging.\n *\n * Supports: Chrome extensions, PWAs, Node/Bun/Deno apps with workers\n *\n * Usage:\n * polly init [name] [--type=TYPE] Create a new project\n * polly check Run all checks (typecheck, lint, test, build)\n * polly build [options] Build the project\n * polly dev Build with watch mode\n * polly typecheck Type check your code\n * polly lint [--fix] Lint your code\n * polly format Format your code\n * polly test [args] Run tests (requires bun test)\n * polly verify [args] Run formal verification\n * polly visualize [args] Generate architecture diagrams\n * polly help Show help\n *\n * Project Types (init --type):\n * extension Chrome/Firefox extension (default)\n * pwa Progressive Web App with workers\n * websocket WebSocket server application\n * generic Generic TypeScript project\n *\n * Options:\n * --prod Build for production (minified)\n * --config <path> Path to config file (default: polly.config.ts)\n * --fix Auto-fix lint/format issues\n * --type=TYPE Project type for init command\n */\n\nimport { existsSync } from \"node:fs\";\nimport {\n type ProjectType,\n getTemplateDir,\n scaffoldFromTemplate,\n validateProjectName,\n} from \"./template-utils\";\n\n// Use Bun built-ins instead of Node.js APIs\nconst __dirname = import.meta.dir;\n\nconst command = process.argv[2];\nconst commandArgs = process.argv.slice(3);\nconst cwd = process.cwd();\n\n// Parse arguments\nconst args = {\n prod: process.argv.includes(\"--prod\"),\n config: process.argv.includes(\"--config\")\n ? process.argv[process.argv.indexOf(\"--config\") + 1]\n : undefined,\n};\n\n/**\n * Load user's configuration\n */\nasync function loadConfig() {\n const configPaths = [\n args.config,\n `${cwd}/polly.config.ts`,\n `${cwd}/polly.config.js`,\n `${cwd}/polly.config.mjs`,\n ].filter(Boolean) as string[];\n\n for (const configPath of configPaths) {\n // Use Bun.file().exists() instead of existsSync\n if (await Bun.file(configPath).exists()) {\n try {\n const config = await import(configPath);\n return config.default || config;\n } catch (error) {\n console.error(`❌ Failed to load config: ${configPath}`);\n throw error;\n }\n }\n }\n return {\n srcDir: \"src\",\n distDir: \"dist\",\n manifest: \"manifest.json\",\n };\n}\n\n/**\n * Build command - build the extension\n */\nasync function build() {\n const config = await loadConfig();\n\n // Check if bundled (published) or in monorepo\n const bundledScript = `${__dirname}/../scripts/build-extension.js`;\n const monorepoScript = `${__dirname}/../scripts/build-extension.ts`;\n const buildScriptPath = (await Bun.file(bundledScript).exists()) ? bundledScript : monorepoScript;\n\n // Pass config via environment\n process.env[\"WEB_EXT_SRC\"] = `${cwd}/${config.srcDir || \"src\"}`;\n process.env[\"WEB_EXT_DIST\"] = `${cwd}/${config.distDir || \"dist\"}`;\n process.env[\"WEB_EXT_MANIFEST\"] = `${cwd}/${config.manifest || \"manifest.json\"}`;\n process.env[\"WEB_EXT_CWD\"] = cwd;\n process.env[\"WEB_EXT_PROD\"] = args.prod ? \"true\" : \"false\";\n\n // Run build\n const proc = Bun.spawn([\"bun\", buildScriptPath], {\n cwd,\n stdout: \"inherit\",\n stderr: \"inherit\",\n });\n\n const exitCode = await proc.exited;\n if (exitCode !== 0) {\n process.exit(exitCode);\n }\n}\n\n/**\n * Dev command - build with watch mode\n */\nasync function dev() {\n await build();\n}\n\n/**\n * Verify command - delegate to @fairfox/web-ext-verify\n */\nasync function verify() {\n // Check if bundled (published) or in monorepo\n const bundledCli = `${__dirname}/../vendor/verify/src/cli.js`;\n const monorepoCli = `${__dirname}/../../verify/src/cli.ts`;\n const verifyCli = (await Bun.file(bundledCli).exists()) ? bundledCli : monorepoCli;\n\n const proc = Bun.spawn([\"bun\", verifyCli, ...commandArgs], {\n cwd,\n stdout: \"inherit\",\n stderr: \"inherit\",\n stdin: \"inherit\",\n });\n\n const exitCode = await proc.exited;\n if (exitCode !== 0) {\n throw new Error(`Verification failed with exit code ${exitCode}`);\n }\n}\n\n/**\n * Visualize command - delegate to @fairfox/polly-visualize\n */\nasync function visualize() {\n // Check if bundled (published) or in monorepo\n const bundledCli = `${__dirname}/../vendor/visualize/src/cli.js`;\n const monorepoCli = `${__dirname}/../../visualize/src/cli.ts`;\n const visualizeCli = (await Bun.file(bundledCli).exists()) ? bundledCli : monorepoCli;\n\n const proc = Bun.spawn([\"bun\", visualizeCli, ...commandArgs], {\n cwd,\n stdout: \"inherit\",\n stderr: \"inherit\",\n stdin: \"inherit\",\n });\n\n const exitCode = await proc.exited;\n if (exitCode !== 0) {\n throw new Error(`Visualization failed with exit code ${exitCode}`);\n }\n}\n\n/**\n * Typecheck command - run TypeScript type checking\n */\nasync function typecheck() {\n const proc = Bun.spawn([\"bunx\", \"tsc\", \"--noEmit\"], {\n cwd,\n stdout: \"inherit\",\n stderr: \"inherit\",\n });\n\n const exitCode = await proc.exited;\n if (exitCode !== 0) {\n throw new Error(`Type checking failed with exit code ${exitCode}`);\n }\n}\n\n/**\n * Lint command - run Biome linter\n */\nasync function lint() {\n const fix = commandArgs.includes(\"--fix\");\n const lintArgs = fix ? [\"check\", \"--write\", \".\"] : [\"check\", \".\"];\n\n const proc = Bun.spawn([\"bunx\", \"@biomejs/biome\", ...lintArgs], {\n cwd,\n stdout: \"inherit\",\n stderr: \"inherit\",\n });\n\n const exitCode = await proc.exited;\n if (exitCode !== 0) {\n throw new Error(`Linting failed with exit code ${exitCode}`);\n }\n}\n\n/**\n * Format command - run Biome formatter\n */\nasync function format() {\n const proc = Bun.spawn([\"bunx\", \"@biomejs/biome\", \"format\", \"--write\", \".\"], {\n cwd,\n stdout: \"inherit\",\n stderr: \"inherit\",\n });\n\n const exitCode = await proc.exited;\n if (exitCode !== 0) {\n throw new Error(`Formatting failed with exit code ${exitCode}`);\n }\n}\n\n/**\n * Test command - run Bun tests\n */\nasync function test() {\n const proc = Bun.spawn([\"bun\", \"test\", ...commandArgs], {\n cwd,\n stdout: \"pipe\",\n stderr: \"pipe\",\n stdin: \"inherit\",\n });\n\n const stdout = await new Response(proc.stdout).text();\n const stderr = await new Response(proc.stderr).text();\n\n // Output the results\n if (stdout) process.stdout.write(stdout);\n if (stderr) process.stderr.write(stderr);\n\n const exitCode = await proc.exited;\n\n // Check if no tests were found (not a failure)\n if (stderr.includes(\"0 test files matching\")) {\n return;\n }\n\n if (exitCode !== 0) {\n throw new Error(`Tests failed with exit code ${exitCode}`);\n }\n}\n\n/**\n * Check command - run all quality checks in sequence\n */\nasync function check() {\n const checks = [\n { name: \"Type checking\", fn: typecheck },\n { name: \"Linting\", fn: lint },\n { name: \"Testing\", fn: test },\n { name: \"Building\", fn: build },\n { name: \"Verification\", fn: verify, optional: true },\n { name: \"Visualization\", fn: visualize, optional: true },\n ];\n\n for (const { name, fn, optional } of checks) {\n try {\n await fn();\n } catch (_error) {\n if (optional) {\n continue;\n }\n console.error(`\\n\\x1b[31m✗ ${name} failed\\x1b[0m\\n`);\n process.exit(1);\n }\n }\n}\n\n/**\n * Init command - scaffold a new project\n */\nasync function init() {\n // Parse arguments\n const projectName = commandArgs[0] || \"my-project\";\n const typeArg =\n commandArgs.find((arg) => arg.startsWith(\"--type=\"))?.split(\"=\")[1] || \"extension\";\n const projectType = typeArg as ProjectType;\n\n // Validate project name\n const validation = validateProjectName(projectName);\n if (!validation.valid) {\n console.error(`\\x1b[31m✗ ${validation.error}\\x1b[0m\\n`);\n process.exit(1);\n }\n\n const projectPath = `${cwd}/${projectName}`;\n\n // Check if directory already exists\n if (existsSync(projectPath)) {\n console.error(`\\x1b[31m✗ Directory '${projectName}' already exists\\x1b[0m\\n`);\n process.exit(1);\n }\n\n // Get template directory\n const templateDir = getTemplateDir(projectType, __dirname);\n\n // Scaffold project\n await scaffoldFromTemplate({\n projectName,\n projectPath,\n projectType,\n templateDir,\n });\n}\n\n/**\n * Help command\n */\nfunction help() {\n // Help is shown automatically via commander\n}\n\n/**\n * Main entry point\n */\nasync function main() {\n try {\n switch (command) {\n case \"init\":\n await init();\n break;\n case \"check\":\n await check();\n break;\n case \"build\":\n await build();\n break;\n case \"dev\":\n await dev();\n break;\n case \"typecheck\":\n await typecheck();\n break;\n case \"lint\":\n await lint();\n break;\n case \"format\":\n await format();\n break;\n case \"test\":\n await test();\n break;\n case \"verify\":\n await verify();\n break;\n case \"visualize\":\n await visualize();\n break;\n case \"help\":\n case \"--help\":\n case \"-h\":\n case undefined:\n help();\n break;\n default:\n console.error(`❌ Unknown command: ${command}\\n`);\n help();\n process.exit(1);\n }\n } catch (error) {\n console.error(\"\\n❌ Command failed:\", error);\n process.exit(1);\n }\n}\n\nmain();\n",
6
- "/**\n * Template utilities for project initialization\n */\n\nimport { cpSync, existsSync, mkdirSync, readFileSync, readdirSync, writeFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\n\nexport type ProjectType = \"extension\" | \"pwa\" | \"websocket\" | \"generic\";\n\nexport interface TemplateOptions {\n projectName: string;\n projectPath: string;\n projectType: ProjectType;\n templateDir: string;\n}\n\n/**\n * Process template file and replace placeholders\n */\nexport function processTemplate(content: string, projectName: string): string {\n return content\n .replace(/\\{\\{PROJECT_NAME\\}\\}/g, projectName)\n .replace(/\\{\\{PROJECT_NAME_UPPER\\}\\}/g, projectName.toUpperCase())\n .replace(/\\{\\{PROJECT_NAME_CAMEL\\}\\}/g, toCamelCase(projectName));\n}\n\n/**\n * Convert string to camelCase\n */\nfunction toCamelCase(str: string): string {\n return str\n .replace(/[-_](.)/g, (_, char) => char.toUpperCase())\n .replace(/^(.)/, (char) => char.toLowerCase());\n}\n\n/**\n * Copy template directory to project\n */\nexport async function scaffoldFromTemplate(options: TemplateOptions): Promise<void> {\n const { projectName, projectPath, templateDir } = options;\n\n // Create project directory\n mkdirSync(projectPath, { recursive: true });\n\n // Copy all template files\n copyTemplateFiles(templateDir, projectPath, projectName);\n}\n\n/**\n * Recursively copy template files and process them\n */\nfunction copyTemplateFiles(\n sourceDir: string,\n targetDir: string,\n projectName: string,\n relativePath = \"\"\n): void {\n if (!existsSync(sourceDir)) {\n throw new Error(`Template directory not found: ${sourceDir}`);\n }\n\n const entries = readdirSync(sourceDir, { withFileTypes: true });\n\n for (const entry of entries) {\n const sourcePath = join(sourceDir, entry.name);\n const targetPath = join(targetDir, relativePath, entry.name.replace(\".template\", \"\"));\n\n if (entry.isDirectory()) {\n // Create directory and recurse\n mkdirSync(targetPath, { recursive: true });\n copyTemplateFiles(sourceDir, targetDir, projectName, join(relativePath, entry.name));\n } else if (entry.name.endsWith(\".template\")) {\n // Process template file\n const content = readFileSync(sourcePath, \"utf-8\");\n const processed = processTemplate(content, projectName);\n writeFileSync(targetPath, processed, \"utf-8\");\n } else {\n // Copy binary files as-is\n cpSync(sourcePath, targetPath);\n }\n }\n}\n\n/**\n * Get available project types\n */\nexport function getAvailableTypes(): ProjectType[] {\n return [\"extension\", \"pwa\", \"websocket\", \"generic\"];\n}\n\n/**\n * Get template directory for project type\n */\nexport function getTemplateDir(projectType: ProjectType, baseDir: string): string {\n return join(baseDir, \"..\", \"templates\", projectType);\n}\n\n/**\n * Validate project name\n */\nexport function validateProjectName(name: string): {\n valid: boolean;\n error?: string;\n} {\n if (!name || name.trim().length === 0) {\n return { valid: false, error: \"Project name cannot be empty\" };\n }\n\n if (!/^[a-z0-9-_]+$/i.test(name)) {\n return {\n valid: false,\n error: \"Project name can only contain letters, numbers, hyphens, and underscores\",\n };\n }\n\n if (name.startsWith(\"-\") || name.startsWith(\"_\")) {\n return {\n valid: false,\n error: \"Project name cannot start with a hyphen or underscore\",\n };\n }\n\n return { valid: true };\n}\n"
5
+ "#!/usr/bin/env bun\n/**\n * Polly CLI\n *\n * Command-line tool for building multi-execution-context applications\n * with reactive state and cross-context messaging.\n *\n * Supports: Chrome extensions, PWAs, Node/Bun/Deno apps with workers\n *\n * Usage:\n * polly init [name] [--type=TYPE] Create a new project\n * polly check Run all checks (typecheck, lint, test, build)\n * polly build [options] Build the project\n * polly dev Build with watch mode\n * polly typecheck Type check your code\n * polly lint [--fix] Lint your code\n * polly format Format your code\n * polly test [args] Run tests (requires bun test)\n * polly verify [args] Run formal verification\n * polly visualize [args] Generate architecture diagrams\n * polly help Show help\n *\n * Project Types (init --type):\n * extension Chrome/Firefox extension (default)\n * pwa Progressive Web App with workers\n * websocket WebSocket server application\n * generic Generic TypeScript project\n *\n * Options:\n * --prod Build for production (minified)\n * --config <path> Path to config file (default: polly.config.ts)\n * --fix Auto-fix lint/format issues\n * --type=TYPE Project type for init command\n */\n\nimport { existsSync } from \"node:fs\";\nimport {\n type ProjectType,\n getTemplateDir,\n scaffoldFromTemplate,\n validateProjectName,\n} from \"./template-utils\";\n\n// Use Bun built-ins instead of Node.js APIs\nconst __dirname = import.meta.dir;\n\nconst command = process.argv[2];\nconst commandArgs = process.argv.slice(3);\nconst cwd = process.cwd();\n\n// Parse arguments\nconst args = {\n prod: process.argv.includes(\"--prod\"),\n config: process.argv.includes(\"--config\")\n ? process.argv[process.argv.indexOf(\"--config\") + 1]\n : undefined,\n};\n\n/**\n * Load user's configuration\n */\nasync function loadConfig() {\n const configPaths = [\n args.config,\n `${cwd}/polly.config.ts`,\n `${cwd}/polly.config.js`,\n `${cwd}/polly.config.mjs`,\n ].filter(Boolean) as string[];\n\n for (const configPath of configPaths) {\n // Use Bun.file().exists() instead of existsSync\n if (await Bun.file(configPath).exists()) {\n try {\n const config = await import(configPath);\n return config.default || config;\n } catch (error) {\n console.log(`❌ Failed to load config: ${configPath}`);\n throw error;\n }\n }\n }\n return {\n srcDir: \"src\",\n distDir: \"dist\",\n manifest: \"manifest.json\",\n };\n}\n\n/**\n * Build command - build the extension\n */\nasync function build() {\n const config = await loadConfig();\n\n // Check if bundled (published) or in monorepo\n const bundledScript = `${__dirname}/../scripts/build-extension.js`;\n const monorepoScript = `${__dirname}/../scripts/build-extension.ts`;\n const buildScriptPath = (await Bun.file(bundledScript).exists()) ? bundledScript : monorepoScript;\n\n // Pass config via environment\n process.env[\"WEB_EXT_SRC\"] = `${cwd}/${config.srcDir || \"src\"}`;\n process.env[\"WEB_EXT_DIST\"] = `${cwd}/${config.distDir || \"dist\"}`;\n process.env[\"WEB_EXT_MANIFEST\"] = `${cwd}/${config.manifest || \"manifest.json\"}`;\n process.env[\"WEB_EXT_CWD\"] = cwd;\n process.env[\"WEB_EXT_PROD\"] = args.prod ? \"true\" : \"false\";\n\n // Run build\n const proc = Bun.spawn([\"bun\", buildScriptPath], {\n cwd,\n stdout: \"inherit\",\n stderr: \"inherit\",\n });\n\n const exitCode = await proc.exited;\n if (exitCode !== 0) {\n process.exit(exitCode);\n }\n}\n\n/**\n * Dev command - build with watch mode\n */\nasync function dev() {\n await build();\n}\n\n/**\n * Verify command - delegate to @fairfox/web-ext-verify\n */\nasync function verify() {\n // Check if bundled (published) or in monorepo\n const bundledCli = `${__dirname}/../vendor/verify/src/cli.js`;\n const monorepoCli = `${__dirname}/../../verify/src/cli.ts`;\n const verifyCli = (await Bun.file(bundledCli).exists()) ? bundledCli : monorepoCli;\n\n const proc = Bun.spawn([\"bun\", verifyCli, ...commandArgs], {\n cwd,\n stdout: \"inherit\",\n stderr: \"inherit\",\n stdin: \"inherit\",\n });\n\n const exitCode = await proc.exited;\n if (exitCode !== 0) {\n throw new Error(`Verification failed with exit code ${exitCode}`);\n }\n}\n\n/**\n * Visualize command - delegate to @fairfox/polly-visualize\n */\nasync function visualize() {\n // Check if bundled (published) or in monorepo\n const bundledCli = `${__dirname}/../vendor/visualize/src/cli.js`;\n const monorepoCli = `${__dirname}/../../visualize/src/cli.ts`;\n const visualizeCli = (await Bun.file(bundledCli).exists()) ? bundledCli : monorepoCli;\n\n const proc = Bun.spawn([\"bun\", visualizeCli, ...commandArgs], {\n cwd,\n stdout: \"inherit\",\n stderr: \"inherit\",\n stdin: \"inherit\",\n });\n\n const exitCode = await proc.exited;\n if (exitCode !== 0) {\n throw new Error(`Visualization failed with exit code ${exitCode}`);\n }\n}\n\n/**\n * Typecheck command - run TypeScript type checking\n */\nasync function typecheck() {\n const proc = Bun.spawn([\"bunx\", \"tsc\", \"--noEmit\"], {\n cwd,\n stdout: \"inherit\",\n stderr: \"inherit\",\n });\n\n const exitCode = await proc.exited;\n if (exitCode !== 0) {\n throw new Error(`Type checking failed with exit code ${exitCode}`);\n }\n}\n\n/**\n * Lint command - run Biome linter\n */\nasync function lint() {\n const fix = commandArgs.includes(\"--fix\");\n const lintArgs = fix ? [\"check\", \"--write\", \".\"] : [\"check\", \".\"];\n\n const proc = Bun.spawn([\"bunx\", \"@biomejs/biome\", ...lintArgs], {\n cwd,\n stdout: \"inherit\",\n stderr: \"inherit\",\n });\n\n const exitCode = await proc.exited;\n if (exitCode !== 0) {\n throw new Error(`Linting failed with exit code ${exitCode}`);\n }\n}\n\n/**\n * Format command - run Biome formatter\n */\nasync function format() {\n const proc = Bun.spawn([\"bunx\", \"@biomejs/biome\", \"format\", \"--write\", \".\"], {\n cwd,\n stdout: \"inherit\",\n stderr: \"inherit\",\n });\n\n const exitCode = await proc.exited;\n if (exitCode !== 0) {\n throw new Error(`Formatting failed with exit code ${exitCode}`);\n }\n}\n\n/**\n * Test command - run Bun tests\n */\nasync function test() {\n const proc = Bun.spawn([\"bun\", \"test\", ...commandArgs], {\n cwd,\n stdout: \"pipe\",\n stderr: \"pipe\",\n stdin: \"inherit\",\n });\n\n const stdout = await new Response(proc.stdout).text();\n const stderr = await new Response(proc.stderr).text();\n\n // Output the results\n if (stdout) process.stdout.write(stdout);\n if (stderr) process.stderr.write(stderr);\n\n const exitCode = await proc.exited;\n\n // Check if no tests were found (not a failure)\n if (stderr.includes(\"0 test files matching\")) {\n return;\n }\n\n if (exitCode !== 0) {\n throw new Error(`Tests failed with exit code ${exitCode}`);\n }\n}\n\n/**\n * Check command - run all quality checks in sequence\n */\nasync function check() {\n const checks = [\n { name: \"Type checking\", fn: typecheck },\n { name: \"Linting\", fn: lint },\n { name: \"Testing\", fn: test },\n { name: \"Building\", fn: build },\n { name: \"Verification\", fn: verify, optional: true },\n { name: \"Visualization\", fn: visualize, optional: true },\n ];\n\n for (const { name, fn, optional } of checks) {\n try {\n await fn();\n } catch (_error) {\n if (optional) {\n continue;\n }\n console.log(`\\n\\x1b[31m✗ ${name} failed\\x1b[0m\\n`);\n process.exit(1);\n }\n }\n}\n\n/**\n * Init command - scaffold a new project\n */\nasync function init() {\n // Parse arguments\n const projectName = commandArgs[0] || \"my-project\";\n const typeArg =\n commandArgs.find((arg) => arg.startsWith(\"--type=\"))?.split(\"=\")[1] || \"extension\";\n const projectType = typeArg as ProjectType;\n\n // Validate project name\n const validation = validateProjectName(projectName);\n if (!validation.valid) {\n console.log(`\\x1b[31m✗ ${validation.error}\\x1b[0m\\n`);\n process.exit(1);\n }\n\n const projectPath = `${cwd}/${projectName}`;\n\n // Check if directory already exists\n if (existsSync(projectPath)) {\n console.log(`\\x1b[31m✗ Directory '${projectName}' already exists\\x1b[0m\\n`);\n process.exit(1);\n }\n\n // Get template directory\n const templateDir = getTemplateDir(projectType, __dirname);\n\n // Scaffold project\n await scaffoldFromTemplate({\n projectName,\n projectPath,\n projectType,\n templateDir,\n });\n}\n\n/**\n * Help command\n */\nfunction help() {\n // Help is shown automatically via commander\n}\n\n/**\n * Main entry point\n */\nasync function main() {\n try {\n switch (command) {\n case \"init\":\n await init();\n break;\n case \"check\":\n await check();\n break;\n case \"build\":\n await build();\n break;\n case \"dev\":\n await dev();\n break;\n case \"typecheck\":\n await typecheck();\n break;\n case \"lint\":\n await lint();\n break;\n case \"format\":\n await format();\n break;\n case \"test\":\n await test();\n break;\n case \"verify\":\n await verify();\n break;\n case \"visualize\":\n await visualize();\n break;\n case \"help\":\n case \"--help\":\n case \"-h\":\n case undefined:\n help();\n break;\n default:\n console.log(`❌ Unknown command: ${command}\\n`);\n help();\n process.exit(1);\n }\n } catch (error) {\n console.log(\"\\n❌ Command failed:\", error);\n process.exit(1);\n }\n}\n\nmain();\n",
6
+ "/**\n * Template utilities for project initialization\n */\n\nimport { cpSync, existsSync, mkdirSync, readFileSync, readdirSync, writeFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\n\nexport type ProjectType = \"extension\" | \"pwa\" | \"websocket\" | \"generic\";\n\nexport interface TemplateOptions {\n projectName: string;\n projectPath: string;\n projectType: ProjectType;\n templateDir: string;\n}\n\n/**\n * Process template file and replace placeholders\n */\nexport function processTemplate(content: string, projectName: string): string {\n return content\n .replace(/\\{\\{PROJECT_NAME\\}\\}/g, projectName)\n .replace(/\\{\\{PROJECT_NAME_UPPER\\}\\}/g, projectName.toUpperCase())\n .replace(/\\{\\{PROJECT_NAME_CAMEL\\}\\}/g, toCamelCase(projectName));\n}\n\n/**\n * Convert string to camelCase\n */\nfunction toCamelCase(str: string): string {\n return str\n .replace(/[-_](.)/g, (_, char) => char.toUpperCase())\n .replace(/^(.)/, (char) => char.toLowerCase());\n}\n\n/**\n * Copy template directory to project\n */\nexport async function scaffoldFromTemplate(options: TemplateOptions): Promise<void> {\n const { projectName, projectPath, templateDir } = options;\n\n // Create project directory\n mkdirSync(projectPath, { recursive: true });\n\n // Copy all template files\n copyTemplateFiles(templateDir, projectPath, projectName);\n}\n\n/**\n * Recursively copy template files and process them\n */\nfunction copyTemplateFiles(\n sourceDir: string,\n targetDir: string,\n projectName: string,\n relativePath = \"\"\n): void {\n if (!existsSync(sourceDir)) {\n throw new Error(`Template directory not found: ${sourceDir}`);\n }\n\n const entries = readdirSync(sourceDir, { withFileTypes: true });\n\n for (const entry of entries) {\n const sourcePath = join(sourceDir, entry.name);\n const targetPath = join(targetDir, relativePath, entry.name.replace(\".template\", \"\"));\n\n if (entry.isDirectory()) {\n // Create directory and recurse\n mkdirSync(targetPath, { recursive: true });\n copyTemplateFiles(sourcePath, targetDir, projectName, join(relativePath, entry.name));\n } else if (entry.name.endsWith(\".template\")) {\n // Process template file\n const content = readFileSync(sourcePath, \"utf-8\");\n const processed = processTemplate(content, projectName);\n writeFileSync(targetPath, processed, \"utf-8\");\n } else {\n // Copy binary files as-is\n cpSync(sourcePath, targetPath);\n }\n }\n}\n\n/**\n * Get available project types\n */\nexport function getAvailableTypes(): ProjectType[] {\n return [\"extension\", \"pwa\", \"websocket\", \"generic\"];\n}\n\n/**\n * Get template directory for project type\n */\nexport function getTemplateDir(projectType: ProjectType, baseDir: string): string {\n return join(baseDir, \"../..\", \"templates\", projectType);\n}\n\n/**\n * Validate project name\n */\nexport function validateProjectName(name: string): {\n valid: boolean;\n error?: string;\n} {\n if (!name || name.trim().length === 0) {\n return { valid: false, error: \"Project name cannot be empty\" };\n }\n\n if (!/^[a-z0-9-_]+$/i.test(name)) {\n return {\n valid: false,\n error: \"Project name can only contain letters, numbers, hyphens, and underscores\",\n };\n }\n\n if (name.startsWith(\"-\") || name.startsWith(\"_\")) {\n return {\n valid: false,\n error: \"Project name cannot start with a hyphen or underscore\",\n };\n }\n\n return { valid: true };\n}\n"
7
7
  ],
8
- "mappings": "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",
9
- "debugId": "498624C7345252DB64756E2164756E21",
8
+ "mappings": "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",
9
+ "debugId": "5CD7097C875BD18664756E2164756E21",
10
10
  "names": []
11
11
  }
@@ -36,7 +36,7 @@ function copyTemplateFiles(sourceDir, targetDir, projectName, relativePath = "")
36
36
  const targetPath = join(targetDir, relativePath, entry.name.replace(".template", ""));
37
37
  if (entry.isDirectory()) {
38
38
  mkdirSync(targetPath, { recursive: true });
39
- copyTemplateFiles(sourceDir, targetDir, projectName, join(relativePath, entry.name));
39
+ copyTemplateFiles(sourcePath, targetDir, projectName, join(relativePath, entry.name));
40
40
  } else if (entry.name.endsWith(".template")) {
41
41
  const content = readFileSync(sourcePath, "utf-8");
42
42
  const processed = processTemplate(content, projectName);
@@ -50,7 +50,7 @@ function getAvailableTypes() {
50
50
  return ["extension", "pwa", "websocket", "generic"];
51
51
  }
52
52
  function getTemplateDir(projectType, baseDir) {
53
- return join(baseDir, "..", "templates", projectType);
53
+ return join(baseDir, "../..", "templates", projectType);
54
54
  }
55
55
  function validateProjectName(name) {
56
56
  if (!name || name.trim().length === 0) {
@@ -78,4 +78,4 @@ export {
78
78
  getAvailableTypes
79
79
  };
80
80
 
81
- //# debugId=88205CACB51C961664756E2164756E21
81
+ //# debugId=7B4FDBF01642B3D364756E2164756E21
@@ -2,9 +2,9 @@
2
2
  "version": 3,
3
3
  "sources": ["../cli/template-utils.ts"],
4
4
  "sourcesContent": [
5
- "/**\n * Template utilities for project initialization\n */\n\nimport { cpSync, existsSync, mkdirSync, readFileSync, readdirSync, writeFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\n\nexport type ProjectType = \"extension\" | \"pwa\" | \"websocket\" | \"generic\";\n\nexport interface TemplateOptions {\n projectName: string;\n projectPath: string;\n projectType: ProjectType;\n templateDir: string;\n}\n\n/**\n * Process template file and replace placeholders\n */\nexport function processTemplate(content: string, projectName: string): string {\n return content\n .replace(/\\{\\{PROJECT_NAME\\}\\}/g, projectName)\n .replace(/\\{\\{PROJECT_NAME_UPPER\\}\\}/g, projectName.toUpperCase())\n .replace(/\\{\\{PROJECT_NAME_CAMEL\\}\\}/g, toCamelCase(projectName));\n}\n\n/**\n * Convert string to camelCase\n */\nfunction toCamelCase(str: string): string {\n return str\n .replace(/[-_](.)/g, (_, char) => char.toUpperCase())\n .replace(/^(.)/, (char) => char.toLowerCase());\n}\n\n/**\n * Copy template directory to project\n */\nexport async function scaffoldFromTemplate(options: TemplateOptions): Promise<void> {\n const { projectName, projectPath, templateDir } = options;\n\n // Create project directory\n mkdirSync(projectPath, { recursive: true });\n\n // Copy all template files\n copyTemplateFiles(templateDir, projectPath, projectName);\n}\n\n/**\n * Recursively copy template files and process them\n */\nfunction copyTemplateFiles(\n sourceDir: string,\n targetDir: string,\n projectName: string,\n relativePath = \"\"\n): void {\n if (!existsSync(sourceDir)) {\n throw new Error(`Template directory not found: ${sourceDir}`);\n }\n\n const entries = readdirSync(sourceDir, { withFileTypes: true });\n\n for (const entry of entries) {\n const sourcePath = join(sourceDir, entry.name);\n const targetPath = join(targetDir, relativePath, entry.name.replace(\".template\", \"\"));\n\n if (entry.isDirectory()) {\n // Create directory and recurse\n mkdirSync(targetPath, { recursive: true });\n copyTemplateFiles(sourceDir, targetDir, projectName, join(relativePath, entry.name));\n } else if (entry.name.endsWith(\".template\")) {\n // Process template file\n const content = readFileSync(sourcePath, \"utf-8\");\n const processed = processTemplate(content, projectName);\n writeFileSync(targetPath, processed, \"utf-8\");\n } else {\n // Copy binary files as-is\n cpSync(sourcePath, targetPath);\n }\n }\n}\n\n/**\n * Get available project types\n */\nexport function getAvailableTypes(): ProjectType[] {\n return [\"extension\", \"pwa\", \"websocket\", \"generic\"];\n}\n\n/**\n * Get template directory for project type\n */\nexport function getTemplateDir(projectType: ProjectType, baseDir: string): string {\n return join(baseDir, \"..\", \"templates\", projectType);\n}\n\n/**\n * Validate project name\n */\nexport function validateProjectName(name: string): {\n valid: boolean;\n error?: string;\n} {\n if (!name || name.trim().length === 0) {\n return { valid: false, error: \"Project name cannot be empty\" };\n }\n\n if (!/^[a-z0-9-_]+$/i.test(name)) {\n return {\n valid: false,\n error: \"Project name can only contain letters, numbers, hyphens, and underscores\",\n };\n }\n\n if (name.startsWith(\"-\") || name.startsWith(\"_\")) {\n return {\n valid: false,\n error: \"Project name cannot start with a hyphen or underscore\",\n };\n }\n\n return { valid: true };\n}\n"
5
+ "/**\n * Template utilities for project initialization\n */\n\nimport { cpSync, existsSync, mkdirSync, readFileSync, readdirSync, writeFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\n\nexport type ProjectType = \"extension\" | \"pwa\" | \"websocket\" | \"generic\";\n\nexport interface TemplateOptions {\n projectName: string;\n projectPath: string;\n projectType: ProjectType;\n templateDir: string;\n}\n\n/**\n * Process template file and replace placeholders\n */\nexport function processTemplate(content: string, projectName: string): string {\n return content\n .replace(/\\{\\{PROJECT_NAME\\}\\}/g, projectName)\n .replace(/\\{\\{PROJECT_NAME_UPPER\\}\\}/g, projectName.toUpperCase())\n .replace(/\\{\\{PROJECT_NAME_CAMEL\\}\\}/g, toCamelCase(projectName));\n}\n\n/**\n * Convert string to camelCase\n */\nfunction toCamelCase(str: string): string {\n return str\n .replace(/[-_](.)/g, (_, char) => char.toUpperCase())\n .replace(/^(.)/, (char) => char.toLowerCase());\n}\n\n/**\n * Copy template directory to project\n */\nexport async function scaffoldFromTemplate(options: TemplateOptions): Promise<void> {\n const { projectName, projectPath, templateDir } = options;\n\n // Create project directory\n mkdirSync(projectPath, { recursive: true });\n\n // Copy all template files\n copyTemplateFiles(templateDir, projectPath, projectName);\n}\n\n/**\n * Recursively copy template files and process them\n */\nfunction copyTemplateFiles(\n sourceDir: string,\n targetDir: string,\n projectName: string,\n relativePath = \"\"\n): void {\n if (!existsSync(sourceDir)) {\n throw new Error(`Template directory not found: ${sourceDir}`);\n }\n\n const entries = readdirSync(sourceDir, { withFileTypes: true });\n\n for (const entry of entries) {\n const sourcePath = join(sourceDir, entry.name);\n const targetPath = join(targetDir, relativePath, entry.name.replace(\".template\", \"\"));\n\n if (entry.isDirectory()) {\n // Create directory and recurse\n mkdirSync(targetPath, { recursive: true });\n copyTemplateFiles(sourcePath, targetDir, projectName, join(relativePath, entry.name));\n } else if (entry.name.endsWith(\".template\")) {\n // Process template file\n const content = readFileSync(sourcePath, \"utf-8\");\n const processed = processTemplate(content, projectName);\n writeFileSync(targetPath, processed, \"utf-8\");\n } else {\n // Copy binary files as-is\n cpSync(sourcePath, targetPath);\n }\n }\n}\n\n/**\n * Get available project types\n */\nexport function getAvailableTypes(): ProjectType[] {\n return [\"extension\", \"pwa\", \"websocket\", \"generic\"];\n}\n\n/**\n * Get template directory for project type\n */\nexport function getTemplateDir(projectType: ProjectType, baseDir: string): string {\n return join(baseDir, \"../..\", \"templates\", projectType);\n}\n\n/**\n * Validate project name\n */\nexport function validateProjectName(name: string): {\n valid: boolean;\n error?: string;\n} {\n if (!name || name.trim().length === 0) {\n return { valid: false, error: \"Project name cannot be empty\" };\n }\n\n if (!/^[a-z0-9-_]+$/i.test(name)) {\n return {\n valid: false,\n error: \"Project name can only contain letters, numbers, hyphens, and underscores\",\n };\n }\n\n if (name.startsWith(\"-\") || name.startsWith(\"_\")) {\n return {\n valid: false,\n error: \"Project name cannot start with a hyphen or underscore\",\n };\n }\n\n return { valid: true };\n}\n"
6
6
  ],
7
- "mappings": ";;;;;;;;;;;;;;;AAIA;AACA;AAcO,SAAS,eAAe,CAAC,SAAiB,aAA6B;AAAA,EAC5E,OAAO,QACJ,QAAQ,yBAAyB,WAAW,EAC5C,QAAQ,+BAA+B,YAAY,YAAY,CAAC,EAChE,QAAQ,+BAA+B,YAAY,WAAW,CAAC;AAAA;AAMpE,SAAS,WAAW,CAAC,KAAqB;AAAA,EACxC,OAAO,IACJ,QAAQ,YAAY,CAAC,GAAG,SAAS,KAAK,YAAY,CAAC,EACnD,QAAQ,QAAQ,CAAC,SAAS,KAAK,YAAY,CAAC;AAAA;AAMjD,eAAsB,oBAAoB,CAAC,SAAyC;AAAA,EAClF,QAAQ,aAAa,aAAa,gBAAgB;AAAA,EAGlD,UAAU,aAAa,EAAE,WAAW,KAAK,CAAC;AAAA,EAG1C,kBAAkB,aAAa,aAAa,WAAW;AAAA;AAMzD,SAAS,iBAAiB,CACxB,WACA,WACA,aACA,eAAe,IACT;AAAA,EACN,IAAI,CAAC,WAAW,SAAS,GAAG;AAAA,IAC1B,MAAM,IAAI,MAAM,iCAAiC,WAAW;AAAA,EAC9D;AAAA,EAEA,MAAM,UAAU,YAAY,WAAW,EAAE,eAAe,KAAK,CAAC;AAAA,EAE9D,WAAW,SAAS,SAAS;AAAA,IAC3B,MAAM,aAAa,KAAK,WAAW,MAAM,IAAI;AAAA,IAC7C,MAAM,aAAa,KAAK,WAAW,cAAc,MAAM,KAAK,QAAQ,aAAa,EAAE,CAAC;AAAA,IAEpF,IAAI,MAAM,YAAY,GAAG;AAAA,MAEvB,UAAU,YAAY,EAAE,WAAW,KAAK,CAAC;AAAA,MACzC,kBAAkB,WAAW,WAAW,aAAa,KAAK,cAAc,MAAM,IAAI,CAAC;AAAA,IACrF,EAAO,SAAI,MAAM,KAAK,SAAS,WAAW,GAAG;AAAA,MAE3C,MAAM,UAAU,aAAa,YAAY,OAAO;AAAA,MAChD,MAAM,YAAY,gBAAgB,SAAS,WAAW;AAAA,MACtD,cAAc,YAAY,WAAW,OAAO;AAAA,IAC9C,EAAO;AAAA,MAEL,OAAO,YAAY,UAAU;AAAA;AAAA,EAEjC;AAAA;AAMK,SAAS,iBAAiB,GAAkB;AAAA,EACjD,OAAO,CAAC,aAAa,OAAO,aAAa,SAAS;AAAA;AAM7C,SAAS,cAAc,CAAC,aAA0B,SAAyB;AAAA,EAChF,OAAO,KAAK,SAAS,MAAM,aAAa,WAAW;AAAA;AAM9C,SAAS,mBAAmB,CAAC,MAGlC;AAAA,EACA,IAAI,CAAC,QAAQ,KAAK,KAAK,EAAE,WAAW,GAAG;AAAA,IACrC,OAAO,EAAE,OAAO,OAAO,OAAO,+BAA+B;AAAA,EAC/D;AAAA,EAEA,IAAI,CAAC,iBAAiB,KAAK,IAAI,GAAG;AAAA,IAChC,OAAO;AAAA,MACL,OAAO;AAAA,MACP,OAAO;AAAA,IACT;AAAA,EACF;AAAA,EAEA,IAAI,KAAK,WAAW,GAAG,KAAK,KAAK,WAAW,GAAG,GAAG;AAAA,IAChD,OAAO;AAAA,MACL,OAAO;AAAA,MACP,OAAO;AAAA,IACT;AAAA,EACF;AAAA,EAEA,OAAO,EAAE,OAAO,KAAK;AAAA;",
8
- "debugId": "88205CACB51C961664756E2164756E21",
7
+ "mappings": ";;;;;;;;;;;;;;;AAIA;AACA;AAcO,SAAS,eAAe,CAAC,SAAiB,aAA6B;AAAA,EAC5E,OAAO,QACJ,QAAQ,yBAAyB,WAAW,EAC5C,QAAQ,+BAA+B,YAAY,YAAY,CAAC,EAChE,QAAQ,+BAA+B,YAAY,WAAW,CAAC;AAAA;AAMpE,SAAS,WAAW,CAAC,KAAqB;AAAA,EACxC,OAAO,IACJ,QAAQ,YAAY,CAAC,GAAG,SAAS,KAAK,YAAY,CAAC,EACnD,QAAQ,QAAQ,CAAC,SAAS,KAAK,YAAY,CAAC;AAAA;AAMjD,eAAsB,oBAAoB,CAAC,SAAyC;AAAA,EAClF,QAAQ,aAAa,aAAa,gBAAgB;AAAA,EAGlD,UAAU,aAAa,EAAE,WAAW,KAAK,CAAC;AAAA,EAG1C,kBAAkB,aAAa,aAAa,WAAW;AAAA;AAMzD,SAAS,iBAAiB,CACxB,WACA,WACA,aACA,eAAe,IACT;AAAA,EACN,IAAI,CAAC,WAAW,SAAS,GAAG;AAAA,IAC1B,MAAM,IAAI,MAAM,iCAAiC,WAAW;AAAA,EAC9D;AAAA,EAEA,MAAM,UAAU,YAAY,WAAW,EAAE,eAAe,KAAK,CAAC;AAAA,EAE9D,WAAW,SAAS,SAAS;AAAA,IAC3B,MAAM,aAAa,KAAK,WAAW,MAAM,IAAI;AAAA,IAC7C,MAAM,aAAa,KAAK,WAAW,cAAc,MAAM,KAAK,QAAQ,aAAa,EAAE,CAAC;AAAA,IAEpF,IAAI,MAAM,YAAY,GAAG;AAAA,MAEvB,UAAU,YAAY,EAAE,WAAW,KAAK,CAAC;AAAA,MACzC,kBAAkB,YAAY,WAAW,aAAa,KAAK,cAAc,MAAM,IAAI,CAAC;AAAA,IACtF,EAAO,SAAI,MAAM,KAAK,SAAS,WAAW,GAAG;AAAA,MAE3C,MAAM,UAAU,aAAa,YAAY,OAAO;AAAA,MAChD,MAAM,YAAY,gBAAgB,SAAS,WAAW;AAAA,MACtD,cAAc,YAAY,WAAW,OAAO;AAAA,IAC9C,EAAO;AAAA,MAEL,OAAO,YAAY,UAAU;AAAA;AAAA,EAEjC;AAAA;AAMK,SAAS,iBAAiB,GAAkB;AAAA,EACjD,OAAO,CAAC,aAAa,OAAO,aAAa,SAAS;AAAA;AAM7C,SAAS,cAAc,CAAC,aAA0B,SAAyB;AAAA,EAChF,OAAO,KAAK,SAAS,SAAS,aAAa,WAAW;AAAA;AAMjD,SAAS,mBAAmB,CAAC,MAGlC;AAAA,EACA,IAAI,CAAC,QAAQ,KAAK,KAAK,EAAE,WAAW,GAAG;AAAA,IACrC,OAAO,EAAE,OAAO,OAAO,OAAO,+BAA+B;AAAA,EAC/D;AAAA,EAEA,IAAI,CAAC,iBAAiB,KAAK,IAAI,GAAG;AAAA,IAChC,OAAO;AAAA,MACL,OAAO;AAAA,MACP,OAAO;AAAA,IACT;AAAA,EACF;AAAA,EAEA,IAAI,KAAK,WAAW,GAAG,KAAK,KAAK,WAAW,GAAG,GAAG;AAAA,IAChD,OAAO;AAAA,MACL,OAAO;AAAA,MACP,OAAO;AAAA,IACT;AAAA,EACF;AAAA,EAEA,OAAO,EAAE,OAAO,KAAK;AAAA;",
8
+ "debugId": "7B4FDBF01642B3D364756E2164756E21",
9
9
  "names": []
10
10
  }
@@ -0,0 +1,8 @@
1
+ import type { Settings } from "../types/messages";
2
+ export { defaultSettings } from "../types/messages";
3
+ export declare const settings: import("@preact/signals-core").Signal<Settings>;
4
+ export declare const currentTab: import("@preact/signals-core").Signal<number | null>;
5
+ export declare const uiState: import("@preact/signals-core").Signal<{
6
+ sidebarOpen: boolean;
7
+ selectedPanel: "main" | "settings" | "debug";
8
+ }>;