@fairfox/polly 0.14.1 → 0.15.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 (32) hide show
  1. package/dist/src/background/index.js +342 -3
  2. package/dist/src/background/index.js.map +7 -4
  3. package/dist/src/background/message-router.js +342 -3
  4. package/dist/src/background/message-router.js.map +7 -4
  5. package/dist/src/index.js +402 -99
  6. package/dist/src/index.js.map +8 -5
  7. package/dist/src/shared/adapters/index.d.ts +3 -0
  8. package/dist/src/shared/adapters/index.js +356 -4
  9. package/dist/src/shared/adapters/index.js.map +7 -4
  10. package/dist/src/shared/lib/adapter-factory.d.ts +80 -0
  11. package/dist/src/shared/lib/context-helpers.js +342 -3
  12. package/dist/src/shared/lib/context-helpers.js.map +7 -4
  13. package/dist/src/shared/lib/message-bus.js +342 -3
  14. package/dist/src/shared/lib/message-bus.js.map +7 -4
  15. package/dist/src/shared/lib/state.d.ts +5 -1
  16. package/dist/src/shared/lib/state.js +274 -1173
  17. package/dist/src/shared/lib/state.js.map +6 -19
  18. package/dist/src/shared/lib/storage-adapter.d.ts +42 -0
  19. package/dist/src/shared/lib/sync-adapter.d.ts +79 -0
  20. package/dist/src/shared/state/app-state.js +294 -1173
  21. package/dist/src/shared/state/app-state.js.map +6 -18
  22. package/dist/tools/analysis/src/extract/handlers.d.ts +48 -1
  23. package/dist/tools/analysis/src/types/core.d.ts +20 -0
  24. package/dist/tools/teach/src/cli.js +376 -7
  25. package/dist/tools/teach/src/cli.js.map +4 -4
  26. package/dist/tools/teach/src/index.js +232 -7
  27. package/dist/tools/teach/src/index.js.map +3 -3
  28. package/dist/tools/verify/src/cli.js +232 -7
  29. package/dist/tools/verify/src/cli.js.map +3 -3
  30. package/dist/tools/visualize/src/cli.js +232 -7
  31. package/dist/tools/visualize/src/cli.js.map +3 -3
  32. package/package.json +1 -1
@@ -1,25 +1,12 @@
1
1
  {
2
2
  "version": 3,
3
- "sources": ["../src/shared/lib/constraints.ts", "../src/shared/adapters/chrome/context-menus.chrome.ts", "../src/shared/adapters/chrome/offscreen.chrome.ts", "../src/shared/adapters/chrome/runtime.chrome.ts", "../src/shared/adapters/chrome/storage.chrome.ts", "../src/shared/adapters/chrome/tabs.chrome.ts", "../src/shared/adapters/chrome/window.chrome.ts", "../src/shared/adapters/fetch.adapter.ts", "../src/shared/adapters/logger.adapter.ts", "../src/shared/adapters/index.ts", "../src/shared/types/messages.ts", "../src/shared/lib/errors.ts", "../src/shared/lib/context-specific-helpers.ts", "../src/shared/lib/handler-execution-tracker.ts", "../src/shared/lib/message-bus.ts", "../src/shared/lib/state.ts"],
3
+ "sources": ["../src/shared/lib/storage-adapter.ts", "../src/shared/lib/sync-adapter.ts", "../src/shared/lib/state.ts"],
4
4
  "sourcesContent": [
5
- "/**\n * Runtime constraint checking for message handlers\n *\n * Provides runtime enforcement of $constraints() declarations.\n * Constraints can be checked before handler execution to ensure preconditions are met.\n */\n\ntype ConstraintFunction = (state: unknown) => boolean;\n\ntype Constraint = {\n requires?: ConstraintFunction;\n ensures?: ConstraintFunction;\n message?: string;\n};\n\n// Registry maps: stateField -> messageType -> constraint\nconst registry = new Map<string, Map<string, Constraint>>();\n\n/**\n * Register a constraint for runtime checking.\n *\n * @param field - State field this constraint applies to\n * @param messageType - Message type this constraint applies to\n * @param constraint - Constraint definition with requires/ensures predicates\n */\nexport function registerConstraint(\n field: string,\n messageType: string,\n constraint: Constraint\n): void {\n if (!registry.has(field)) {\n registry.set(field, new Map());\n }\n registry.get(field)?.set(messageType, constraint);\n}\n\n/**\n * Register multiple constraints for a state field.\n *\n * @param field - State field these constraints apply to\n * @param constraints - Map of messageType to constraint definitions\n */\nexport function registerConstraints(\n field: string,\n constraints: Record<\n string,\n {\n requires?: ConstraintFunction | string;\n ensures?: ConstraintFunction | string;\n message?: string;\n }\n >\n): void {\n for (const [messageType, constraint] of Object.entries(constraints)) {\n // Only register function-based constraints (strings are for TLA+ generation)\n const runtimeConstraint: Constraint = {\n message: constraint.message,\n };\n\n if (typeof constraint.requires === \"function\") {\n runtimeConstraint.requires = constraint.requires;\n }\n\n if (typeof constraint.ensures === \"function\") {\n runtimeConstraint.ensures = constraint.ensures;\n }\n\n // Only register if there's at least one function-based constraint\n if (runtimeConstraint.requires || runtimeConstraint.ensures) {\n registerConstraint(field, messageType, runtimeConstraint);\n }\n }\n}\n\n/**\n * Execute a constraint predicate and handle errors\n */\nfunction executeConstraint(\n predicate: ConstraintFunction,\n state: unknown,\n messageType: string,\n field: string,\n customMessage?: string,\n constraintType: \"Precondition\" | \"Postcondition\" = \"Precondition\"\n): void {\n try {\n const result = predicate(state);\n if (!result) {\n const message =\n customMessage || `${constraintType} failed for ${messageType} on field ${field}`;\n throw new Error(message);\n }\n } catch (error) {\n // Re-throw if it's already our constraint error\n if (error instanceof Error && error.message.includes(`${constraintType} failed`)) {\n throw error;\n }\n // Wrap other errors with context\n const message =\n customMessage || `${constraintType} check error for ${messageType} on field ${field}`;\n throw new Error(`${message}: ${error}`);\n }\n}\n\n/**\n * Check preconditions for a message type before handler execution.\n *\n * @param messageType - The message type being handled\n * @param state - Current state object to check against\n * @throws Error if any precondition fails\n */\nexport function checkPreconditions(messageType: string, state: unknown): void {\n for (const [field, constraints] of registry) {\n const constraint = constraints.get(messageType);\n if (constraint?.requires) {\n executeConstraint(\n constraint.requires,\n state,\n messageType,\n field,\n constraint.message,\n \"Precondition\"\n );\n }\n }\n}\n\n/**\n * Check postconditions for a message type after handler execution.\n *\n * @param messageType - The message type that was handled\n * @param state - Current state object to check against\n * @throws Error if any postcondition fails\n */\nexport function checkPostconditions(messageType: string, state: unknown): void {\n for (const [field, constraints] of registry) {\n const constraint = constraints.get(messageType);\n if (constraint?.ensures) {\n executeConstraint(\n constraint.ensures,\n state,\n messageType,\n field,\n constraint.message,\n \"Postcondition\"\n );\n }\n }\n}\n\n/**\n * Clear all registered constraints (useful for testing).\n */\nexport function clearConstraints(): void {\n registry.clear();\n}\n\n/**\n * Get all registered constraints (for debugging).\n */\nexport function getRegisteredConstraints(): Map<string, Map<string, Constraint>> {\n return new Map(registry);\n}\n\n/**\n * Check if runtime constraint checking is enabled.\n * Controlled via POLLY_RUNTIME_CONSTRAINTS environment variable.\n */\nexport function isRuntimeConstraintsEnabled(): boolean {\n // Check environment variable\n if (typeof process !== \"undefined\" && process.env) {\n return process.env[\"POLLY_RUNTIME_CONSTRAINTS\"] === \"true\";\n }\n\n // Check for Bun\n if (typeof Bun !== \"undefined\" && Bun.env) {\n return Bun.env[\"POLLY_RUNTIME_CONSTRAINTS\"] === \"true\";\n }\n\n // Default: disabled\n return false;\n}\n",
6
- "// Chrome context menus adapter implementation\n\nimport type { ContextMenusAdapter } from \"../context-menus.adapter\";\n\nexport class ChromeContextMenusAdapter implements ContextMenusAdapter {\n async create(createProperties: chrome.contextMenus.CreateProperties): Promise<void> {\n return new Promise((resolve, reject) => {\n chrome.contextMenus.create(createProperties, () => {\n if (chrome.runtime.lastError) {\n reject(new Error(chrome.runtime.lastError.message));\n } else {\n resolve();\n }\n });\n });\n }\n\n async update(\n id: string,\n updateProperties: Omit<chrome.contextMenus.CreateProperties, \"id\">\n ): Promise<void> {\n await chrome.contextMenus.update(id, updateProperties);\n }\n\n async remove(id: string): Promise<void> {\n await chrome.contextMenus.remove(id);\n }\n\n async removeAll(): Promise<void> {\n await chrome.contextMenus.removeAll();\n }\n\n onClicked(\n callback: (info: chrome.contextMenus.OnClickData, tab?: chrome.tabs.Tab) => void\n ): void {\n chrome.contextMenus.onClicked.addListener(callback);\n }\n}\n",
7
- "// Chrome offscreen adapter implementation\n\nimport type { CreateOffscreenDocumentParameters, OffscreenAdapter } from \"../offscreen.adapter\";\n\nexport class ChromeOffscreenAdapter implements OffscreenAdapter {\n async createDocument(parameters: CreateOffscreenDocumentParameters): Promise<void> {\n await chrome.offscreen.createDocument({\n url: parameters.url,\n reasons: parameters.reasons as chrome.offscreen.Reason[],\n justification: parameters.justification,\n });\n }\n\n async closeDocument(): Promise<void> {\n await chrome.offscreen.closeDocument();\n }\n\n async hasDocument(): Promise<boolean> {\n // Chrome doesn't provide a direct API, so we query for offscreen contexts\n const existingContexts = await chrome.runtime.getContexts({\n contextTypes: [\"OFFSCREEN_DOCUMENT\" as chrome.runtime.ContextType],\n });\n return existingContexts.length > 0;\n }\n}\n",
8
- "// Chrome runtime adapter implementation\n\nimport type { MessageSender, PortAdapter, RuntimeAdapter } from \"../runtime.adapter\";\n\ntype MessageListener = (\n message: unknown,\n sender: MessageSender,\n sendResponse: (response: unknown) => void\n) => undefined | boolean;\n\ntype ChromeMessageListener = (\n message: unknown,\n sender: chrome.runtime.MessageSender,\n sendResponse: (response?: unknown) => void\n) => undefined | boolean;\n\nexport class ChromeRuntimeAdapter implements RuntimeAdapter {\n private messageListeners = new Map<MessageListener, ChromeMessageListener>();\n private static listenerCount = 0;\n\n sendMessage<T>(message: T): Promise<unknown> {\n return chrome.runtime.sendMessage(message);\n }\n\n onMessage(\n callback: (\n message: unknown,\n sender: MessageSender,\n sendResponse: (response: unknown) => void\n ) => undefined | boolean\n ): void {\n const wrappedCallback = (\n message: unknown,\n sender: chrome.runtime.MessageSender,\n sendResponse: (response?: unknown) => void\n ) => {\n const mappedSender: MessageSender = {\n ...(sender.tab && {\n tab: {\n id: sender.tab.id ?? 0,\n url: sender.tab.url ?? \"\",\n title: sender.tab.title ?? \"\",\n },\n }),\n ...(sender.frameId !== undefined && { frameId: sender.frameId }),\n ...(sender.url && { url: sender.url }),\n };\n return callback(message, mappedSender, sendResponse);\n };\n\n this.messageListeners.set(callback, wrappedCallback);\n // Chrome's listener signature uses void | boolean, ours uses undefined | boolean\n // These are compatible - undefined is assignable to void for return types\n chrome.runtime.onMessage.addListener(\n wrappedCallback as (\n message: unknown,\n sender: chrome.runtime.MessageSender,\n sendResponse: (response?: unknown) => void\n ) => undefined | boolean\n );\n\n // Track listener count and warn if multiple listeners registered\n ChromeRuntimeAdapter.listenerCount++;\n\n if (ChromeRuntimeAdapter.listenerCount > 1) {\n console.warn(\n `⚠️ WARNING: ${ChromeRuntimeAdapter.listenerCount} chrome.runtime.onMessage listeners registered!\\n\\nMultiple listeners will cause message handlers to execute multiple times.\\nThis is usually caused by:\\n 1. Creating both MessageBus and MessageRouter with separate listeners\\n 2. Calling createBackground() multiple times\\n 3. Calling getMessageBus('background') after createBackground()\\n\\nFix: In background scripts, use createBackground() ONCE at startup.\\nDo not call getMessageBus('background') separately.`\n );\n }\n }\n\n removeMessageListener(\n callback: (\n message: unknown,\n sender: MessageSender,\n sendResponse: (response: unknown) => void\n ) => undefined | boolean\n ): void {\n const wrappedCallback = this.messageListeners.get(callback);\n if (wrappedCallback) {\n // Type-safe cast: wrappedCallback is stored with compatible signature\n chrome.runtime.onMessage.removeListener(\n wrappedCallback as (\n message: unknown,\n sender: chrome.runtime.MessageSender,\n sendResponse: (response?: unknown) => void\n ) => undefined | boolean\n );\n this.messageListeners.delete(callback);\n\n // Decrement listener count\n ChromeRuntimeAdapter.listenerCount = Math.max(0, ChromeRuntimeAdapter.listenerCount - 1);\n }\n }\n\n connect(name: string): PortAdapter {\n const port = chrome.runtime.connect({ name });\n return new ChromePortAdapter(port);\n }\n\n onConnect(callback: (port: PortAdapter) => void): void {\n chrome.runtime.onConnect.addListener((port) => {\n callback(new ChromePortAdapter(port));\n });\n }\n\n getURL(path: string): string {\n return chrome.runtime.getURL(path);\n }\n\n getId(): string {\n return chrome.runtime.id;\n }\n\n openOptionsPage(): void {\n chrome.runtime.openOptionsPage();\n }\n}\n\nclass ChromePortAdapter implements PortAdapter {\n private listeners = {\n message: new Set<(message: unknown) => void>(),\n disconnect: new Set<() => void>(),\n };\n\n constructor(private port: chrome.runtime.Port) {\n // Set up Chrome port listeners\n this.port.onMessage.addListener((message) => {\n for (const callback of this.listeners.message) {\n callback(message);\n }\n });\n\n this.port.onDisconnect.addListener(() => {\n for (const callback of this.listeners.disconnect) {\n callback();\n }\n });\n }\n\n get name(): string {\n return this.port.name;\n }\n\n postMessage(message: unknown): void {\n this.port.postMessage(message);\n }\n\n onMessage(callback: (message: unknown) => void): void {\n this.listeners.message.add(callback);\n }\n\n onDisconnect(callback: () => void): void {\n this.listeners.disconnect.add(callback);\n }\n\n disconnect(): void {\n this.port.disconnect();\n }\n}\n",
9
- "// Chrome storage adapter implementation\n\nimport type { StorageAdapter, StorageChanges } from \"../storage.adapter\";\n\nexport class ChromeStorageAdapter implements StorageAdapter {\n async get<T = Record<string, unknown>>(keys: string | string[] | null): Promise<T> {\n if (keys === null) {\n return (await chrome.storage.local.get()) as T;\n }\n return (await chrome.storage.local.get(keys as never)) as T;\n }\n\n async set(items: Record<string, unknown>): Promise<void> {\n await chrome.storage.local.set(items);\n }\n\n async remove(keys: string | string[]): Promise<void> {\n await chrome.storage.local.remove(keys);\n }\n\n async clear(): Promise<void> {\n await chrome.storage.local.clear();\n }\n\n onChanged(callback: (changes: StorageChanges, areaName: string) => void): void {\n chrome.storage.onChanged.addListener((changes, areaName) => {\n const mappedChanges: StorageChanges = {};\n for (const [key, change] of Object.entries(changes)) {\n mappedChanges[key] = {\n oldValue: change.oldValue,\n newValue: change.newValue,\n };\n }\n callback(mappedChanges, areaName);\n });\n }\n}\n",
10
- "// Chrome tabs adapter implementation\n\nimport type { TabsAdapter } from \"../tabs.adapter\";\n\nexport class ChromeTabsAdapter implements TabsAdapter {\n async query(queryInfo: chrome.tabs.QueryInfo): Promise<chrome.tabs.Tab[]> {\n return chrome.tabs.query(queryInfo);\n }\n\n async get(tabId: number): Promise<chrome.tabs.Tab> {\n return chrome.tabs.get(tabId);\n }\n\n async sendMessage(tabId: number, message: unknown): Promise<unknown> {\n return chrome.tabs.sendMessage(tabId, message);\n }\n\n async reload(tabId: number, reloadProperties?: { bypassCache?: boolean }): Promise<void> {\n if (reloadProperties) {\n await chrome.tabs.reload(tabId, reloadProperties);\n } else {\n await chrome.tabs.reload(tabId);\n }\n }\n\n onRemoved(callback: (tabId: number, removeInfo: chrome.tabs.OnRemovedInfo) => void): void {\n chrome.tabs.onRemoved.addListener(callback);\n }\n\n onUpdated(\n callback: (tabId: number, changeInfo: chrome.tabs.OnUpdatedInfo, tab: chrome.tabs.Tab) => void\n ): void {\n chrome.tabs.onUpdated.addListener(callback);\n }\n\n onActivated(callback: (activeInfo: { tabId: number; windowId: number }) => void): void {\n chrome.tabs.onActivated.addListener(callback);\n }\n\n async create(createProperties: chrome.tabs.CreateProperties): Promise<chrome.tabs.Tab> {\n return chrome.tabs.create(createProperties);\n }\n}\n",
11
- "// Chrome window adapter implementation\n\nimport type { WindowAdapter } from \"../window.adapter\";\n\nexport class ChromeWindowAdapter implements WindowAdapter {\n postMessage(message: unknown, targetOrigin: string): void {\n window.postMessage(message, targetOrigin);\n }\n\n addEventListener(type: \"message\", listener: (event: MessageEvent) => void): void {\n window.addEventListener(type, listener as EventListener);\n }\n\n removeEventListener(type: \"message\", listener: (event: MessageEvent) => void): void {\n window.removeEventListener(type, listener as EventListener);\n }\n}\n",
12
- "// Fetch adapter interface (wraps fetch API)\n\nexport interface FetchAdapter {\n fetch(input: string | URL, init?: RequestInit): Promise<Response>;\n}\n\nexport class BrowserFetchAdapter implements FetchAdapter {\n fetch(input: string | URL, init?: RequestInit): Promise<Response> {\n return fetch(input, init);\n }\n}\n",
13
- "import type { Context, LogLevel } from \"../types/messages\";\n// Logger adapter interface (message-based centralized logging)\nimport type { RuntimeAdapter } from \"./runtime.adapter\";\n\nexport interface LoggerAdapter {\n /**\n * Debug-level logging (verbose, development info)\n */\n debug(message: string, context?: Record<string, unknown>): void;\n\n /**\n * Info-level logging (general information)\n */\n info(message: string, context?: Record<string, unknown>): void;\n\n /**\n * Warning-level logging (non-critical issues)\n */\n warn(message: string, context?: Record<string, unknown>): void;\n\n /**\n * Error-level logging (errors and exceptions)\n */\n error(message: string, error?: Error, context?: Record<string, unknown>): void;\n\n /**\n * Log with explicit level\n */\n log(level: LogLevel, message: string, context?: Record<string, unknown>): void;\n}\n\nexport interface MessageLoggerOptions {\n consoleMirror?: boolean; // Also log to console (for development)\n fallbackToConsole?: boolean; // Log to console if message send fails (default: true)\n}\n\n/**\n * Message-based logger that sends LOG messages to background LogStore\n * Uses RuntimeAdapter directly to avoid circular dependency with MessageBus\n */\nexport class MessageLoggerAdapter implements LoggerAdapter {\n constructor(\n private runtime: RuntimeAdapter,\n private sourceContext: Context,\n private options?: MessageLoggerOptions\n ) {}\n\n debug(message: string, context?: Record<string, unknown>): void {\n this.sendLog(\"debug\", message, context);\n }\n\n info(message: string, context?: Record<string, unknown>): void {\n this.sendLog(\"info\", message, context);\n }\n\n warn(message: string, context?: Record<string, unknown>): void {\n this.sendLog(\"warn\", message, context);\n }\n\n error(message: string, error?: Error, context?: Record<string, unknown>): void {\n this.sendLog(\"error\", message, context, error);\n }\n\n log(level: LogLevel, message: string, context?: Record<string, unknown>): void {\n this.sendLog(level, message, context);\n }\n\n private sendLog(\n level: LogLevel,\n message: string,\n context?: Record<string, unknown>,\n error?: Error\n ): void {\n // Optional console mirror for development\n if (this.options?.consoleMirror) {\n const consoleMethod = console[level] || console.log;\n consoleMethod(`[${this.sourceContext}]`, message, context || \"\", error || \"\");\n }\n\n // Send LOG message to background (fire-and-forget)\n const logMessage = {\n type: \"LOG\" as const,\n level,\n message,\n context,\n error: error?.message,\n stack: error?.stack,\n source: this.sourceContext,\n timestamp: Date.now(),\n };\n\n // Use runtime.sendMessage for fire-and-forget messaging\n this.runtime.sendMessage(logMessage).catch((sendError) => {\n // Fallback to console if messaging fails\n if (this.options?.fallbackToConsole !== false) {\n console[level](`[${this.sourceContext}] ${message}`, context || \"\", error || \"\");\n console.warn(\"Failed to send log message:\", sendError);\n }\n });\n }\n}\n",
14
- "// Adapter factory and exports\n\nimport type { Context } from \"../types/messages\";\nimport { ChromeContextMenusAdapter } from \"./chrome/context-menus.chrome\";\nimport { ChromeOffscreenAdapter } from \"./chrome/offscreen.chrome\";\nimport { ChromeRuntimeAdapter } from \"./chrome/runtime.chrome\";\nimport { ChromeStorageAdapter } from \"./chrome/storage.chrome\";\nimport { ChromeTabsAdapter } from \"./chrome/tabs.chrome\";\nimport { ChromeWindowAdapter } from \"./chrome/window.chrome\";\nimport type { ContextMenusAdapter } from \"./context-menus.adapter\";\nimport type { FetchAdapter } from \"./fetch.adapter\";\nimport { BrowserFetchAdapter } from \"./fetch.adapter\";\nimport type { LoggerAdapter } from \"./logger.adapter\";\nimport { MessageLoggerAdapter } from \"./logger.adapter\";\nimport type { OffscreenAdapter } from \"./offscreen.adapter\";\nimport type { RuntimeAdapter } from \"./runtime.adapter\";\nimport type { StorageAdapter } from \"./storage.adapter\";\nimport type { TabsAdapter } from \"./tabs.adapter\";\nimport type { WindowAdapter } from \"./window.adapter\";\n\nexport interface ExtensionAdapters {\n runtime: RuntimeAdapter;\n storage: StorageAdapter;\n tabs: TabsAdapter;\n window: WindowAdapter;\n offscreen: OffscreenAdapter;\n contextMenus: ContextMenusAdapter;\n fetch: FetchAdapter;\n logger: LoggerAdapter;\n}\n\nexport interface CreateChromeAdaptersOptions {\n consoleMirror?: boolean; // Mirror logs to console for development\n}\n\n/**\n * Create Chrome-specific adapters with context\n */\nexport function createChromeAdapters(\n context: Context,\n options?: CreateChromeAdaptersOptions\n): ExtensionAdapters {\n const runtime = new ChromeRuntimeAdapter();\n\n return {\n runtime,\n storage: new ChromeStorageAdapter(),\n tabs: new ChromeTabsAdapter(),\n window: new ChromeWindowAdapter(),\n offscreen: new ChromeOffscreenAdapter(),\n contextMenus: new ChromeContextMenusAdapter(),\n fetch: new BrowserFetchAdapter(),\n logger: new MessageLoggerAdapter(runtime, context, {\n ...(options?.consoleMirror !== undefined && { consoleMirror: options.consoleMirror }),\n fallbackToConsole: true,\n }),\n };\n}\n\nexport * from \"./context-menus.adapter\";\nexport * from \"./fetch.adapter\";\nexport * from \"./logger.adapter\";\nexport * from \"./offscreen.adapter\";\n// Re-export types\nexport * from \"./runtime.adapter\";\nexport * from \"./storage.adapter\";\nexport * from \"./tabs.adapter\";\nexport * from \"./window.adapter\";\n",
15
- "// Type definitions for all messages in the extension\n\n/**\n * Base message interface that all messages must satisfy.\n * This allows users to define custom messages alongside framework messages.\n */\nexport interface BaseMessage {\n type: string;\n}\n\nexport type Context =\n | \"background\"\n | \"content\"\n | \"page\"\n | \"devtools\"\n | \"popup\"\n | \"options\"\n | \"sidepanel\"\n | \"offscreen\";\n\n// All contexts (useful for broadcast)\nexport const ALL_CONTEXTS: Context[] = [\n \"background\",\n \"content\",\n \"page\",\n \"devtools\",\n \"popup\",\n \"options\",\n \"sidepanel\",\n \"offscreen\",\n] as const;\n\n// Settings schema\nexport type Settings = {\n theme: \"light\" | \"dark\" | \"auto\";\n autoSync: boolean;\n debugMode: boolean;\n notifications: boolean;\n apiEndpoint: string;\n refreshInterval: number;\n};\n\nexport const defaultSettings: Settings = {\n theme: \"auto\",\n autoSync: true,\n debugMode: false,\n notifications: true,\n apiEndpoint: \"https://api.example.com\",\n refreshInterval: 60000,\n};\n\n// Logging types\nexport type LogLevel = \"debug\" | \"info\" | \"warn\" | \"error\";\n\nexport type LogEntry = {\n id: string;\n level: LogLevel;\n message: string;\n context?: Record<string, unknown>;\n error?: string;\n stack?: string;\n source: Context;\n timestamp: number;\n};\n\n// All possible messages (discriminated union)\nexport type ExtensionMessage =\n // DOM Operations (handled by Content Script)\n | { type: \"DOM_QUERY\"; selector: string }\n | { type: \"DOM_UPDATE\"; selector: string; content: string }\n | {\n type: \"DOM_INSERT\";\n position: \"beforebegin\" | \"afterbegin\" | \"beforeend\" | \"afterend\";\n html: string;\n }\n | { type: \"DOM_REMOVE\"; selector: string }\n\n // Page Script Operations (handled by Page Script)\n | { type: \"PAGE_EVAL\"; code: string }\n | { type: \"PAGE_GET_VAR\"; varName: string }\n | { type: \"PAGE_CALL_FN\"; fnName: string; args: unknown[] }\n | { type: \"PAGE_SET_VAR\"; varName: string; value: unknown }\n\n // API Operations (handled by Background)\n | {\n type: \"API_REQUEST\";\n endpoint: string;\n method: \"GET\" | \"POST\" | \"PUT\" | \"DELETE\";\n body?: unknown;\n headers?: Record<string, string>;\n }\n | {\n type: \"API_BATCH\";\n requests: Array<{ endpoint: string; method: string; body?: unknown }>;\n }\n\n // Clipboard Operations (handled by Offscreen)\n | { type: \"CLIPBOARD_WRITE\"; text: string }\n | { type: \"CLIPBOARD_WRITE_HTML\"; html: string }\n | { type: \"CLIPBOARD_WRITE_RICH\"; data: { text: string; html: string } }\n | { type: \"CLIPBOARD_READ\" }\n\n // Context Menu (handled by Background)\n | {\n type: \"CONTEXT_MENU_CLICKED\";\n menuId: string;\n info: chrome.contextMenus.OnClickData;\n tabId: number;\n }\n | {\n type: \"CONTEXT_MENU_CREATE\";\n id: string;\n title: string;\n contexts: chrome.contextMenus.ContextType[];\n }\n | { type: \"CONTEXT_MENU_REMOVE\"; id: string }\n\n // State Sync (broadcast) - Internal only, handled by state primitives\n | {\n type: \"STATE_SYNC\";\n key: string;\n value: unknown;\n clock: number;\n }\n\n // Tab Operations (handled by Background)\n | { type: \"TAB_QUERY\"; queryInfo: chrome.tabs.QueryInfo }\n | { type: \"TAB_GET_CURRENT\" }\n | { type: \"TAB_RELOAD\"; tabId: number }\n\n // DevTools Operations\n | { type: \"DEVTOOLS_INSPECT_ELEMENT\"; selector: string }\n | {\n type: \"DEVTOOLS_LOG\";\n level: \"log\" | \"warn\" | \"error\";\n message: string;\n data?: unknown;\n }\n\n // Logging (handled by Background LogStore)\n | {\n type: \"LOG\";\n level: LogLevel;\n message: string;\n context?: Record<string, unknown>;\n error?: string;\n stack?: string;\n source: Context;\n timestamp: number;\n }\n | {\n type: \"LOGS_GET\";\n filters?: {\n level?: LogLevel;\n source?: Context;\n since?: number;\n limit?: number;\n };\n }\n | { type: \"LOGS_CLEAR\" }\n | { type: \"LOGS_EXPORT\" }\n\n // Test Messages (only used in tests)\n | { type: \"TEST_MESSAGE\"; data?: unknown }\n | { type: \"TEST\"; iteration?: number }\n | { type: \"CUSTOM_MESSAGE\"; data?: unknown }\n | { type: \"SETTINGS_GET\" }\n | {\n type: \"SIGNAL_UPDATE\";\n key?: string;\n value?: unknown;\n signalId?: string;\n source?: Context;\n }\n | {\n type: \"USER_DATA\";\n password?: string;\n apiKey?: string;\n [key: string]: unknown;\n };\n\n// Helper: Look up the full message type from a union based on the 'type' discriminator\ntype LookupMessage<TUnion, TType extends string> = TUnion extends { type: TType } ? TUnion : never;\n\n// Extract response type from message using phantom type\n// If message has __response field (phantom type), use it; otherwise infer from framework messages\nexport type MessageResponse<T extends BaseMessage> =\n // First, try to find the matching message in the union by type discriminator\n T extends { type: infer TType extends string }\n ? LookupMessage<T, TType> extends { readonly __response?: infer R }\n ? R // Found phantom type, use it\n : T extends ExtensionMessage\n ? // Framework message - infer from type\n // DOM Operations\n T extends { type: \"DOM_QUERY\" }\n ? {\n elements: Array<{\n tag: string;\n text: string;\n html: string;\n attrs: Record<string, string>;\n rect?: DOMRect;\n }>;\n }\n : T extends { type: \"DOM_UPDATE\" }\n ? { success: boolean }\n : T extends { type: \"DOM_INSERT\" }\n ? { success: boolean }\n : T extends { type: \"DOM_REMOVE\" }\n ? { success: boolean; count: number }\n : // Page Script Operations\n T extends { type: \"PAGE_EVAL\" }\n ? { result: unknown; error?: string }\n : T extends { type: \"PAGE_GET_VAR\" }\n ? { value: unknown; exists: boolean }\n : T extends { type: \"PAGE_CALL_FN\" }\n ? { result: unknown; error?: string }\n : T extends { type: \"PAGE_SET_VAR\" }\n ? { success: boolean }\n : // API Operations\n T extends { type: \"API_REQUEST\" }\n ? {\n data: unknown;\n status: number;\n statusText: string;\n headers: Record<string, string>;\n error?: string;\n }\n : T extends { type: \"API_BATCH\" }\n ? {\n results: Array<{\n data: unknown;\n status: number;\n error?: string;\n }>;\n }\n : // Clipboard Operations\n T extends { type: \"CLIPBOARD_WRITE\" }\n ? { success: boolean }\n : T extends { type: \"CLIPBOARD_WRITE_HTML\" }\n ? { success: boolean }\n : T extends { type: \"CLIPBOARD_WRITE_RICH\" }\n ? { success: boolean }\n : T extends { type: \"CLIPBOARD_READ\" }\n ? { text: string }\n : // Context Menu\n T extends { type: \"CONTEXT_MENU_CLICKED\" }\n ? undefined\n : T extends { type: \"CONTEXT_MENU_CREATE\" }\n ? { success: boolean }\n : T extends { type: \"CONTEXT_MENU_REMOVE\" }\n ? { success: boolean }\n : // State Sync\n T extends { type: \"STATE_SYNC\" }\n ? undefined\n : // Tab Operations\n T extends { type: \"TAB_QUERY\" }\n ? { tabs: chrome.tabs.Tab[] }\n : T extends {\n type: \"TAB_GET_CURRENT\";\n }\n ? { tab: chrome.tabs.Tab }\n : T extends {\n type: \"TAB_RELOAD\";\n }\n ? { success: boolean }\n : // DevTools Operations\n T extends {\n type: \"DEVTOOLS_INSPECT_ELEMENT\";\n }\n ? {\n success: boolean;\n }\n : T extends {\n type: \"DEVTOOLS_LOG\";\n }\n ? undefined\n : // Logging Operations\n T extends {\n type: \"LOG\";\n }\n ? {\n success: boolean;\n }\n : T extends {\n type: \"LOGS_GET\";\n }\n ? {\n logs: LogEntry[];\n }\n : T extends {\n type: \"LOGS_CLEAR\";\n }\n ? {\n success: boolean;\n count: number;\n }\n : T extends {\n type: \"LOGS_EXPORT\";\n }\n ? {\n json: string;\n count: number;\n }\n : T extends {\n type: \"SETTINGS_GET\";\n }\n ? {\n settings: unknown;\n }\n : undefined\n : unknown // For custom messages outside ExtensionMessage, require phantom type\n : unknown; // Fallback for messages without type field\n\n// Message handler mapping (which context handles which message)\n// Can be a single context or an array for multi-target routing\nexport type MessageHandler = {\n DOM_QUERY: \"content\";\n DOM_UPDATE: \"content\";\n DOM_INSERT: \"content\";\n DOM_REMOVE: \"content\";\n\n PAGE_EVAL: \"page\";\n PAGE_GET_VAR: \"page\";\n PAGE_CALL_FN: \"page\";\n PAGE_SET_VAR: \"page\";\n\n API_REQUEST: \"background\";\n API_BATCH: \"background\";\n\n CLIPBOARD_WRITE: \"offscreen\";\n CLIPBOARD_WRITE_HTML: \"offscreen\";\n CLIPBOARD_WRITE_RICH: \"offscreen\";\n CLIPBOARD_READ: \"offscreen\";\n\n CONTEXT_MENU_CLICKED: \"background\";\n CONTEXT_MENU_CREATE: \"background\";\n CONTEXT_MENU_REMOVE: \"background\";\n\n STATE_SYNC: Context[]; // Broadcast to all contexts\n\n TAB_QUERY: \"background\";\n TAB_GET_CURRENT: \"background\";\n TAB_RELOAD: \"background\";\n\n DEVTOOLS_INSPECT_ELEMENT: \"content\";\n DEVTOOLS_LOG: \"background\";\n\n LOG: \"background\";\n LOGS_GET: \"background\";\n LOGS_CLEAR: \"background\";\n LOGS_EXPORT: \"background\";\n};\n\n// Routed message envelope\nexport type RoutedMessage<T extends BaseMessage = ExtensionMessage> = {\n id: string; // Correlation ID (UUID)\n source: Context; // Which context sent it\n targets: Context[]; // Which contexts should receive this (can be multiple)\n tabId?: number; // Required for per-tab contexts\n timestamp: number; // When it was sent\n payload: T; // The actual message\n};\n\n// Routed response envelope\nexport type RoutedResponse<T extends BaseMessage = ExtensionMessage> = {\n id: string; // Matches request ID\n success: boolean; // Whether operation succeeded\n data?: MessageResponse<T>; // Response data\n error?: string; // Error message if failed\n timestamp: number; // When response was sent\n};\n",
16
- "// Custom error classes for the extension\n\nimport type { LoggerAdapter } from \"../adapters/logger.adapter\";\n\n/**\n * Base error class for all extension errors\n */\nexport class ExtensionError extends Error {\n constructor(\n message: string,\n public readonly code: string,\n public readonly context?: Record<string, unknown>\n ) {\n super(message);\n this.name = this.constructor.name;\n Error.captureStackTrace?.(this, this.constructor);\n }\n}\n\n/**\n * Error thrown when a message times out\n */\nexport class TimeoutError extends ExtensionError {\n constructor(\n message: string,\n public readonly timeoutMs: number,\n context?: Record<string, unknown>\n ) {\n super(message, \"TIMEOUT_ERROR\", { ...context, timeoutMs });\n }\n}\n\n/**\n * Error thrown when a connection is lost or unavailable\n */\nexport class ConnectionError extends ExtensionError {\n constructor(message: string, context?: Record<string, unknown>) {\n super(message, \"CONNECTION_ERROR\", context);\n }\n}\n\n/**\n * Error thrown by MessageRouter\n */\nexport class MessageRouterError extends ExtensionError {\n constructor(message: string, context?: Record<string, unknown>) {\n super(message, \"MESSAGE_ROUTER_ERROR\", context);\n }\n}\n\n/**\n * Error thrown when a message handler fails\n */\nexport class HandlerError extends ExtensionError {\n constructor(\n message: string,\n public readonly messageType: string,\n context?: Record<string, unknown>\n ) {\n super(message, \"HANDLER_ERROR\", { ...context, messageType });\n }\n}\n\n/**\n * Error thrown when an API request fails\n */\nexport class APIError extends ExtensionError {\n constructor(\n message: string,\n public readonly statusCode: number,\n context?: Record<string, unknown>\n ) {\n super(message, \"API_ERROR\", { ...context, statusCode });\n }\n}\n\n/**\n * Error thrown when offscreen document operations fail\n */\nexport class OffscreenError extends ExtensionError {\n constructor(message: string, context?: Record<string, unknown>) {\n super(message, \"OFFSCREEN_ERROR\", context);\n }\n}\n\n/**\n * Error utility for logging and throwing errors\n */\nexport class ErrorHandler {\n constructor(private logger: LoggerAdapter) {}\n\n /**\n * Log an error and then throw it\n */\n throw(error: ExtensionError): never {\n this.logger.error(error.message, error, error.context);\n throw error;\n }\n\n /**\n * Log an error and return it (for Promise.reject)\n */\n reject(error: ExtensionError): ExtensionError {\n this.logger.error(error.message, error, error.context);\n return error;\n }\n\n /**\n * Wrap an unknown error in an ExtensionError\n */\n wrap(\n error: unknown,\n message: string,\n code: string,\n context?: Record<string, unknown>\n ): ExtensionError {\n const originalError = error instanceof Error ? error : new Error(String(error));\n const wrappedError = new ExtensionError(`${message}: ${originalError.message}`, code, {\n ...context,\n originalError: originalError.message,\n originalStack: originalError.stack,\n });\n\n // Preserve original stack if available\n if (originalError.stack) {\n wrappedError.stack = originalError.stack;\n }\n\n this.logger.error(wrappedError.message, wrappedError, wrappedError.context);\n return wrappedError;\n }\n}\n",
17
- "/**\n * Context-Specific Helpers\n *\n * Provides context-specific utility methods to make common patterns easier.\n */\n\nimport type { ExtensionAdapters } from \"../adapters\";\n\n/**\n * View types for extension views.\n * This is a local type definition since chrome.extension.ViewType is deprecated.\n */\ntype ExtensionViewType = \"tab\" | \"popup\" | \"notification\";\n\n/**\n * Helpers for content script context.\n * Content scripts have access to the DOM and page context.\n */\nexport interface ContentScriptHelpers {\n /**\n * Get basic page information.\n */\n getPageInfo(): {\n url: string;\n title: string;\n host: string;\n pathname: string;\n readyState: DocumentReadyState;\n };\n\n /**\n * Query DOM elements (returns serializable data).\n */\n queryElements(selector: string): Array<{\n tagName: string;\n id: string;\n className: string;\n textContent: string;\n }>;\n\n /**\n * Get page metadata (meta tags).\n */\n getPageMetadata(): Record<string, string>;\n\n /**\n * Inject CSS into the page.\n */\n injectCSS(css: string): void;\n\n /**\n * Remove injected CSS.\n */\n removeCSS(styleId: string): void;\n}\n\nexport function createContentScriptHelpers(): ContentScriptHelpers {\n return {\n getPageInfo() {\n return {\n url: window.location.href,\n title: document.title,\n host: window.location.host,\n pathname: window.location.pathname,\n readyState: document.readyState,\n };\n },\n\n queryElements(selector: string) {\n const elements = document.querySelectorAll(selector);\n return Array.from(elements).map((el) => ({\n tagName: el.tagName,\n id: el.id,\n className: el.className,\n textContent: el.textContent?.slice(0, 100) || \"\",\n }));\n },\n\n getPageMetadata() {\n const metadata: Record<string, string> = {};\n const metaTags = document.querySelectorAll(\"meta\");\n\n for (const tag of Array.from(metaTags)) {\n const name = tag.getAttribute(\"name\") || tag.getAttribute(\"property\");\n const content = tag.getAttribute(\"content\");\n\n if (name && content) {\n metadata[name] = content;\n }\n }\n\n return metadata;\n },\n\n injectCSS(css: string) {\n const styleId = `ext-injected-${Date.now()}`;\n const style = document.createElement(\"style\");\n style.id = styleId;\n style.textContent = css;\n document.head.appendChild(style);\n },\n\n removeCSS(styleId: string) {\n const style = document.getElementById(styleId);\n if (style) {\n style.remove();\n }\n },\n };\n}\n\n/**\n * Helpers for DevTools panel context.\n * DevTools can inspect the page and access Chrome DevTools APIs.\n */\nexport interface DevToolsHelpers {\n /**\n * Get the ID of the tab being inspected.\n */\n get inspectedTabId(): number | undefined;\n\n /**\n * Execute code in the inspected page context.\n */\n evalInPage<T = unknown>(code: string): Promise<T>;\n\n /**\n * Get page resource content (HTML, CSS, JS files).\n */\n getPageResource(url: string): Promise<string>;\n\n /**\n * Reload the inspected page.\n */\n reloadInspectedPage(options?: { ignoreCache?: boolean; userAgent?: string }): void;\n}\n\nexport function createDevToolsHelpers(): DevToolsHelpers {\n return {\n get inspectedTabId() {\n return chrome.devtools?.inspectedWindow?.tabId;\n },\n\n evalInPage<T = unknown>(code: string): Promise<T> {\n return new Promise((resolve, reject) => {\n if (!chrome.devtools?.inspectedWindow) {\n reject(new Error(\"DevTools inspectedWindow not available\"));\n return;\n }\n\n chrome.devtools.inspectedWindow.eval(code, (result, error) => {\n if (error) {\n reject(new Error(error.isException ? error.value : \"Execution error\"));\n } else {\n resolve(result as T);\n }\n });\n });\n },\n\n getPageResource(url: string): Promise<string> {\n return new Promise((resolve, reject) => {\n if (!chrome.devtools?.inspectedWindow) {\n reject(new Error(\"DevTools inspectedWindow not available\"));\n return;\n }\n\n chrome.devtools.inspectedWindow.getResources((resources) => {\n const resource = resources.find((r) => r.url === url);\n if (!resource) {\n reject(new Error(`Resource not found: ${url}`));\n return;\n }\n\n resource.getContent((content, encoding) => {\n if (encoding === \"base64\") {\n resolve(atob(content));\n } else {\n resolve(content);\n }\n });\n });\n });\n },\n\n reloadInspectedPage(options = {}) {\n if (!chrome.devtools?.inspectedWindow) {\n console.warn(\"DevTools inspectedWindow not available\");\n return;\n }\n\n chrome.devtools.inspectedWindow.reload(options);\n },\n };\n}\n\n/**\n * Helpers for popup context.\n * Popups are short-lived UI windows.\n */\nexport interface PopupHelpers {\n /**\n * Get current active tab.\n */\n getCurrentTab(): Promise<chrome.tabs.Tab | undefined>;\n\n /**\n * Close the popup programmatically.\n */\n closePopup(): void;\n\n /**\n * Set popup dimensions.\n */\n setDimensions(width: number, height: number): void;\n}\n\nexport function createPopupHelpers(adapters: ExtensionAdapters): PopupHelpers {\n return {\n async getCurrentTab() {\n const tabs = await adapters.tabs.query({ active: true, currentWindow: true });\n return tabs[0];\n },\n\n closePopup() {\n window.close();\n },\n\n setDimensions(width: number, height: number) {\n document.body.style.width = `${width}px`;\n document.body.style.height = `${height}px`;\n },\n };\n}\n\n/**\n * Helpers for options page context.\n * Options pages are full-page settings interfaces.\n */\nexport interface OptionsHelpers {\n /**\n * Open extension in new tab (for external links).\n */\n openInNewTab(path: string): void;\n\n /**\n * Show save confirmation message.\n */\n showSaveConfirmation(message?: string, duration?: number): void;\n\n /**\n * Show error message.\n */\n showError(message: string, duration?: number): void;\n}\n\nexport function createOptionsHelpers(adapters: ExtensionAdapters): OptionsHelpers {\n return {\n openInNewTab(path: string) {\n adapters.tabs.create({ url: path });\n },\n\n showSaveConfirmation(message = \"Settings saved!\", duration = 3000) {\n const notification = document.createElement(\"div\");\n notification.textContent = message;\n notification.style.cssText = `\n position: fixed;\n top: 20px;\n right: 20px;\n background: #4caf50;\n color: white;\n padding: 12px 24px;\n border-radius: 4px;\n box-shadow: 0 2px 8px rgba(0,0,0,0.2);\n z-index: 10000;\n animation: slideIn 0.3s ease;\n `;\n\n document.body.appendChild(notification);\n\n setTimeout(() => {\n notification.style.animation = \"slideOut 0.3s ease\";\n setTimeout(() => notification.remove(), 300);\n }, duration);\n },\n\n showError(message: string, duration = 5000) {\n const notification = document.createElement(\"div\");\n notification.textContent = message;\n notification.style.cssText = `\n position: fixed;\n top: 20px;\n right: 20px;\n background: #f44336;\n color: white;\n padding: 12px 24px;\n border-radius: 4px;\n box-shadow: 0 2px 8px rgba(0,0,0,0.2);\n z-index: 10000;\n animation: slideIn 0.3s ease;\n `;\n\n document.body.appendChild(notification);\n\n setTimeout(() => {\n notification.style.animation = \"slideOut 0.3s ease\";\n setTimeout(() => notification.remove(), 300);\n }, duration);\n },\n };\n}\n\n/**\n * Helpers for side panel context.\n * Side panels are persistent companion interfaces.\n */\nexport interface SidePanelHelpers {\n /**\n * Get current active tab.\n */\n getCurrentTab(): Promise<chrome.tabs.Tab | undefined>;\n\n /**\n * Check if side panel is currently visible.\n */\n isVisible(): boolean;\n\n /**\n * Set side panel width (if supported by browser).\n */\n setWidth(width: number): void;\n}\n\nexport function createSidePanelHelpers(adapters: ExtensionAdapters): SidePanelHelpers {\n return {\n async getCurrentTab() {\n const tabs = await adapters.tabs.query({ active: true, currentWindow: true });\n return tabs[0];\n },\n\n isVisible() {\n return document.visibilityState === \"visible\";\n },\n\n setWidth(width: number) {\n document.body.style.width = `${width}px`;\n },\n };\n}\n\n/**\n * Helpers for background context.\n * Background scripts coordinate extension behavior.\n */\nexport interface BackgroundHelpers {\n /**\n * Get all open tabs.\n */\n getAllTabs(): Promise<chrome.tabs.Tab[]>;\n\n /**\n * Get extension views (popup, options, devtools, etc).\n */\n getExtensionViews(type?: ExtensionViewType): Window[];\n\n /**\n * Open options page.\n */\n openOptionsPage(): void;\n\n /**\n * Set extension badge.\n */\n setBadge(text: string, color?: string): void;\n\n /**\n * Clear extension badge.\n */\n clearBadge(): void;\n}\n\nexport function createBackgroundHelpers(adapters: ExtensionAdapters): BackgroundHelpers {\n return {\n async getAllTabs() {\n return adapters.tabs.query({});\n },\n\n getExtensionViews(type?: ExtensionViewType) {\n return chrome.extension.getViews(\n type && (type === \"popup\" || type === \"tab\") ? { type } : undefined\n );\n },\n\n openOptionsPage() {\n adapters.runtime.openOptionsPage();\n },\n\n setBadge(text: string, color = \"#f44336\") {\n chrome.action.setBadgeText({ text });\n chrome.action.setBadgeBackgroundColor({ color });\n },\n\n clearBadge() {\n chrome.action.setBadgeText({ text: \"\" });\n },\n };\n}\n",
18
- "// Runtime handler execution tracking to prevent double-execution bugs\n// Only active in development mode, zero runtime cost in production\n\nexport class HandlerExecutionTracker {\n private executions = new Map<string, Map<string, number>>(); // messageId → handlerType → count\n private readonly isDevelopment: boolean;\n\n constructor() {\n // Check if we're in development mode or test mode\n this.isDevelopment =\n typeof process !== \"undefined\" &&\n (process.env?.NODE_ENV === \"development\" || process.env?.NODE_ENV === \"test\");\n }\n\n /**\n * Track a handler execution. Throws error if handler executes multiple times\n * for the same message ID.\n *\n * @param messageId - Unique message identifier\n * @param handlerType - Handler type (e.g., 'TODO_ADD')\n * @throws Error if handler already executed for this message\n */\n track(messageId: string, handlerType: string): void {\n if (!this.isDevelopment) return;\n\n let handlerCounts = this.executions.get(messageId);\n if (!handlerCounts) {\n handlerCounts = new Map();\n this.executions.set(messageId, handlerCounts);\n }\n\n const count = (handlerCounts.get(handlerType) || 0) + 1;\n handlerCounts.set(handlerType, count);\n\n if (count > 1) {\n const error = new Error(\n `🔴 DOUBLE EXECUTION DETECTED\\n\\nHandler \"${handlerType}\" executed ${count} times for message ${messageId}.\\n\\nThis indicates multiple chrome.runtime.onMessage listeners are registered.\\nCommon causes:\\n 1. Both MessageBus and MessageRouter registered listeners\\n 2. createBackground() called multiple times\\n 3. Handler registered in multiple places\\n\\nFix: Ensure only ONE listener is registered. In background scripts,\\nuse createBackground() instead of getMessageBus().\\n`\n );\n\n console.error(error);\n\n // Also log the execution trace\n console.error(\"Execution trace for message:\", messageId);\n console.error(Array.from(handlerCounts.entries()));\n\n throw error;\n }\n\n // Cleanup old messages after 5 seconds to prevent memory leak\n setTimeout(() => {\n this.executions.delete(messageId);\n }, 5000);\n }\n\n /**\n * Reset all tracked executions. Useful for testing.\n */\n reset(): void {\n this.executions.clear();\n }\n\n /**\n * Get execution count for a specific message and handler.\n * Useful for testing.\n */\n getExecutionCount(messageId: string, handlerType: string): number {\n return this.executions.get(messageId)?.get(handlerType) || 0;\n }\n}\n\n// Global singleton instance\nexport const globalExecutionTracker = new HandlerExecutionTracker();\n",
19
- "// Type-safe message bus for extension communication\n\nimport type { ExtensionAdapters, MessageSender } from \"../adapters\";\nimport { createChromeAdapters } from \"../adapters\";\nimport type {\n BaseMessage,\n Context,\n ExtensionMessage,\n MessageResponse,\n RoutedMessage,\n RoutedResponse,\n} from \"../types/messages\";\nimport { ALL_CONTEXTS } from \"../types/messages\";\nimport type {\n BackgroundHelpers,\n ContentScriptHelpers,\n DevToolsHelpers,\n OptionsHelpers,\n PopupHelpers,\n SidePanelHelpers,\n} from \"./context-specific-helpers\";\nimport {\n createBackgroundHelpers,\n createContentScriptHelpers,\n createDevToolsHelpers,\n createOptionsHelpers,\n createPopupHelpers,\n createSidePanelHelpers,\n} from \"./context-specific-helpers\";\nimport { ConnectionError, ErrorHandler, HandlerError, TimeoutError } from \"./errors\";\nimport { globalExecutionTracker } from \"./handler-execution-tracker\";\n\n// Type guards for runtime message validation\n// Note: These validate structure but can't validate TMessage type at runtime\nexport function isRoutedMessage<TMessage extends BaseMessage = BaseMessage>(\n value: unknown\n): value is RoutedMessage<TMessage> {\n if (typeof value !== \"object\" || value === null) return false;\n // Use 'in' operator for type narrowing - no cast needed\n if (!(\"id\" in value) || !(\"source\" in value) || !(\"targets\" in value) || !(\"payload\" in value)) {\n return false;\n }\n // TypeScript now knows these properties exist\n return (\n typeof value.id === \"string\" &&\n typeof value.source === \"string\" &&\n Array.isArray(value.targets) &&\n typeof value.payload === \"object\" &&\n value.payload !== null\n );\n}\n\nexport function isRoutedResponse<TMessage extends BaseMessage = BaseMessage>(\n value: unknown\n): value is RoutedResponse<TMessage> {\n if (typeof value !== \"object\" || value === null) return false;\n // Use 'in' operator for type narrowing - no cast needed\n if (!(\"id\" in value) || !(\"success\" in value)) {\n return false;\n }\n // TypeScript now knows these properties exist\n return typeof value.id === \"string\" && typeof value.success === \"boolean\";\n}\n\ntype PendingRequest<TMessage extends BaseMessage = ExtensionMessage> = {\n // Accepts the union of all possible response types\n // Type safety is enforced at handler registration (.on) and invocation (send)\n resolve: (value: MessageResponse<TMessage> | undefined) => void;\n reject: (error: Error) => void;\n timestamp: number;\n timeout: NodeJS.Timeout;\n};\n\nexport class MessageBus<TMessage extends BaseMessage = ExtensionMessage> {\n public context: Context;\n public adapters: ExtensionAdapters;\n public helpers:\n | ContentScriptHelpers\n | DevToolsHelpers\n | PopupHelpers\n | OptionsHelpers\n | SidePanelHelpers\n | BackgroundHelpers\n | Record<string, never>;\n public pendingRequests = new Map<string, PendingRequest<TMessage>>();\n // Handlers Map stores arrays of functions with varying signatures\n // Type safety is enforced at registration (.on()) and invocation (send())\n // biome-ignore lint/complexity/noBannedTypes: Function type needed for dynamic handler map\n private handlers = new Map<string, Function[]>();\n private port: ReturnType<ExtensionAdapters[\"runtime\"][\"connect\"]> | null = null;\n private errorHandler: ErrorHandler;\n private userErrorHandlers: Array<(error: Error, bus: MessageBus<TMessage>) => void> = [];\n private stateAccessor: (() => unknown) | null = null;\n public messageListener:\n | ((\n message: unknown,\n sender: MessageSender,\n sendResponse: (response: unknown) => void\n ) => boolean)\n | null = null;\n\n constructor(\n context: Context,\n adapters?: ExtensionAdapters,\n options?: { skipListenerSetup?: boolean }\n ) {\n this.context = context;\n this.adapters = adapters || createChromeAdapters(context);\n this.errorHandler = new ErrorHandler(this.adapters.logger);\n this.helpers = this.createContextHelpers();\n\n // Skip listener setup if MessageRouter will handle it\n if (!options?.skipListenerSetup) {\n this.setupListeners();\n }\n }\n\n /**\n * Send a message with type safety.\n * Response type is inferred from message type, though TypeScript requires\n * the return type to be widened due to Map storage limitations.\n * Runtime type safety is ensured by handler registration and invocation.\n */\n async send<T extends TMessage>(\n payload: T,\n options?: {\n target?: Context | Context[];\n tabId?: number;\n timeout?: number;\n }\n ): Promise<\n | MessageResponse<Extract<TMessage, { type: T extends { type: infer TType } ? TType : never }>>\n | undefined\n > {\n const id = crypto.randomUUID();\n\n // For custom messages (not ExtensionMessage), targets must be explicitly provided\n let targets: Context[];\n if (options?.target) {\n // Handle single target from options\n if (Array.isArray(options.target)) {\n targets = options.target;\n } else {\n targets = [options.target];\n }\n } else {\n const inferredTarget = this.inferTarget(payload.type);\n if (!inferredTarget) {\n throw new Error(\n `Message type \"${payload.type}\" is not a framework message. Please provide explicit 'target' option.`\n );\n }\n // inferredTarget can be a single context or an array\n targets = Array.isArray(inferredTarget) ? inferredTarget : [inferredTarget];\n }\n\n const message: RoutedMessage<T> = {\n id,\n source: this.context,\n targets,\n ...(options?.tabId !== undefined && { tabId: options.tabId }),\n timestamp: Date.now(),\n payload,\n };\n\n return new Promise<MessageResponse<T> | undefined>((resolve, reject) => {\n const timeoutMs = options?.timeout || 5000;\n const timeout = setTimeout(() => {\n this.pendingRequests.delete(id);\n const error = new TimeoutError(`Message timeout: ${payload.type}`, timeoutMs, {\n messageType: payload.type,\n targets,\n });\n this.notifyErrorHandlers(error);\n reject(this.errorHandler.reject(error));\n }, timeoutMs);\n\n this.pendingRequests.set(id, {\n resolve: (value) => {\n clearTimeout(timeout);\n resolve(value);\n },\n reject: (error) => {\n clearTimeout(timeout);\n reject(error);\n },\n timestamp: Date.now(),\n timeout,\n });\n\n // Send via appropriate channel\n this.sendMessage(message);\n });\n }\n\n /**\n * Broadcast message to all contexts.\n * Used for state synchronization.\n */\n broadcast<T extends TMessage>(payload: T): void {\n const message: RoutedMessage<T> = {\n id: crypto.randomUUID(),\n source: this.context,\n targets: ALL_CONTEXTS,\n timestamp: Date.now(),\n payload,\n };\n\n this.sendMessage(message);\n }\n\n /**\n * Register a typed message handler.\n * Handler signature is enforced based on message type.\n * Multiple handlers can be registered for the same message type.\n */\n on<T extends TMessage[\"type\"]>(\n type: T,\n handler: (\n payload: Extract<TMessage, { type: T }>,\n message: RoutedMessage<Extract<TMessage, { type: T }>>\n ) =>\n | Promise<MessageResponse<Extract<TMessage, { type: T }>>>\n | MessageResponse<Extract<TMessage, { type: T }>>\n ): void {\n // Store handler with runtime type safety\n // TypeScript can't verify cross-boundary type safety through the Map storage,\n // but the .on() signature ensures the handler matches the message type\n const existing = this.handlers.get(type) || [];\n existing.push(handler);\n this.handlers.set(type, existing);\n }\n\n /**\n * Register multiple message handlers at once.\n * Reduces boilerplate when defining many handlers.\n *\n * @example\n * ```typescript\n * bus.registerHandlers({\n * 'MY_MESSAGE': async (payload) => ({ success: true }),\n * 'ANOTHER_MESSAGE': async (payload) => ({ data: payload }),\n * })\n * ```\n */\n // biome-ignore lint/complexity/noBannedTypes: Need to accept user-defined message types beyond ExtensionMessage\n registerHandlers(handlers: Record<string, Function | undefined>): void {\n for (const [type, handler] of Object.entries(handlers)) {\n if (handler) {\n const existing = this.handlers.get(type) || [];\n existing.push(handler);\n this.handlers.set(type, existing);\n }\n }\n }\n\n /**\n * Register a global error handler.\n * Called when errors occur during message handling.\n *\n * @example\n * ```typescript\n * bus.onError((error, bus) => {\n * console.error(`[${bus.context}] Error:`, error)\n * // Report to error tracking service\n * })\n * ```\n */\n onError(handler: (error: Error, bus: MessageBus<TMessage>) => void): void {\n this.userErrorHandlers.push(handler);\n }\n\n /**\n * Register a state accessor for runtime constraint checking.\n * The accessor function should return the current state object that constraints will check against.\n *\n * @param accessor - Function that returns the current state object\n *\n * @example\n * ```typescript\n * const state = { loggedIn: false };\n * bus.setStateAccessor(() => state);\n *\n * // Now constraints can be checked against this state\n * $constraints(\"loggedIn\", {\n * USER_LOGOUT: {\n * requires: (s) => s.loggedIn === true,\n * message: \"Must be logged in\"\n * }\n * }, { runtime: true });\n * ```\n */\n setStateAccessor(accessor: () => unknown): void {\n this.stateAccessor = accessor;\n }\n\n /**\n * Send message to background context.\n * Explicit routing API for better DX.\n *\n * @example\n * ```typescript\n * const result = await bus.sendToBackground({ type: 'GET_SETTINGS' })\n * ```\n */\n async sendToBackground<T extends TMessage>(\n payload: T,\n options?: { timeout?: number }\n ): Promise<\n | MessageResponse<Extract<TMessage, { type: T extends { type: infer TType } ? TType : never }>>\n | undefined\n > {\n return this.send(payload, { ...options, target: \"background\" });\n }\n\n /**\n * Send message to a specific content script.\n *\n * @example\n * ```typescript\n * const result = await bus.sendToContentScript(tabId, { type: 'ANALYZE_PAGE' })\n * ```\n */\n async sendToContentScript<T extends TMessage>(\n tabId: number,\n payload: T,\n options?: { timeout?: number }\n ): Promise<\n | MessageResponse<Extract<TMessage, { type: T extends { type: infer TType } ? TType : never }>>\n | undefined\n > {\n return this.send(payload, { ...options, target: \"content\", tabId });\n }\n\n /**\n * Send message to all tabs.\n * Useful for broadcasting updates to all content scripts.\n *\n * @example\n * ```typescript\n * await bus.sendToAllTabs({ type: 'REFRESH_UI' })\n * ```\n */\n async sendToAllTabs<T extends TMessage>(\n payload: T,\n options?: { timeout?: number }\n ): Promise<\n Array<\n | MessageResponse<\n Extract<TMessage, { type: T extends { type: infer TType } ? TType : never }>\n >\n | undefined\n >\n > {\n const tabs = await this.adapters.tabs.query({});\n return Promise.all(\n tabs.map((tab) =>\n tab.id ? this.sendToContentScript(tab.id, payload, options) : Promise.resolve(undefined)\n )\n );\n }\n\n /**\n * Send message to popup context.\n *\n * @example\n * ```typescript\n * await bus.sendToPopup({ type: 'UPDATE_UI', data: newData })\n * ```\n */\n async sendToPopup<T extends TMessage>(\n payload: T,\n options?: { timeout?: number }\n ): Promise<\n | MessageResponse<Extract<TMessage, { type: T extends { type: infer TType } ? TType : never }>>\n | undefined\n > {\n return this.send(payload, { ...options, target: \"popup\" });\n }\n\n /**\n * Send message to options page.\n *\n * @example\n * ```typescript\n * await bus.sendToOptions({ type: 'SETTINGS_UPDATED' })\n * ```\n */\n async sendToOptions<T extends TMessage>(\n payload: T,\n options?: { timeout?: number }\n ): Promise<\n | MessageResponse<Extract<TMessage, { type: T extends { type: infer TType } ? TType : never }>>\n | undefined\n > {\n return this.send(payload, { ...options, target: \"options\" });\n }\n\n /**\n * Send message to devtools panel.\n *\n * @example\n * ```typescript\n * await bus.sendToDevTools({ type: 'INSPECTION_DATA', data: pageData })\n * ```\n */\n async sendToDevTools<T extends TMessage>(\n payload: T,\n options?: { timeout?: number }\n ): Promise<\n | MessageResponse<Extract<TMessage, { type: T extends { type: infer TType } ? TType : never }>>\n | undefined\n > {\n return this.send(payload, { ...options, target: \"devtools\" });\n }\n\n /**\n * Send message to side panel.\n *\n * @example\n * ```typescript\n * await bus.sendToSidePanel({ type: 'UPDATE_ACTIVITY_LOG' })\n * ```\n */\n async sendToSidePanel<T extends TMessage>(\n payload: T,\n options?: { timeout?: number }\n ): Promise<\n | MessageResponse<Extract<TMessage, { type: T extends { type: infer TType } ? TType : never }>>\n | undefined\n > {\n return this.send(payload, { ...options, target: \"sidepanel\" });\n }\n\n /**\n * Connect with long-lived port.\n * Used for persistent connections (DevTools, Content Scripts).\n */\n connect(name: string): void {\n if (this.port) {\n console.warn(`[${this.context}] Port already connected: ${this.port.name}`);\n return;\n }\n\n this.port = this.adapters.runtime.connect(name);\n\n this.port.onMessage((message: unknown) => {\n if (isRoutedMessage<TMessage>(message) || isRoutedResponse<TMessage>(message)) {\n this.handleMessage(message);\n }\n });\n\n this.port.onDisconnect(() => {\n this.adapters.logger.warn(\"Port disconnected\", {\n context: this.context,\n portName: name,\n });\n this.port = null;\n\n // Reject all pending requests\n for (const [id, pending] of this.pendingRequests.entries()) {\n const error = new ConnectionError(\"Port disconnected\", {\n context: this.context,\n portName: name,\n requestId: id,\n });\n this.notifyErrorHandlers(error);\n pending.reject(this.errorHandler.reject(error));\n clearTimeout(pending.timeout);\n this.pendingRequests.delete(id);\n }\n });\n }\n\n /**\n * Disconnect port if connected.\n */\n disconnect(): void {\n if (this.port) {\n this.port.disconnect();\n this.port = null;\n }\n }\n\n /**\n * Remove all handlers and clean up.\n */\n destroy(): void {\n this.disconnect();\n this.handlers.clear();\n\n // Clear all pending requests\n for (const pending of this.pendingRequests.values()) {\n clearTimeout(pending.timeout);\n }\n this.pendingRequests.clear();\n\n // Remove message listener to prevent leaks\n if (this.messageListener) {\n this.adapters.runtime.removeMessageListener(this.messageListener);\n }\n }\n\n private setupListeners(): void {\n // Listen for one-off messages via chrome.runtime.sendMessage\n this.messageListener = (\n message: unknown,\n sender: unknown,\n sendResponse: (response: unknown) => void\n ) => {\n if (isRoutedMessage<TMessage>(message) || isRoutedResponse<TMessage>(message)) {\n this.handleMessage(message, sender)\n .then((response) => sendResponse(response))\n .catch((error) => {\n sendResponse({ success: false, error: error.message });\n });\n }\n return true; // Indicates async response\n };\n this.adapters.runtime.onMessage(this.messageListener);\n\n // Content/Page script window messaging\n if (this.context === \"content\" || this.context === \"page\") {\n this.adapters.window.addEventListener(\"message\", (event: MessageEvent) => {\n if (event.source !== window) return;\n if (event.data?.__extensionMessage) {\n this.handleMessage(event.data.message);\n }\n });\n }\n }\n\n // biome-ignore lint/complexity/noExcessiveCognitiveComplexity: Message handling requires routing logic for different message types\n public async handleMessage(\n message: RoutedMessage<TMessage> | RoutedResponse<TMessage>,\n _sender?: unknown\n ): Promise<unknown> {\n // Handle response to our request\n if (\"success\" in message) {\n const pending = this.pendingRequests.get(message.id);\n if (pending) {\n this.pendingRequests.delete(message.id);\n clearTimeout(pending.timeout);\n\n if (message.success) {\n // Message data is typed as MessageResponse<TMessage> from RoutedResponse\n pending.resolve(message.data ?? undefined);\n } else {\n const error = new HandlerError(message.error || \"Unknown error\", \"unknown\", {\n messageId: message.id,\n });\n this.notifyErrorHandlers(error);\n pending.reject(this.errorHandler.reject(error));\n }\n }\n return;\n }\n\n // Ignore messages not targeted at us\n if (!message.targets.includes(this.context)) {\n // If we're background, we need to route it\n if (this.context === \"background\") {\n return; // Routing handled elsewhere\n }\n return;\n }\n\n // Handle incoming request\n const handlers = this.handlers.get(message.payload.type);\n if (!handlers || handlers.length === 0) {\n // For multi-target messages, don't warn if we have no handler\n if (message.targets.length === 1) {\n console.warn(`[${this.context}] No handler for message type: ${message.payload.type}`);\n }\n return { success: false, error: \"No handler\" };\n }\n\n // For multi-target messages (including broadcasts), call all handlers (don't send responses)\n if (message.targets.length > 1) {\n try {\n // Track execution to detect double-handler invocation\n globalExecutionTracker.track(message.id, message.payload.type);\n\n await Promise.all(handlers.map((handler) => handler(message.payload, message)));\n return { success: true, data: undefined, timestamp: Date.now() };\n } catch (error) {\n return {\n success: false,\n error: error instanceof Error ? error.message : \"Unknown error\",\n timestamp: Date.now(),\n };\n }\n }\n\n // For LOG messages, call all handlers but still send response (for backwards compat)\n if (message.payload.type === \"LOG\") {\n try {\n // Track execution to detect double-handler invocation\n globalExecutionTracker.track(message.id, message.payload.type);\n\n await Promise.all(handlers.map((handler) => handler(message.payload, message)));\n const response: RoutedResponse<TMessage> = {\n id: message.id,\n success: true,\n timestamp: Date.now(),\n };\n this.sendResponse(message, response);\n return response;\n } catch (error) {\n const response: RoutedResponse<TMessage> = {\n id: message.id,\n success: false,\n error: error instanceof Error ? error.message : \"Unknown error\",\n timestamp: Date.now(),\n };\n this.sendResponse(message, response);\n return response;\n }\n }\n\n // For other targeted messages, call first handler and send response\n try {\n // Track execution to detect double-handler invocation\n globalExecutionTracker.track(message.id, message.payload.type);\n\n // Check runtime constraints before handler execution\n if (this.stateAccessor) {\n try {\n const { checkPreconditions, isRuntimeConstraintsEnabled } = await import(\n \"./constraints.js\"\n );\n if (isRuntimeConstraintsEnabled()) {\n const currentState = this.stateAccessor();\n checkPreconditions(message.payload.type, currentState);\n }\n } catch (error) {\n // If constraint check fails, throw immediately\n if (error instanceof Error) {\n throw error;\n }\n // If import fails, log warning but continue (constraints not available)\n if (\n error &&\n typeof error === \"object\" &&\n \"code\" in error &&\n error.code === \"MODULE_NOT_FOUND\"\n ) {\n // Constraints module not available - continue without checking\n } else {\n throw error;\n }\n }\n }\n\n // We've already checked handlers.length > 0 above, so handlers[0] exists\n const handler = handlers[0];\n if (!handler) {\n throw new Error(`Handler not found for ${message.payload.type}`);\n }\n const data = await handler(message.payload, message);\n\n // Check postconditions after handler execution (optional)\n if (this.stateAccessor) {\n try {\n const { checkPostconditions, isRuntimeConstraintsEnabled } = await import(\n \"./constraints.js\"\n );\n if (isRuntimeConstraintsEnabled()) {\n const currentState = this.stateAccessor();\n checkPostconditions(message.payload.type, currentState);\n }\n } catch (error) {\n // Postcondition failures should be logged but not block response\n if (error instanceof Error && error.message.includes(\"Postcondition\")) {\n console.error(`[${this.context}] Postcondition failed:`, error.message);\n // Continue - postcondition failures don't block the response\n }\n }\n }\n\n const response: RoutedResponse<TMessage> = {\n id: message.id,\n success: true,\n data,\n timestamp: Date.now(),\n };\n\n this.sendResponse(message, response);\n return response;\n } catch (error) {\n const response: RoutedResponse<TMessage> = {\n id: message.id,\n success: false,\n error: error instanceof Error ? error.message : \"Unknown error\",\n timestamp: Date.now(),\n };\n\n this.sendResponse(message, response);\n return response;\n }\n }\n\n public sendMessage<T extends TMessage = TMessage>(message: RoutedMessage<T>): void {\n if (this.context === \"content\" && message.targets.includes(\"page\")) {\n // Content → Page via window.postMessage\n this.adapters.window.postMessage({ __extensionMessage: true, message }, \"*\");\n } else if (this.context === \"page\") {\n // Page → Content via window.postMessage\n this.adapters.window.postMessage({ __extensionMessage: true, message }, \"*\");\n } else if (this.port) {\n // Use long-lived port if connected (devtools, content, popup, options)\n this.port.postMessage(message);\n } else {\n // Use chrome.runtime.sendMessage (fallback for unconnected contexts)\n this.adapters.runtime.sendMessage(message);\n }\n }\n\n private sendResponse(request: RoutedMessage<TMessage>, response: RoutedResponse<TMessage>): void {\n if (this.context === \"content\" && request.source === \"page\") {\n // Content → Page response\n this.adapters.window.postMessage({ __extensionMessage: true, message: response }, \"*\");\n } else if (this.context === \"page\" && request.source === \"content\") {\n // Page → Content response\n this.adapters.window.postMessage({ __extensionMessage: true, message: response }, \"*\");\n } else if (this.port && (this.context === \"devtools\" || this.context === \"content\")) {\n // Use port for response\n this.port.postMessage(response);\n } else {\n // Use chrome.runtime.sendMessage\n this.adapters.runtime.sendMessage(response);\n }\n }\n\n private inferTarget(type: string): Context | Context[] | undefined {\n const handlers = {\n DOM_QUERY: \"content\",\n DOM_UPDATE: \"content\",\n DOM_INSERT: \"content\",\n DOM_REMOVE: \"content\",\n PAGE_EVAL: \"page\",\n PAGE_GET_VAR: \"page\",\n PAGE_CALL_FN: \"page\",\n PAGE_SET_VAR: \"page\",\n API_REQUEST: \"background\",\n API_BATCH: \"background\",\n CLIPBOARD_WRITE: \"offscreen\",\n CLIPBOARD_WRITE_HTML: \"offscreen\",\n CLIPBOARD_WRITE_RICH: \"offscreen\",\n CLIPBOARD_READ: \"offscreen\",\n CONTEXT_MENU_CREATE: \"background\",\n CONTEXT_MENU_REMOVE: \"background\",\n STATE_SYNC: ALL_CONTEXTS,\n TAB_QUERY: \"background\",\n TAB_GET_CURRENT: \"background\",\n TAB_RELOAD: \"background\",\n LOG: \"background\",\n LOGS_GET: \"background\",\n LOGS_CLEAR: \"background\",\n LOGS_EXPORT: \"background\",\n } as const;\n\n // Helper type guard: narrows string to a key of the object\n function isHandlerKey(key: string): key is keyof typeof handlers {\n return key in handlers;\n }\n\n // Type guard narrows string to known keys\n if (isHandlerKey(type)) {\n // TypeScript now knows type is keyof typeof handlers - no cast needed!\n return handlers[type];\n }\n\n // Unknown message type - caller must provide explicit target\n return undefined;\n }\n\n /**\n * Create context-specific helpers based on current context.\n * @private\n */\n private createContextHelpers():\n | ContentScriptHelpers\n | DevToolsHelpers\n | PopupHelpers\n | OptionsHelpers\n | SidePanelHelpers\n | BackgroundHelpers\n | Record<string, never> {\n switch (this.context) {\n case \"content\":\n return createContentScriptHelpers();\n case \"devtools\":\n return createDevToolsHelpers();\n case \"popup\":\n return createPopupHelpers(this.adapters);\n case \"options\":\n return createOptionsHelpers(this.adapters);\n case \"sidepanel\":\n return createSidePanelHelpers(this.adapters);\n case \"background\":\n return createBackgroundHelpers(this.adapters);\n default:\n return {};\n }\n }\n\n /**\n * Notify all registered error handlers.\n * @private\n */\n private notifyErrorHandlers(error: Error): void {\n for (const handler of this.userErrorHandlers) {\n try {\n handler(error, this);\n } catch (handlerError) {\n console.error(`[${this.context}] Error in error handler:`, handlerError);\n }\n }\n }\n}\n\n/**\n * Create a MessageBus for the given context.\n *\n * IMPORTANT: Only call this ONCE per context in your application.\n * Calling it multiple times will create multiple message listeners, causing\n * handlers to execute multiple times. Store the returned bus and reuse it.\n *\n * For background scripts, use createBackground() instead.\n */\nexport function getMessageBus<TMessage extends BaseMessage = ExtensionMessage>(\n context: Context,\n adapters?: ExtensionAdapters,\n options?: { skipListenerSetup?: boolean }\n): MessageBus<TMessage> {\n return new MessageBus<TMessage>(context, adapters, options);\n}\n",
20
- "// State primitives with optional sync and persistence\n\nimport { effect, type Signal, signal } from \"@preact/signals\";\nimport type { Context } from \"../types/messages\";\nimport { getMessageBus, type MessageBus } from \"./message-bus\";\n\n/**\n * Signal extended with .loaded promise for hydration control\n */\ntype SignalWithLoaded<T> = Signal<T> & { loaded: Promise<void> };\n\n/**\n * Signal extended with .loaded and .verify properties for verification tracking\n */\ntype SignalWithVerify<T> = Signal<T> & { loaded: Promise<void>; verify: T };\n\ntype StateEntry<T> = {\n signal: Signal<T>;\n clock: number; // Lamport clock for causal ordering\n loaded: Promise<void>;\n updating: boolean;\n};\n\ntype StateOptions<T = unknown> = {\n bus?: MessageBus; // Custom message bus (for testing)\n debounceMs?: number; // Debounce storage writes\n validator?: (value: unknown) => value is T; // Runtime type validation\n verify?: boolean; // Enable verification tracking (creates plain object mirror)\n};\n\nconst stateRegistry = new Map<string, StateEntry<unknown>>();\n\n// biome-ignore lint/complexity/noExcessiveCognitiveComplexity: Context detection requires multiple checks\nfunction getCurrentContext(): Context {\n // Detect current context\n if (typeof chrome !== \"undefined\") {\n if (typeof chrome.devtools !== \"undefined\") {\n return \"devtools\";\n }\n if (typeof chrome.runtime !== \"undefined\") {\n try {\n if (typeof self !== \"undefined\" && \"ServiceWorkerGlobalScope\" in self) {\n return \"background\";\n }\n // biome-ignore lint/suspicious/noEmptyBlockStatements: ServiceWorkerGlobalScope check may fail in some contexts\n } catch {}\n }\n }\n\n if (typeof window !== \"undefined\" && typeof document !== \"undefined\") {\n if (typeof chrome === \"undefined\" || typeof chrome.runtime === \"undefined\") {\n return \"page\";\n }\n\n // Distinguish popup/options/offscreen from content scripts by URL\n if (\n typeof window.location !== \"undefined\" &&\n window.location.protocol === \"chrome-extension:\"\n ) {\n const path = window.location.pathname;\n if (path.includes(\"/popup\")) return \"popup\";\n if (path.includes(\"/options\")) return \"options\";\n if (path.includes(\"/offscreen\")) return \"offscreen\";\n // Could also be a custom extension page, default to content\n }\n\n return \"content\";\n }\n\n return \"background\";\n}\n\n/**\n * Shared state: synced across all contexts AND persisted to storage\n *\n * Uses Lamport clock for conflict resolution. State is automatically:\n * - Loaded from chrome.storage on initialization\n * - Synced to other contexts via broadcast messages\n * - Persisted to chrome.storage on every change\n *\n * Available in: background, popup, options, devtools, content scripts\n * ⚠️ NOT available in page scripts (use content script state instead)\n *\n * @param key - Unique identifier for this state (e.g., \"app-settings\")\n * @param initialValue - Default value if nothing is in storage\n * @param options - Optional configuration (bus, debounceMs)\n * @returns Reactive signal that stays in sync across all contexts\n *\n * @example\n * ```typescript\n * // Define once, use everywhere\n * const settings = $sharedState(\"settings\", { theme: \"dark\" })\n *\n * // Changes automatically sync\n * settings.value = { theme: \"light\" }\n * ```\n */\nexport function $sharedState<T>(\n key: string,\n initialValue: T,\n options: StateOptions<T> = {}\n): Signal<T> & { loaded: Promise<void>; verify?: T } {\n const sig = createState(key, initialValue, {\n ...options,\n sync: true,\n persist: true,\n });\n\n // Expose loaded promise for awaiting hydration\n const entry = stateRegistry.get(key);\n if (entry) {\n (sig as SignalWithLoaded<T>).loaded = entry.loaded;\n }\n\n return sig as Signal<T> & { loaded: Promise<void> };\n}\n\n/**\n * Synced state: synced across all contexts but NOT persisted\n *\n * State is broadcast to all contexts in real-time but resets on extension reload.\n *\n * Available in: background, popup, options, devtools, content scripts\n * ⚠️ NOT available in page scripts\n *\n * @param key - Unique identifier for this state\n * @param initialValue - Default value\n * @param options - Optional configuration\n * @returns Reactive signal synced across contexts (but not persisted)\n *\n * @example\n * ```typescript\n * // Temporary shared state\n * const activeTabId = $syncedState(\"active-tab\", null)\n * ```\n */\nexport function $syncedState<T>(\n key: string,\n initialValue: T,\n options: StateOptions<T> = {}\n): Signal<T> {\n return createState(key, initialValue, {\n ...options,\n sync: true,\n persist: false,\n });\n}\n\n/**\n * Persisted state: persisted to storage but NOT synced across contexts\n *\n * Each context has its own copy of the state, persisted independently.\n *\n * Available in: background, popup, options, devtools, content scripts\n * ⚠️ NOT available in page scripts\n *\n * @param key - Unique identifier (use prefix like \"popup:state\" to avoid collisions)\n * @param initialValue - Default value\n * @param options - Optional configuration\n * @returns Reactive signal persisted to storage (but not synced)\n *\n * @example\n * ```typescript\n * // Each context has its own persisted state\n * const popupState = $persistedState(\"popup:last-panel\", \"home\")\n * const devtoolsState = $persistedState(\"devtools:expanded\", true)\n * ```\n */\nexport function $persistedState<T>(\n key: string,\n initialValue: T,\n options: StateOptions<T> = {}\n): Signal<T> & { loaded: Promise<void> } {\n const sig = createState(key, initialValue, {\n ...options,\n sync: false,\n persist: true,\n });\n\n // Expose loaded promise for awaiting hydration\n const entry = stateRegistry.get(key);\n if (entry) {\n (sig as SignalWithLoaded<T>).loaded = entry.loaded;\n }\n\n return sig as Signal<T> & { loaded: Promise<void> };\n}\n\n/**\n * Local state: not synced, not persisted (like regular Preact signal)\n *\n * Simple reactive state that lives only in the current context.\n * Resets on reload or context restart.\n *\n * Available in: all contexts (including page scripts)\n *\n * @param initialValue - Default value\n * @returns Reactive signal (local only)\n *\n * @example\n * ```typescript\n * // Local UI state\n * const isLoading = $state(false)\n * const error = $state<string | null>(null)\n * ```\n */\nexport function $state<T>(initialValue: T): Signal<T> {\n return signal(initialValue);\n}\n\ntype InternalStateOptions<T = unknown> = StateOptions<T> & {\n sync: boolean;\n persist: boolean;\n};\n\n// Deep equality check to prevent redundant updates\nfunction deepEqual(a: unknown, b: unknown): boolean {\n if (a === b) return true;\n if (a == null || b == null) return false;\n if (typeof a !== \"object\" || typeof b !== \"object\") return false;\n\n const keysA = Object.keys(a);\n const keysB = Object.keys(b);\n\n if (keysA.length !== keysB.length) return false;\n\n for (const key of keysA) {\n if (!keysB.includes(key)) return false;\n if (!deepEqual((a as Record<string, unknown>)[key], (b as Record<string, unknown>)[key]))\n return false;\n }\n\n return true;\n}\n\n// biome-ignore lint/complexity/noExcessiveCognitiveComplexity: Core state management logic requires coordination of multiple concerns\nfunction createState<T>(key: string, initialValue: T, options: InternalStateOptions<T>): Signal<T> {\n // Return existing signal if already registered\n if (stateRegistry.has(key)) {\n return stateRegistry.get(key)?.signal as Signal<T>;\n }\n\n const currentContext = getCurrentContext();\n\n // Page scripts should not have stateful operations\n // They are execution contexts for content scripts, not independent contexts\n if (currentContext === \"page\" && (options.sync || options.persist)) {\n const stateFn =\n options.sync && options.persist\n ? \"$sharedState\"\n : options.persist\n ? \"$persistedState\"\n : \"$syncedState\";\n\n throw new Error(\n `[web-ext] ${stateFn}() is not available in page context.\\nPage scripts are execution contexts for content scripts and should not maintain state.\\nUse state in the content script instead, or use $state() for local-only state.`\n );\n }\n\n const sig = signal(initialValue);\n\n // Create verification mirror if requested\n if (options.verify) {\n // Create plain object mirror for verification\n const mirror = JSON.parse(JSON.stringify(initialValue)) as T;\n (sig as SignalWithVerify<T>).verify = mirror;\n }\n\n const entry: StateEntry<T> = {\n signal: sig,\n clock: 0,\n loaded: Promise.resolve(),\n updating: false,\n };\n\n // Lazy bus initialization\n let bus: MessageBus | null = null;\n const getBus = () => {\n if (!bus && typeof chrome !== \"undefined\") {\n bus = options.bus || getMessageBus(currentContext);\n }\n return bus;\n };\n\n // Load from storage if persist is enabled\n if (options.persist) {\n entry.loaded = loadFromStorage(key, sig, entry, getBus(), options.validator);\n }\n\n // Watch for changes after initial load\n entry.loaded.then(() => {\n let debounceTimer: NodeJS.Timeout | null = null;\n let previousValue = sig.value;\n let isFirstRun = true;\n\n // biome-ignore lint/complexity/noExcessiveCognitiveComplexity: Sync effect requires coordination of multiple state change scenarios\n effect(() => {\n // Skip if update in progress (from incoming message)\n if (entry.updating) return;\n\n const value = sig.value;\n\n // Skip first run (effect fires immediately on registration)\n if (isFirstRun) {\n isFirstRun = false;\n return;\n }\n\n // Skip if value hasn't changed (deep equality check)\n if (deepEqual(value, previousValue)) {\n return;\n }\n\n previousValue = value;\n\n // Update verification mirror if enabled\n if (options.verify) {\n const verifySignal = sig as SignalWithVerify<T>;\n if (verifySignal.verify) {\n Object.assign(verifySignal.verify, JSON.parse(JSON.stringify(value)));\n }\n }\n\n // Increment clock monotonically\n entry.clock++;\n\n const doUpdate = () => {\n // Persist to storage\n if (options.persist) {\n persistToStorage(key, value, entry.clock, getBus());\n }\n\n // Broadcast to other contexts\n if (options.sync) {\n broadcastUpdate(key, value, entry.clock, getBus());\n }\n };\n\n // Debounce if specified\n if (options.debounceMs) {\n if (debounceTimer) clearTimeout(debounceTimer);\n debounceTimer = setTimeout(doUpdate, options.debounceMs);\n } else {\n doUpdate();\n }\n });\n });\n\n // Listen for updates from other contexts (only if sync enabled)\n if (options.sync) {\n const messageBus = getBus();\n if (messageBus) {\n // Auto-connect for contexts that need it (popup, options, devtools)\n // Safe to call multiple times because:\n // - getMessageBus() returns a singleton per context\n // - connect() is idempotent (checks if port exists and returns early)\n // Background doesn't need to connect (it's the hub)\n // Content scripts connect explicitly when needed\n if (\n currentContext === \"popup\" ||\n currentContext === \"options\" ||\n currentContext === \"devtools\"\n ) {\n messageBus.connect(currentContext);\n }\n\n // biome-ignore lint/complexity/noExcessiveCognitiveComplexity: State sync requires validation and conflict resolution\n messageBus.on(\"STATE_SYNC\", async (payload) => {\n if (payload.key !== key) return undefined;\n\n const oldClock = entry.clock;\n\n // Lamport clock rule: Always update to max(local, received)\n // This maintains causal ordering even when rejecting updates\n entry.clock = Math.max(entry.clock, payload.clock);\n\n // Only accept value updates if received clock is strictly greater than old local clock\n // This ensures we only apply causally newer updates\n if (payload.clock > oldClock) {\n // Validate incoming value if validator provided\n if (options.validator && !options.validator(payload.value)) {\n console.warn(\n `[web-ext] State \"${key}\": Received invalid value from sync (clock: ${payload.clock})`,\n payload.value\n );\n return undefined;\n }\n\n // Skip redundant updates (deep equality check)\n if (deepEqual(entry.signal.value, payload.value)) {\n return undefined;\n }\n\n applyUpdate(entry, payload.value as T, payload.clock);\n }\n\n return undefined;\n });\n }\n }\n\n stateRegistry.set(key, entry as StateEntry<unknown>);\n return sig;\n}\n\nasync function loadFromStorage<T>(\n key: string,\n sig: Signal<T>,\n entry: StateEntry<T>,\n bus: MessageBus | null,\n validator?: (value: unknown) => value is T\n): Promise<void> {\n if (!bus) return;\n\n try {\n const result = await bus.adapters.storage.get([key, `${key}:clock`]);\n\n if (result[key] !== undefined) {\n const storedValue = result[key];\n\n // Validate stored value if validator provided\n if (validator) {\n if (validator(storedValue)) {\n sig.value = storedValue;\n } else {\n console.warn(\n `[web-ext] State \"${key}\": Stored value failed validation, using initial value`,\n storedValue\n );\n }\n } else {\n sig.value = storedValue as T;\n }\n }\n\n if (result[`${key}:clock`] !== undefined) {\n entry.clock = result[`${key}:clock`] as number;\n }\n } catch (error) {\n console.warn(`[web-ext] Failed to load state from storage: ${key}`, error);\n }\n}\n\nfunction persistToStorage<T>(key: string, value: T, clock: number, bus: MessageBus | null): void {\n if (!bus) return;\n\n try {\n bus.adapters.storage.set({\n [key]: value,\n [`${key}:clock`]: clock,\n });\n } catch (error) {\n console.warn(`Failed to persist state to storage: ${key}`, error);\n }\n}\n\nfunction broadcastUpdate<T>(key: string, value: T, clock: number, bus: MessageBus | null): void {\n if (!bus) return;\n\n try {\n bus.broadcast({\n type: \"STATE_SYNC\",\n key,\n value,\n clock,\n });\n } catch (error) {\n console.warn(`Failed to broadcast state update: ${key}`, error);\n }\n}\n\nfunction applyUpdate<T>(entry: StateEntry<T>, value: T, clock: number): void {\n entry.updating = true;\n entry.signal.value = value;\n entry.clock = clock;\n entry.updating = false;\n}\n\n/**\n * Get state by key (useful for retrieving state without re-creating)\n */\nexport function getStateByKey<T>(key: string): Signal<T> | undefined {\n const entry = stateRegistry.get(key);\n return entry?.signal as Signal<T> | undefined;\n}\n\n/**\n * Clear state registry (useful for testing)\n */\nexport function clearStateRegistry(): void {\n stateRegistry.clear();\n}\n"
5
+ "// Storage adapters for different execution contexts\n// Automatically chooses the right storage mechanism based on environment\n\n/**\n * Universal storage adapter interface\n */\nexport interface StorageAdapter {\n get<T = unknown>(keys: string[]): Promise<Record<string, T>>;\n set(items: Record<string, unknown>): Promise<void>;\n remove(keys: string[]): Promise<void>;\n}\n\n/**\n * IndexedDB adapter for web apps\n */\nexport class IndexedDBAdapter implements StorageAdapter {\n private dbName: string;\n private storeName = \"state\";\n private dbPromise: Promise<IDBDatabase> | null = null;\n\n constructor(dbName = \"polly-state\") {\n this.dbName = dbName;\n }\n\n private getDB(): Promise<IDBDatabase> {\n if (this.dbPromise) return this.dbPromise;\n\n this.dbPromise = new Promise((resolve, reject) => {\n const request = indexedDB.open(this.dbName, 1);\n\n request.onerror = () => reject(request.error);\n request.onsuccess = () => resolve(request.result);\n\n request.onupgradeneeded = (event) => {\n const db = (event.target as IDBOpenDBRequest).result;\n if (!db.objectStoreNames.contains(this.storeName)) {\n db.createObjectStore(this.storeName);\n }\n };\n });\n\n return this.dbPromise;\n }\n\n async get<T = unknown>(keys: string[]): Promise<Record<string, T>> {\n try {\n const db = await this.getDB();\n const result: Record<string, T> = {};\n\n await Promise.all(\n keys.map(\n (key) =>\n new Promise<void>((resolve, reject) => {\n const transaction = db.transaction([this.storeName], \"readonly\");\n const store = transaction.objectStore(this.storeName);\n const request = store.get(key);\n\n request.onerror = () => reject(request.error);\n request.onsuccess = () => {\n if (request.result !== undefined) {\n result[key] = request.result as T;\n }\n resolve();\n };\n })\n )\n );\n\n return result;\n } catch (error) {\n console.warn(\"[Polly] IndexedDB get failed:\", error);\n return {};\n }\n }\n\n async set(items: Record<string, unknown>): Promise<void> {\n try {\n const db = await this.getDB();\n await Promise.all(\n Object.entries(items).map(\n ([key, value]) =>\n new Promise<void>((resolve, reject) => {\n const transaction = db.transaction([this.storeName], \"readwrite\");\n const store = transaction.objectStore(this.storeName);\n const request = store.put(value, key);\n\n request.onerror = () => reject(request.error);\n request.onsuccess = () => resolve();\n })\n )\n );\n } catch (error) {\n console.warn(\"[Polly] IndexedDB set failed:\", error);\n }\n }\n\n async remove(keys: string[]): Promise<void> {\n try {\n const db = await this.getDB();\n await Promise.all(\n keys.map(\n (key) =>\n new Promise<void>((resolve, reject) => {\n const transaction = db.transaction([this.storeName], \"readwrite\");\n const store = transaction.objectStore(this.storeName);\n const request = store.delete(key);\n\n request.onerror = () => reject(request.error);\n request.onsuccess = () => resolve();\n })\n )\n );\n } catch (error) {\n console.warn(\"[Polly] IndexedDB remove failed:\", error);\n }\n }\n}\n\n/**\n * Chrome storage adapter for extensions\n */\nexport class ChromeStorageAdapter implements StorageAdapter {\n async get<T = unknown>(keys: string[]): Promise<Record<string, T>> {\n if (typeof chrome === \"undefined\" || !chrome.storage) {\n return {};\n }\n\n try {\n return (await chrome.storage.local.get(keys)) as Record<string, T>;\n } catch (error) {\n console.warn(\"[Polly] Chrome storage get failed:\", error);\n return {};\n }\n }\n\n async set(items: Record<string, unknown>): Promise<void> {\n if (typeof chrome === \"undefined\" || !chrome.storage) {\n return;\n }\n\n try {\n await chrome.storage.local.set(items);\n } catch (error) {\n console.warn(\"[Polly] Chrome storage set failed:\", error);\n }\n }\n\n async remove(keys: string[]): Promise<void> {\n if (typeof chrome === \"undefined\" || !chrome.storage) {\n return;\n }\n\n try {\n await chrome.storage.local.remove(keys);\n } catch (error) {\n console.warn(\"[Polly] Chrome storage remove failed:\", error);\n }\n }\n}\n\n/**\n * In-memory adapter (no persistence) for testing or server contexts\n */\nexport class MemoryStorageAdapter implements StorageAdapter {\n private storage = new Map<string, unknown>();\n\n async get<T = unknown>(keys: string[]): Promise<Record<string, T>> {\n const result: Record<string, T> = {};\n for (const key of keys) {\n const value = this.storage.get(key);\n if (value !== undefined) {\n result[key] = value as T;\n }\n }\n return result;\n }\n\n async set(items: Record<string, unknown>): Promise<void> {\n for (const [key, value] of Object.entries(items)) {\n this.storage.set(key, value);\n }\n }\n\n async remove(keys: string[]): Promise<void> {\n for (const key of keys) {\n this.storage.delete(key);\n }\n }\n}\n\n/**\n * Detect execution context and return appropriate storage adapter\n */\nexport function createStorageAdapter(): StorageAdapter {\n // Chrome extension context\n if (typeof chrome !== \"undefined\" && chrome.storage && chrome.runtime) {\n return new ChromeStorageAdapter();\n }\n\n // Web app context (has IndexedDB)\n if (typeof indexedDB !== \"undefined\") {\n return new IndexedDBAdapter();\n }\n\n // Server/test context (no persistent storage available)\n return new MemoryStorageAdapter();\n}\n",
6
+ "// Sync adapter interface for cross-context state synchronization\n// Abstracts the transport mechanism (chrome.runtime, BroadcastChannel, etc.)\n//\n// Architecture Decision: BroadcastChannel vs SharedWorker\n// ------------------------------------------------------\n// We currently use BroadcastChannel for web app sync because:\n// - Simpler API with no lifecycle management complexity\n// - Decentralized (aligns with local-first/offline-first architecture)\n// - Better browser support (especially Safari and mobile)\n// - Perfect for message-passing with Lamport clock conflict resolution\n// - No single point of failure\n//\n// Future Consideration: SharedWorker Support\n// ------------------------------------------\n// SharedWorker could be added as an optional adapter for use cases requiring:\n// - Central coordination point for complex multi-tab workflows\n// - Shared WebSocket connections (one connection for all tabs)\n// - Heavy computation done once and shared across tabs\n// - Persistent background work when tabs are closed\n// - Transaction coordination across tabs\n//\n// For most Polly use cases, BroadcastChannel's peer-to-peer model is preferred,\n// but SharedWorker support could be valuable for advanced scenarios.\n\n/**\n * Message format for state synchronization\n */\nexport interface StateSyncMessage<T = unknown> {\n key: string;\n value: T;\n clock: number;\n}\n\n/**\n * Sync adapter interface - abstracts the transport mechanism for state sync\n *\n * Different contexts use different transports:\n * - Chrome extensions: chrome.runtime messaging\n * - Web apps (multi-tab): BroadcastChannel\n * - PWAs: BroadcastChannel + Service Worker messaging\n * - Single-context: NoOp (no sync needed)\n */\nexport interface SyncAdapter {\n /**\n * Broadcast a state update to other contexts\n */\n broadcast<T>(message: StateSyncMessage<T>): void;\n\n /**\n * Register a callback for incoming state updates\n */\n onMessage<T>(callback: (message: StateSyncMessage<T>) => void): () => void;\n\n /**\n * Optional: Connect to the sync mechanism\n * Some transports require explicit connection setup\n */\n connect?(): Promise<void>;\n\n /**\n * Optional: Disconnect from the sync mechanism\n */\n disconnect?(): void;\n\n /**\n * Optional: Check if connected\n */\n isConnected?(): boolean;\n}\n\n/**\n * NoOp sync adapter for single-context scenarios (no sync needed)\n */\nexport class NoOpSyncAdapter implements SyncAdapter {\n broadcast<T>(_message: StateSyncMessage<T>): void {\n // No-op: single context, no need to sync\n }\n\n onMessage<T>(_callback: (message: StateSyncMessage<T>) => void): () => void {\n // No-op: no messages will ever arrive\n return () => {\n // Empty cleanup function - nothing to clean up for null adapter\n };\n }\n}\n\n/**\n * Chrome runtime sync adapter for Chrome extensions\n * Uses chrome.runtime.sendMessage for cross-context messaging\n */\nexport class ChromeRuntimeSyncAdapter implements SyncAdapter {\n private listeners: Array<(message: StateSyncMessage<unknown>) => void> = [];\n private port: chrome.runtime.Port | null = null;\n\n constructor() {\n // Set up listener for incoming messages\n if (typeof chrome !== \"undefined\" && chrome.runtime) {\n chrome.runtime.onMessage.addListener((message, _sender, _sendResponse) => {\n if (message.type === \"STATE_SYNC\") {\n this.listeners.forEach((listener) => {\n listener(message);\n });\n }\n });\n }\n }\n\n broadcast<T>(message: StateSyncMessage<T>): void {\n if (typeof chrome === \"undefined\" || !chrome.runtime) {\n console.warn(\"[SyncAdapter] chrome.runtime not available\");\n return;\n }\n\n try {\n chrome.runtime.sendMessage({\n type: \"STATE_SYNC\",\n key: message.key,\n value: message.value,\n clock: message.clock,\n });\n } catch (error) {\n console.warn(\"[SyncAdapter] Failed to broadcast state update:\", error);\n }\n }\n\n onMessage<T>(callback: (message: StateSyncMessage<T>) => void): () => void {\n this.listeners.push(callback as (message: StateSyncMessage<unknown>) => void);\n\n // Return cleanup function\n return () => {\n const index = this.listeners.indexOf(\n callback as (message: StateSyncMessage<unknown>) => void\n );\n if (index > -1) {\n this.listeners.splice(index, 1);\n }\n };\n }\n\n connect(): Promise<void> {\n // Chrome runtime is always connected\n return Promise.resolve();\n }\n\n disconnect(): void {\n this.listeners = [];\n if (this.port) {\n this.port.disconnect();\n this.port = null;\n }\n }\n\n isConnected(): boolean {\n return typeof chrome !== \"undefined\" && !!chrome.runtime;\n }\n}\n\n/**\n * BroadcastChannel sync adapter for web apps (multi-tab)\n * Uses BroadcastChannel API for cross-tab messaging\n */\nexport class BroadcastChannelSyncAdapter implements SyncAdapter {\n private channel: BroadcastChannel | null = null;\n private listeners: Array<(message: StateSyncMessage<unknown>) => void> = [];\n\n constructor(channelName = \"polly-sync\") {\n if (typeof BroadcastChannel !== \"undefined\") {\n this.channel = new BroadcastChannel(channelName);\n\n this.channel.onmessage = (event) => {\n if (event.data.type === \"STATE_SYNC\") {\n this.listeners.forEach((listener) => {\n listener(event.data);\n });\n }\n };\n } else {\n console.warn(\"[SyncAdapter] BroadcastChannel not available\");\n }\n }\n\n broadcast<T>(message: StateSyncMessage<T>): void {\n if (!this.channel) {\n console.warn(\"[SyncAdapter] BroadcastChannel not initialized\");\n return;\n }\n\n try {\n this.channel.postMessage({\n type: \"STATE_SYNC\",\n key: message.key,\n value: message.value,\n clock: message.clock,\n });\n } catch (error) {\n console.warn(\"[SyncAdapter] Failed to broadcast state update:\", error);\n }\n }\n\n onMessage<T>(callback: (message: StateSyncMessage<T>) => void): () => void {\n this.listeners.push(callback as (message: StateSyncMessage<unknown>) => void);\n\n // Return cleanup function\n return () => {\n const index = this.listeners.indexOf(\n callback as (message: StateSyncMessage<unknown>) => void\n );\n if (index > -1) {\n this.listeners.splice(index, 1);\n }\n };\n }\n\n connect(): Promise<void> {\n // BroadcastChannel connects immediately on construction\n return Promise.resolve();\n }\n\n disconnect(): void {\n this.listeners = [];\n if (this.channel) {\n this.channel.close();\n this.channel = null;\n }\n }\n\n isConnected(): boolean {\n return this.channel !== null;\n }\n}\n\n/**\n * Detect available sync mechanisms and create appropriate adapter\n */\nexport function createSyncAdapter(): SyncAdapter {\n // Chrome extension context - use chrome.runtime\n if (typeof chrome !== \"undefined\" && chrome.runtime) {\n return new ChromeRuntimeSyncAdapter();\n }\n\n // Web app with multi-tab support - use BroadcastChannel\n if (typeof BroadcastChannel !== \"undefined\") {\n return new BroadcastChannelSyncAdapter();\n }\n\n // Single context or no sync available - use NoOp\n return new NoOpSyncAdapter();\n}\n",
7
+ "// State primitives with optional sync and persistence\n\nimport { effect, type Signal, signal } from \"@preact/signals\";\nimport type { MessageBus } from \"./message-bus\";\nimport { createStorageAdapter, type StorageAdapter } from \"./storage-adapter\";\nimport { createSyncAdapter, type SyncAdapter } from \"./sync-adapter\";\n\n/**\n * Signal extended with .loaded promise for hydration control\n */\ntype SignalWithLoaded<T> = Signal<T> & { loaded: Promise<void> };\n\n/**\n * Signal extended with .loaded and .verify properties for verification tracking\n */\ntype SignalWithVerify<T> = Signal<T> & { loaded: Promise<void>; verify: T };\n\ntype StateEntry<T> = {\n signal: Signal<T>;\n clock: number; // Lamport clock for causal ordering\n loaded: Promise<void>;\n updating: boolean;\n};\n\ntype StateOptions<T = unknown> = {\n // Legacy MessageBus support (deprecated, will be removed in v2.0)\n bus?: MessageBus;\n\n // New adapter system (recommended)\n storage?: StorageAdapter; // Custom storage adapter\n sync?: SyncAdapter; // Custom sync adapter\n\n // Behavior options\n debounceMs?: number; // Debounce storage writes\n validator?: (value: unknown) => value is T; // Runtime type validation\n verify?: boolean; // Enable verification tracking (creates plain object mirror)\n};\n\nconst stateRegistry = new Map<string, StateEntry<unknown>>();\n\n/**\n * Shared state: synced across all contexts AND persisted to storage\n *\n * Uses Lamport clock for conflict resolution. State is automatically:\n * - Loaded from chrome.storage on initialization\n * - Synced to other contexts via broadcast messages\n * - Persisted to chrome.storage on every change\n *\n * Available in: background, popup, options, devtools, content scripts\n * ⚠️ NOT available in page scripts (use content script state instead)\n *\n * @param key - Unique identifier for this state (e.g., \"app-settings\")\n * @param initialValue - Default value if nothing is in storage\n * @param options - Optional configuration (bus, debounceMs)\n * @returns Reactive signal that stays in sync across all contexts\n *\n * @example\n * ```typescript\n * // Define once, use everywhere\n * const settings = $sharedState(\"settings\", { theme: \"dark\" })\n *\n * // Changes automatically sync\n * settings.value = { theme: \"light\" }\n * ```\n */\nexport function $sharedState<T>(\n key: string,\n initialValue: T,\n options: StateOptions<T> = {}\n): Signal<T> & { loaded: Promise<void>; verify?: T } {\n const sig = createState(key, initialValue, {\n ...options,\n enableSync: true,\n enablePersist: true,\n });\n\n // Expose loaded promise for awaiting hydration\n const entry = stateRegistry.get(key);\n if (entry) {\n (sig as SignalWithLoaded<T>).loaded = entry.loaded;\n }\n\n return sig as Signal<T> & { loaded: Promise<void> };\n}\n\n/**\n * Synced state: synced across all contexts but NOT persisted\n *\n * State is broadcast to all contexts in real-time but resets on extension reload.\n *\n * Available in: background, popup, options, devtools, content scripts\n * ⚠️ NOT available in page scripts\n *\n * @param key - Unique identifier for this state\n * @param initialValue - Default value\n * @param options - Optional configuration\n * @returns Reactive signal synced across contexts (but not persisted)\n *\n * @example\n * ```typescript\n * // Temporary shared state\n * const activeTabId = $syncedState(\"active-tab\", null)\n * ```\n */\nexport function $syncedState<T>(\n key: string,\n initialValue: T,\n options: StateOptions<T> = {}\n): Signal<T> {\n return createState(key, initialValue, {\n ...options,\n enableSync: true,\n enablePersist: false,\n });\n}\n\n/**\n * Persisted state: persisted to storage but NOT synced across contexts\n *\n * Each context has its own copy of the state, persisted independently.\n *\n * Available in: background, popup, options, devtools, content scripts\n * ⚠️ NOT available in page scripts\n *\n * @param key - Unique identifier (use prefix like \"popup:state\" to avoid collisions)\n * @param initialValue - Default value\n * @param options - Optional configuration\n * @returns Reactive signal persisted to storage (but not synced)\n *\n * @example\n * ```typescript\n * // Each context has its own persisted state\n * const popupState = $persistedState(\"popup:last-panel\", \"home\")\n * const devtoolsState = $persistedState(\"devtools:expanded\", true)\n * ```\n */\nexport function $persistedState<T>(\n key: string,\n initialValue: T,\n options: StateOptions<T> = {}\n): Signal<T> & { loaded: Promise<void> } {\n const sig = createState(key, initialValue, {\n ...options,\n enableSync: false,\n enablePersist: true,\n });\n\n // Expose loaded promise for awaiting hydration\n const entry = stateRegistry.get(key);\n if (entry) {\n (sig as SignalWithLoaded<T>).loaded = entry.loaded;\n }\n\n return sig as Signal<T> & { loaded: Promise<void> };\n}\n\n/**\n * Local state: not synced, not persisted (like regular Preact signal)\n *\n * Simple reactive state that lives only in the current context.\n * Resets on reload or context restart.\n *\n * Available in: all contexts (including page scripts)\n *\n * @param initialValue - Default value\n * @returns Reactive signal (local only)\n *\n * @example\n * ```typescript\n * // Local UI state\n * const isLoading = $state(false)\n * const error = $state<string | null>(null)\n * ```\n */\nexport function $state<T>(initialValue: T): Signal<T> {\n return signal(initialValue);\n}\n\ntype InternalStateOptions<T = unknown> = StateOptions<T> & {\n enableSync: boolean; // Whether to enable sync (avoid collision with sync?: SyncAdapter)\n enablePersist: boolean; // Whether to enable persistence\n};\n\n// Deep equality check to prevent redundant updates\nfunction deepEqual(a: unknown, b: unknown): boolean {\n if (a === b) return true;\n if (a == null || b == null) return false;\n if (typeof a !== \"object\" || typeof b !== \"object\") return false;\n\n const keysA = Object.keys(a);\n const keysB = Object.keys(b);\n\n if (keysA.length !== keysB.length) return false;\n\n for (const key of keysA) {\n if (!keysB.includes(key)) return false;\n if (!deepEqual((a as Record<string, unknown>)[key], (b as Record<string, unknown>)[key]))\n return false;\n }\n\n return true;\n}\n\n/**\n * Resolve storage and sync adapters with three-tier priority:\n * 1. Explicit adapters from options (highest priority)\n * 2. MessageBus adapters (legacy, deprecated)\n * 3. Auto-detected adapters (default behavior)\n */\nfunction resolveAdapters(options: InternalStateOptions): {\n storage: StorageAdapter | null;\n sync: SyncAdapter | null;\n} {\n // Priority 1: Explicit adapters (partial or full)\n if (options.storage || options.sync) {\n return {\n storage: options.storage || (options.enablePersist ? createStorageAdapter() : null),\n sync: options.sync || (options.enableSync ? createSyncAdapter() : null),\n };\n }\n\n // Priority 2: MessageBus (legacy support)\n // Note: MessageBus doesn't provide sync adapter, only Chrome storage\n if (options.bus) {\n return {\n storage: options.bus.adapters.storage,\n sync: options.enableSync ? createSyncAdapter() : null,\n };\n }\n\n // Priority 3: Auto-detect based on enableSync and enablePersist flags\n return {\n storage: options.enablePersist ? createStorageAdapter() : null,\n sync: options.enableSync ? createSyncAdapter() : null,\n };\n}\n\nfunction createState<T>(key: string, initialValue: T, options: InternalStateOptions<T>): Signal<T> {\n // Return existing signal if already registered\n if (stateRegistry.has(key)) {\n return stateRegistry.get(key)?.signal as Signal<T>;\n }\n\n const sig = signal(initialValue);\n\n // Create verification mirror if requested\n if (options.verify) {\n // Create plain object mirror for verification\n const mirror = JSON.parse(JSON.stringify(initialValue)) as T;\n (sig as SignalWithVerify<T>).verify = mirror;\n }\n\n const entry: StateEntry<T> = {\n signal: sig,\n clock: 0,\n loaded: Promise.resolve(),\n updating: false,\n };\n\n // Resolve adapters (explicit, MessageBus, or auto-detect)\n const adapters = resolveAdapters(options);\n\n // Load from storage if persist is enabled\n if (options.enablePersist && adapters.storage) {\n entry.loaded = loadFromStorage(key, sig, entry, adapters.storage, options.validator);\n }\n\n // Watch for changes after initial load\n entry.loaded.then(() => {\n let debounceTimer: NodeJS.Timeout | null = null;\n let previousValue = sig.value;\n let isFirstRun = true;\n\n // biome-ignore lint/complexity/noExcessiveCognitiveComplexity: Sync effect requires coordination of multiple state change scenarios\n effect(() => {\n // Skip if update in progress (from incoming message)\n if (entry.updating) return;\n\n const value = sig.value;\n\n // Skip first run (effect fires immediately on registration)\n if (isFirstRun) {\n isFirstRun = false;\n return;\n }\n\n // Skip if value hasn't changed (deep equality check)\n if (deepEqual(value, previousValue)) {\n return;\n }\n\n previousValue = value;\n\n // Update verification mirror if enabled\n if (options.verify) {\n const verifySignal = sig as SignalWithVerify<T>;\n if (verifySignal.verify) {\n Object.assign(verifySignal.verify, JSON.parse(JSON.stringify(value)));\n }\n }\n\n // Increment clock monotonically\n entry.clock++;\n\n const doUpdate = () => {\n // Persist to storage\n if (options.enablePersist && adapters.storage) {\n persistToStorage(key, value, entry.clock, adapters.storage);\n }\n\n // Broadcast to other contexts\n if (options.enableSync && adapters.sync) {\n broadcastUpdate(key, value, entry.clock, adapters.sync);\n }\n };\n\n // Debounce if specified\n if (options.debounceMs) {\n if (debounceTimer) clearTimeout(debounceTimer);\n debounceTimer = setTimeout(doUpdate, options.debounceMs);\n } else {\n doUpdate();\n }\n });\n });\n\n // Listen for updates from other contexts (only if sync enabled)\n if (options.enableSync && adapters.sync) {\n // Connect if needed (some adapters require explicit connection)\n if (adapters.sync.connect) {\n adapters.sync.connect();\n }\n\n // Register sync message listener\n adapters.sync.onMessage<T>((message) => {\n if (message.key !== key) return;\n\n const oldClock = entry.clock;\n\n // Lamport clock rule: Always update to max(local, received)\n // This maintains causal ordering even when rejecting updates\n entry.clock = Math.max(entry.clock, message.clock);\n\n // Only accept value updates if received clock is strictly greater than old local clock\n // This ensures we only apply causally newer updates\n if (message.clock > oldClock) {\n // Validate incoming value if validator provided\n if (options.validator && !options.validator(message.value)) {\n console.warn(\n `[Polly] State \"${key}\": Received invalid value from sync (clock: ${message.clock})`,\n message.value\n );\n return;\n }\n\n // Skip redundant updates (deep equality check)\n if (deepEqual(entry.signal.value, message.value)) {\n return;\n }\n\n applyUpdate(entry, message.value as T, message.clock);\n }\n });\n }\n\n stateRegistry.set(key, entry as StateEntry<unknown>);\n return sig;\n}\n\nasync function loadFromStorage<T>(\n key: string,\n sig: Signal<T>,\n entry: StateEntry<T>,\n storage: StorageAdapter,\n validator?: (value: unknown) => value is T\n): Promise<void> {\n try {\n const result = await storage.get([key, `${key}:clock`]);\n\n if (result[key] !== undefined) {\n const storedValue = result[key];\n\n // Validate stored value if validator provided\n if (validator) {\n if (validator(storedValue)) {\n sig.value = storedValue;\n } else {\n console.warn(\n `[Polly] State \"${key}\": Stored value failed validation, using initial value`,\n storedValue\n );\n }\n } else {\n sig.value = storedValue as T;\n }\n }\n\n if (result[`${key}:clock`] !== undefined) {\n entry.clock = result[`${key}:clock`] as number;\n }\n } catch (error) {\n console.warn(`[Polly] Failed to load state from storage: ${key}`, error);\n }\n}\n\nfunction persistToStorage<T>(key: string, value: T, clock: number, storage: StorageAdapter): void {\n try {\n storage.set({\n [key]: value,\n [`${key}:clock`]: clock,\n });\n } catch (error) {\n console.warn(`[Polly] Failed to persist state to storage: ${key}`, error);\n }\n}\n\nfunction broadcastUpdate<T>(key: string, value: T, clock: number, sync: SyncAdapter): void {\n try {\n sync.broadcast({\n key,\n value,\n clock,\n });\n } catch (error) {\n console.warn(`[Polly] Failed to broadcast state update: ${key}`, error);\n }\n}\n\nfunction applyUpdate<T>(entry: StateEntry<T>, value: T, clock: number): void {\n entry.updating = true;\n entry.signal.value = value;\n entry.clock = clock;\n entry.updating = false;\n}\n\n/**\n * Get state by key (useful for retrieving state without re-creating)\n */\nexport function getStateByKey<T>(key: string): Signal<T> | undefined {\n const entry = stateRegistry.get(key);\n return entry?.signal as Signal<T> | undefined;\n}\n\n/**\n * Clear state registry (useful for testing)\n */\nexport function clearStateRegistry(): void {\n stateRegistry.clear();\n}\n"
21
8
  ],
22
- "mappings": "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",
23
- "debugId": "30E85DBFA071F76364756E2164756E21",
9
+ "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAeO,MAAM,iBAA2C;AAAA,EAC9C;AAAA,EACA,YAAY;AAAA,EACZ,YAAyC;AAAA,EAEjD,WAAW,CAAC,SAAS,eAAe;AAAA,IAClC,KAAK,SAAS;AAAA;AAAA,EAGR,KAAK,GAAyB;AAAA,IACpC,IAAI,KAAK;AAAA,MAAW,OAAO,KAAK;AAAA,IAEhC,KAAK,YAAY,IAAI,QAAQ,CAAC,SAAS,WAAW;AAAA,MAChD,MAAM,UAAU,UAAU,KAAK,KAAK,QAAQ,CAAC;AAAA,MAE7C,QAAQ,UAAU,MAAM,OAAO,QAAQ,KAAK;AAAA,MAC5C,QAAQ,YAAY,MAAM,QAAQ,QAAQ,MAAM;AAAA,MAEhD,QAAQ,kBAAkB,CAAC,UAAU;AAAA,QACnC,MAAM,KAAM,MAAM,OAA4B;AAAA,QAC9C,IAAI,CAAC,GAAG,iBAAiB,SAAS,KAAK,SAAS,GAAG;AAAA,UACjD,GAAG,kBAAkB,KAAK,SAAS;AAAA,QACrC;AAAA;AAAA,KAEH;AAAA,IAED,OAAO,KAAK;AAAA;AAAA,OAGR,IAAgB,CAAC,MAA4C;AAAA,IACjE,IAAI;AAAA,MACF,MAAM,KAAK,MAAM,KAAK,MAAM;AAAA,MAC5B,MAAM,SAA4B,CAAC;AAAA,MAEnC,MAAM,QAAQ,IACZ,KAAK,IACH,CAAC,QACC,IAAI,QAAc,CAAC,SAAS,WAAW;AAAA,QACrC,MAAM,cAAc,GAAG,YAAY,CAAC,KAAK,SAAS,GAAG,UAAU;AAAA,QAC/D,MAAM,QAAQ,YAAY,YAAY,KAAK,SAAS;AAAA,QACpD,MAAM,UAAU,MAAM,IAAI,GAAG;AAAA,QAE7B,QAAQ,UAAU,MAAM,OAAO,QAAQ,KAAK;AAAA,QAC5C,QAAQ,YAAY,MAAM;AAAA,UACxB,IAAI,QAAQ,WAAW,WAAW;AAAA,YAChC,OAAO,OAAO,QAAQ;AAAA,UACxB;AAAA,UACA,QAAQ;AAAA;AAAA,OAEX,CACL,CACF;AAAA,MAEA,OAAO;AAAA,MACP,OAAO,OAAO;AAAA,MACd,QAAQ,KAAK,iCAAiC,KAAK;AAAA,MACnD,OAAO,CAAC;AAAA;AAAA;AAAA,OAIN,IAAG,CAAC,OAA+C;AAAA,IACvD,IAAI;AAAA,MACF,MAAM,KAAK,MAAM,KAAK,MAAM;AAAA,MAC5B,MAAM,QAAQ,IACZ,OAAO,QAAQ,KAAK,EAAE,IACpB,EAAE,KAAK,WACL,IAAI,QAAc,CAAC,SAAS,WAAW;AAAA,QACrC,MAAM,cAAc,GAAG,YAAY,CAAC,KAAK,SAAS,GAAG,WAAW;AAAA,QAChE,MAAM,QAAQ,YAAY,YAAY,KAAK,SAAS;AAAA,QACpD,MAAM,UAAU,MAAM,IAAI,OAAO,GAAG;AAAA,QAEpC,QAAQ,UAAU,MAAM,OAAO,QAAQ,KAAK;AAAA,QAC5C,QAAQ,YAAY,MAAM,QAAQ;AAAA,OACnC,CACL,CACF;AAAA,MACA,OAAO,OAAO;AAAA,MACd,QAAQ,KAAK,iCAAiC,KAAK;AAAA;AAAA;AAAA,OAIjD,OAAM,CAAC,MAA+B;AAAA,IAC1C,IAAI;AAAA,MACF,MAAM,KAAK,MAAM,KAAK,MAAM;AAAA,MAC5B,MAAM,QAAQ,IACZ,KAAK,IACH,CAAC,QACC,IAAI,QAAc,CAAC,SAAS,WAAW;AAAA,QACrC,MAAM,cAAc,GAAG,YAAY,CAAC,KAAK,SAAS,GAAG,WAAW;AAAA,QAChE,MAAM,QAAQ,YAAY,YAAY,KAAK,SAAS;AAAA,QACpD,MAAM,UAAU,MAAM,OAAO,GAAG;AAAA,QAEhC,QAAQ,UAAU,MAAM,OAAO,QAAQ,KAAK;AAAA,QAC5C,QAAQ,YAAY,MAAM,QAAQ;AAAA,OACnC,CACL,CACF;AAAA,MACA,OAAO,OAAO;AAAA,MACd,QAAQ,KAAK,oCAAoC,KAAK;AAAA;AAAA;AAG5D;AAAA;AAKO,MAAM,qBAA+C;AAAA,OACpD,IAAgB,CAAC,MAA4C;AAAA,IACjE,IAAI,OAAO,WAAW,eAAe,CAAC,OAAO,SAAS;AAAA,MACpD,OAAO,CAAC;AAAA,IACV;AAAA,IAEA,IAAI;AAAA,MACF,OAAQ,MAAM,OAAO,QAAQ,MAAM,IAAI,IAAI;AAAA,MAC3C,OAAO,OAAO;AAAA,MACd,QAAQ,KAAK,sCAAsC,KAAK;AAAA,MACxD,OAAO,CAAC;AAAA;AAAA;AAAA,OAIN,IAAG,CAAC,OAA+C;AAAA,IACvD,IAAI,OAAO,WAAW,eAAe,CAAC,OAAO,SAAS;AAAA,MACpD;AAAA,IACF;AAAA,IAEA,IAAI;AAAA,MACF,MAAM,OAAO,QAAQ,MAAM,IAAI,KAAK;AAAA,MACpC,OAAO,OAAO;AAAA,MACd,QAAQ,KAAK,sCAAsC,KAAK;AAAA;AAAA;AAAA,OAItD,OAAM,CAAC,MAA+B;AAAA,IAC1C,IAAI,OAAO,WAAW,eAAe,CAAC,OAAO,SAAS;AAAA,MACpD;AAAA,IACF;AAAA,IAEA,IAAI;AAAA,MACF,MAAM,OAAO,QAAQ,MAAM,OAAO,IAAI;AAAA,MACtC,OAAO,OAAO;AAAA,MACd,QAAQ,KAAK,yCAAyC,KAAK;AAAA;AAAA;AAGjE;AAAA;AAKO,MAAM,qBAA+C;AAAA,EAClD,UAAU,IAAI;AAAA,OAEhB,IAAgB,CAAC,MAA4C;AAAA,IACjE,MAAM,SAA4B,CAAC;AAAA,IACnC,WAAW,OAAO,MAAM;AAAA,MACtB,MAAM,QAAQ,KAAK,QAAQ,IAAI,GAAG;AAAA,MAClC,IAAI,UAAU,WAAW;AAAA,QACvB,OAAO,OAAO;AAAA,MAChB;AAAA,IACF;AAAA,IACA,OAAO;AAAA;AAAA,OAGH,IAAG,CAAC,OAA+C;AAAA,IACvD,YAAY,KAAK,UAAU,OAAO,QAAQ,KAAK,GAAG;AAAA,MAChD,KAAK,QAAQ,IAAI,KAAK,KAAK;AAAA,IAC7B;AAAA;AAAA,OAGI,OAAM,CAAC,MAA+B;AAAA,IAC1C,WAAW,OAAO,MAAM;AAAA,MACtB,KAAK,QAAQ,OAAO,GAAG;AAAA,IACzB;AAAA;AAEJ;AAKO,SAAS,oBAAoB,GAAmB;AAAA,EAErD,IAAI,OAAO,WAAW,eAAe,OAAO,WAAW,OAAO,SAAS;AAAA,IACrE,OAAO,IAAI;AAAA,EACb;AAAA,EAGA,IAAI,OAAO,cAAc,aAAa;AAAA,IACpC,OAAO,IAAI;AAAA,EACb;AAAA,EAGA,OAAO,IAAI;AAAA;;;;;;;;;;;ACpIN,MAAM,gBAAuC;AAAA,EAClD,SAAY,CAAC,UAAqC;AAAA,EAIlD,SAAY,CAAC,WAA+D;AAAA,IAE1E,OAAO,MAAM;AAAA;AAIjB;AAAA;AAMO,MAAM,yBAAgD;AAAA,EACnD,YAAiE,CAAC;AAAA,EAClE,OAAmC;AAAA,EAE3C,WAAW,GAAG;AAAA,IAEZ,IAAI,OAAO,WAAW,eAAe,OAAO,SAAS;AAAA,MACnD,OAAO,QAAQ,UAAU,YAAY,CAAC,SAAS,SAAS,kBAAkB;AAAA,QACxE,IAAI,QAAQ,SAAS,cAAc;AAAA,UACjC,KAAK,UAAU,QAAQ,CAAC,aAAa;AAAA,YACnC,SAAS,OAAO;AAAA,WACjB;AAAA,QACH;AAAA,OACD;AAAA,IACH;AAAA;AAAA,EAGF,SAAY,CAAC,SAAoC;AAAA,IAC/C,IAAI,OAAO,WAAW,eAAe,CAAC,OAAO,SAAS;AAAA,MACpD,QAAQ,KAAK,4CAA4C;AAAA,MACzD;AAAA,IACF;AAAA,IAEA,IAAI;AAAA,MACF,OAAO,QAAQ,YAAY;AAAA,QACzB,MAAM;AAAA,QACN,KAAK,QAAQ;AAAA,QACb,OAAO,QAAQ;AAAA,QACf,OAAO,QAAQ;AAAA,MACjB,CAAC;AAAA,MACD,OAAO,OAAO;AAAA,MACd,QAAQ,KAAK,mDAAmD,KAAK;AAAA;AAAA;AAAA,EAIzE,SAAY,CAAC,UAA8D;AAAA,IACzE,KAAK,UAAU,KAAK,QAAwD;AAAA,IAG5E,OAAO,MAAM;AAAA,MACX,MAAM,QAAQ,KAAK,UAAU,QAC3B,QACF;AAAA,MACA,IAAI,QAAQ,IAAI;AAAA,QACd,KAAK,UAAU,OAAO,OAAO,CAAC;AAAA,MAChC;AAAA;AAAA;AAAA,EAIJ,OAAO,GAAkB;AAAA,IAEvB,OAAO,QAAQ,QAAQ;AAAA;AAAA,EAGzB,UAAU,GAAS;AAAA,IACjB,KAAK,YAAY,CAAC;AAAA,IAClB,IAAI,KAAK,MAAM;AAAA,MACb,KAAK,KAAK,WAAW;AAAA,MACrB,KAAK,OAAO;AAAA,IACd;AAAA;AAAA,EAGF,WAAW,GAAY;AAAA,IACrB,OAAO,OAAO,WAAW,eAAe,CAAC,CAAC,OAAO;AAAA;AAErD;AAAA;AAMO,MAAM,4BAAmD;AAAA,EACtD,UAAmC;AAAA,EACnC,YAAiE,CAAC;AAAA,EAE1E,WAAW,CAAC,cAAc,cAAc;AAAA,IACtC,IAAI,OAAO,qBAAqB,aAAa;AAAA,MAC3C,KAAK,UAAU,IAAI,iBAAiB,WAAW;AAAA,MAE/C,KAAK,QAAQ,YAAY,CAAC,UAAU;AAAA,QAClC,IAAI,MAAM,KAAK,SAAS,cAAc;AAAA,UACpC,KAAK,UAAU,QAAQ,CAAC,aAAa;AAAA,YACnC,SAAS,MAAM,IAAI;AAAA,WACpB;AAAA,QACH;AAAA;AAAA,IAEJ,EAAO;AAAA,MACL,QAAQ,KAAK,8CAA8C;AAAA;AAAA;AAAA,EAI/D,SAAY,CAAC,SAAoC;AAAA,IAC/C,IAAI,CAAC,KAAK,SAAS;AAAA,MACjB,QAAQ,KAAK,gDAAgD;AAAA,MAC7D;AAAA,IACF;AAAA,IAEA,IAAI;AAAA,MACF,KAAK,QAAQ,YAAY;AAAA,QACvB,MAAM;AAAA,QACN,KAAK,QAAQ;AAAA,QACb,OAAO,QAAQ;AAAA,QACf,OAAO,QAAQ;AAAA,MACjB,CAAC;AAAA,MACD,OAAO,OAAO;AAAA,MACd,QAAQ,KAAK,mDAAmD,KAAK;AAAA;AAAA;AAAA,EAIzE,SAAY,CAAC,UAA8D;AAAA,IACzE,KAAK,UAAU,KAAK,QAAwD;AAAA,IAG5E,OAAO,MAAM;AAAA,MACX,MAAM,QAAQ,KAAK,UAAU,QAC3B,QACF;AAAA,MACA,IAAI,QAAQ,IAAI;AAAA,QACd,KAAK,UAAU,OAAO,OAAO,CAAC;AAAA,MAChC;AAAA;AAAA;AAAA,EAIJ,OAAO,GAAkB;AAAA,IAEvB,OAAO,QAAQ,QAAQ;AAAA;AAAA,EAGzB,UAAU,GAAS;AAAA,IACjB,KAAK,YAAY,CAAC;AAAA,IAClB,IAAI,KAAK,SAAS;AAAA,MAChB,KAAK,QAAQ,MAAM;AAAA,MACnB,KAAK,UAAU;AAAA,IACjB;AAAA;AAAA,EAGF,WAAW,GAAY;AAAA,IACrB,OAAO,KAAK,YAAY;AAAA;AAE5B;AAKO,SAAS,iBAAiB,GAAgB;AAAA,EAE/C,IAAI,OAAO,WAAW,eAAe,OAAO,SAAS;AAAA,IACnD,OAAO,IAAI;AAAA,EACb;AAAA,EAGA,IAAI,OAAO,qBAAqB,aAAa;AAAA,IAC3C,OAAO,IAAI;AAAA,EACb;AAAA,EAGA,OAAO,IAAI;AAAA;;;ACpPb;AAoCA,IAAM,gBAAgB,IAAI;AA2BnB,SAAS,YAAe,CAC7B,KACA,cACA,UAA2B,CAAC,GACuB;AAAA,EACnD,MAAM,MAAM,YAAY,KAAK,cAAc;AAAA,OACtC;AAAA,IACH,YAAY;AAAA,IACZ,eAAe;AAAA,EACjB,CAAC;AAAA,EAGD,MAAM,QAAQ,cAAc,IAAI,GAAG;AAAA,EACnC,IAAI,OAAO;AAAA,IACR,IAA4B,SAAS,MAAM;AAAA,EAC9C;AAAA,EAEA,OAAO;AAAA;AAsBF,SAAS,YAAe,CAC7B,KACA,cACA,UAA2B,CAAC,GACjB;AAAA,EACX,OAAO,YAAY,KAAK,cAAc;AAAA,OACjC;AAAA,IACH,YAAY;AAAA,IACZ,eAAe;AAAA,EACjB,CAAC;AAAA;AAuBI,SAAS,eAAkB,CAChC,KACA,cACA,UAA2B,CAAC,GACW;AAAA,EACvC,MAAM,MAAM,YAAY,KAAK,cAAc;AAAA,OACtC;AAAA,IACH,YAAY;AAAA,IACZ,eAAe;AAAA,EACjB,CAAC;AAAA,EAGD,MAAM,QAAQ,cAAc,IAAI,GAAG;AAAA,EACnC,IAAI,OAAO;AAAA,IACR,IAA4B,SAAS,MAAM;AAAA,EAC9C;AAAA,EAEA,OAAO;AAAA;AAqBF,SAAS,MAAS,CAAC,cAA4B;AAAA,EACpD,OAAO,OAAO,YAAY;AAAA;AAS5B,SAAS,SAAS,CAAC,GAAY,GAAqB;AAAA,EAClD,IAAI,MAAM;AAAA,IAAG,OAAO;AAAA,EACpB,IAAI,KAAK,QAAQ,KAAK;AAAA,IAAM,OAAO;AAAA,EACnC,IAAI,OAAO,MAAM,YAAY,OAAO,MAAM;AAAA,IAAU,OAAO;AAAA,EAE3D,MAAM,QAAQ,OAAO,KAAK,CAAC;AAAA,EAC3B,MAAM,QAAQ,OAAO,KAAK,CAAC;AAAA,EAE3B,IAAI,MAAM,WAAW,MAAM;AAAA,IAAQ,OAAO;AAAA,EAE1C,WAAW,OAAO,OAAO;AAAA,IACvB,IAAI,CAAC,MAAM,SAAS,GAAG;AAAA,MAAG,OAAO;AAAA,IACjC,IAAI,CAAC,UAAW,EAA8B,MAAO,EAA8B,IAAI;AAAA,MACrF,OAAO;AAAA,EACX;AAAA,EAEA,OAAO;AAAA;AAST,SAAS,eAAe,CAAC,SAGvB;AAAA,EAEA,IAAI,QAAQ,WAAW,QAAQ,MAAM;AAAA,IACnC,OAAO;AAAA,MACL,SAAS,QAAQ,YAAY,QAAQ,gBAAgB,qBAAqB,IAAI;AAAA,MAC9E,MAAM,QAAQ,SAAS,QAAQ,aAAa,kBAAkB,IAAI;AAAA,IACpE;AAAA,EACF;AAAA,EAIA,IAAI,QAAQ,KAAK;AAAA,IACf,OAAO;AAAA,MACL,SAAS,QAAQ,IAAI,SAAS;AAAA,MAC9B,MAAM,QAAQ,aAAa,kBAAkB,IAAI;AAAA,IACnD;AAAA,EACF;AAAA,EAGA,OAAO;AAAA,IACL,SAAS,QAAQ,gBAAgB,qBAAqB,IAAI;AAAA,IAC1D,MAAM,QAAQ,aAAa,kBAAkB,IAAI;AAAA,EACnD;AAAA;AAGF,SAAS,WAAc,CAAC,KAAa,cAAiB,SAA6C;AAAA,EAEjG,IAAI,cAAc,IAAI,GAAG,GAAG;AAAA,IAC1B,OAAO,cAAc,IAAI,GAAG,GAAG;AAAA,EACjC;AAAA,EAEA,MAAM,MAAM,OAAO,YAAY;AAAA,EAG/B,IAAI,QAAQ,QAAQ;AAAA,IAElB,MAAM,SAAS,KAAK,MAAM,KAAK,UAAU,YAAY,CAAC;AAAA,IACrD,IAA4B,SAAS;AAAA,EACxC;AAAA,EAEA,MAAM,QAAuB;AAAA,IAC3B,QAAQ;AAAA,IACR,OAAO;AAAA,IACP,QAAQ,QAAQ,QAAQ;AAAA,IACxB,UAAU;AAAA,EACZ;AAAA,EAGA,MAAM,WAAW,gBAAgB,OAAO;AAAA,EAGxC,IAAI,QAAQ,iBAAiB,SAAS,SAAS;AAAA,IAC7C,MAAM,SAAS,gBAAgB,KAAK,KAAK,OAAO,SAAS,SAAS,QAAQ,SAAS;AAAA,EACrF;AAAA,EAGA,MAAM,OAAO,KAAK,MAAM;AAAA,IACtB,IAAI,gBAAuC;AAAA,IAC3C,IAAI,gBAAgB,IAAI;AAAA,IACxB,IAAI,aAAa;AAAA,IAGjB,OAAO,MAAM;AAAA,MAEX,IAAI,MAAM;AAAA,QAAU;AAAA,MAEpB,MAAM,QAAQ,IAAI;AAAA,MAGlB,IAAI,YAAY;AAAA,QACd,aAAa;AAAA,QACb;AAAA,MACF;AAAA,MAGA,IAAI,UAAU,OAAO,aAAa,GAAG;AAAA,QACnC;AAAA,MACF;AAAA,MAEA,gBAAgB;AAAA,MAGhB,IAAI,QAAQ,QAAQ;AAAA,QAClB,MAAM,eAAe;AAAA,QACrB,IAAI,aAAa,QAAQ;AAAA,UACvB,OAAO,OAAO,aAAa,QAAQ,KAAK,MAAM,KAAK,UAAU,KAAK,CAAC,CAAC;AAAA,QACtE;AAAA,MACF;AAAA,MAGA,MAAM;AAAA,MAEN,MAAM,WAAW,MAAM;AAAA,QAErB,IAAI,QAAQ,iBAAiB,SAAS,SAAS;AAAA,UAC7C,iBAAiB,KAAK,OAAO,MAAM,OAAO,SAAS,OAAO;AAAA,QAC5D;AAAA,QAGA,IAAI,QAAQ,cAAc,SAAS,MAAM;AAAA,UACvC,gBAAgB,KAAK,OAAO,MAAM,OAAO,SAAS,IAAI;AAAA,QACxD;AAAA;AAAA,MAIF,IAAI,QAAQ,YAAY;AAAA,QACtB,IAAI;AAAA,UAAe,aAAa,aAAa;AAAA,QAC7C,gBAAgB,WAAW,UAAU,QAAQ,UAAU;AAAA,MACzD,EAAO;AAAA,QACL,SAAS;AAAA;AAAA,KAEZ;AAAA,GACF;AAAA,EAGD,IAAI,QAAQ,cAAc,SAAS,MAAM;AAAA,IAEvC,IAAI,SAAS,KAAK,SAAS;AAAA,MACzB,SAAS,KAAK,QAAQ;AAAA,IACxB;AAAA,IAGA,SAAS,KAAK,UAAa,CAAC,YAAY;AAAA,MACtC,IAAI,QAAQ,QAAQ;AAAA,QAAK;AAAA,MAEzB,MAAM,WAAW,MAAM;AAAA,MAIvB,MAAM,QAAQ,KAAK,IAAI,MAAM,OAAO,QAAQ,KAAK;AAAA,MAIjD,IAAI,QAAQ,QAAQ,UAAU;AAAA,QAE5B,IAAI,QAAQ,aAAa,CAAC,QAAQ,UAAU,QAAQ,KAAK,GAAG;AAAA,UAC1D,QAAQ,KACN,kBAAkB,kDAAkD,QAAQ,UAC5E,QAAQ,KACV;AAAA,UACA;AAAA,QACF;AAAA,QAGA,IAAI,UAAU,MAAM,OAAO,OAAO,QAAQ,KAAK,GAAG;AAAA,UAChD;AAAA,QACF;AAAA,QAEA,YAAY,OAAO,QAAQ,OAAY,QAAQ,KAAK;AAAA,MACtD;AAAA,KACD;AAAA,EACH;AAAA,EAEA,cAAc,IAAI,KAAK,KAA4B;AAAA,EACnD,OAAO;AAAA;AAGT,eAAe,eAAkB,CAC/B,KACA,KACA,OACA,SACA,WACe;AAAA,EACf,IAAI;AAAA,IACF,MAAM,SAAS,MAAM,QAAQ,IAAI,CAAC,KAAK,GAAG,WAAW,CAAC;AAAA,IAEtD,IAAI,OAAO,SAAS,WAAW;AAAA,MAC7B,MAAM,cAAc,OAAO;AAAA,MAG3B,IAAI,WAAW;AAAA,QACb,IAAI,UAAU,WAAW,GAAG;AAAA,UAC1B,IAAI,QAAQ;AAAA,QACd,EAAO;AAAA,UACL,QAAQ,KACN,kBAAkB,6DAClB,WACF;AAAA;AAAA,MAEJ,EAAO;AAAA,QACL,IAAI,QAAQ;AAAA;AAAA,IAEhB;AAAA,IAEA,IAAI,OAAO,GAAG,iBAAiB,WAAW;AAAA,MACxC,MAAM,QAAQ,OAAO,GAAG;AAAA,IAC1B;AAAA,IACA,OAAO,OAAO;AAAA,IACd,QAAQ,KAAK,8CAA8C,OAAO,KAAK;AAAA;AAAA;AAI3E,SAAS,gBAAmB,CAAC,KAAa,OAAU,OAAe,SAA+B;AAAA,EAChG,IAAI;AAAA,IACF,QAAQ,IAAI;AAAA,OACT,MAAM;AAAA,OACN,GAAG,cAAc;AAAA,IACpB,CAAC;AAAA,IACD,OAAO,OAAO;AAAA,IACd,QAAQ,KAAK,+CAA+C,OAAO,KAAK;AAAA;AAAA;AAI5E,SAAS,eAAkB,CAAC,KAAa,OAAU,OAAe,MAAyB;AAAA,EACzF,IAAI;AAAA,IACF,KAAK,UAAU;AAAA,MACb;AAAA,MACA;AAAA,MACA;AAAA,IACF,CAAC;AAAA,IACD,OAAO,OAAO;AAAA,IACd,QAAQ,KAAK,6CAA6C,OAAO,KAAK;AAAA;AAAA;AAI1E,SAAS,WAAc,CAAC,OAAsB,OAAU,OAAqB;AAAA,EAC3E,MAAM,WAAW;AAAA,EACjB,MAAM,OAAO,QAAQ;AAAA,EACrB,MAAM,QAAQ;AAAA,EACd,MAAM,WAAW;AAAA;AAMZ,SAAS,aAAgB,CAAC,KAAoC;AAAA,EACnE,MAAM,QAAQ,cAAc,IAAI,GAAG;AAAA,EACnC,OAAO,OAAO;AAAA;AAMT,SAAS,kBAAkB,GAAS;AAAA,EACzC,cAAc,MAAM;AAAA;",
10
+ "debugId": "44A9C265AE724F4564756E2164756E21",
24
11
  "names": []
25
12
  }
@@ -0,0 +1,42 @@
1
+ /**
2
+ * Universal storage adapter interface
3
+ */
4
+ export interface StorageAdapter {
5
+ get<T = unknown>(keys: string[]): Promise<Record<string, T>>;
6
+ set(items: Record<string, unknown>): Promise<void>;
7
+ remove(keys: string[]): Promise<void>;
8
+ }
9
+ /**
10
+ * IndexedDB adapter for web apps
11
+ */
12
+ export declare class IndexedDBAdapter implements StorageAdapter {
13
+ private dbName;
14
+ private storeName;
15
+ private dbPromise;
16
+ constructor(dbName?: string);
17
+ private getDB;
18
+ get<T = unknown>(keys: string[]): Promise<Record<string, T>>;
19
+ set(items: Record<string, unknown>): Promise<void>;
20
+ remove(keys: string[]): Promise<void>;
21
+ }
22
+ /**
23
+ * Chrome storage adapter for extensions
24
+ */
25
+ export declare class ChromeStorageAdapter implements StorageAdapter {
26
+ get<T = unknown>(keys: string[]): Promise<Record<string, T>>;
27
+ set(items: Record<string, unknown>): Promise<void>;
28
+ remove(keys: string[]): Promise<void>;
29
+ }
30
+ /**
31
+ * In-memory adapter (no persistence) for testing or server contexts
32
+ */
33
+ export declare class MemoryStorageAdapter implements StorageAdapter {
34
+ private storage;
35
+ get<T = unknown>(keys: string[]): Promise<Record<string, T>>;
36
+ set(items: Record<string, unknown>): Promise<void>;
37
+ remove(keys: string[]): Promise<void>;
38
+ }
39
+ /**
40
+ * Detect execution context and return appropriate storage adapter
41
+ */
42
+ export declare function createStorageAdapter(): StorageAdapter;
@@ -0,0 +1,79 @@
1
+ /**
2
+ * Message format for state synchronization
3
+ */
4
+ export interface StateSyncMessage<T = unknown> {
5
+ key: string;
6
+ value: T;
7
+ clock: number;
8
+ }
9
+ /**
10
+ * Sync adapter interface - abstracts the transport mechanism for state sync
11
+ *
12
+ * Different contexts use different transports:
13
+ * - Chrome extensions: chrome.runtime messaging
14
+ * - Web apps (multi-tab): BroadcastChannel
15
+ * - PWAs: BroadcastChannel + Service Worker messaging
16
+ * - Single-context: NoOp (no sync needed)
17
+ */
18
+ export interface SyncAdapter {
19
+ /**
20
+ * Broadcast a state update to other contexts
21
+ */
22
+ broadcast<T>(message: StateSyncMessage<T>): void;
23
+ /**
24
+ * Register a callback for incoming state updates
25
+ */
26
+ onMessage<T>(callback: (message: StateSyncMessage<T>) => void): () => void;
27
+ /**
28
+ * Optional: Connect to the sync mechanism
29
+ * Some transports require explicit connection setup
30
+ */
31
+ connect?(): Promise<void>;
32
+ /**
33
+ * Optional: Disconnect from the sync mechanism
34
+ */
35
+ disconnect?(): void;
36
+ /**
37
+ * Optional: Check if connected
38
+ */
39
+ isConnected?(): boolean;
40
+ }
41
+ /**
42
+ * NoOp sync adapter for single-context scenarios (no sync needed)
43
+ */
44
+ export declare class NoOpSyncAdapter implements SyncAdapter {
45
+ broadcast<T>(_message: StateSyncMessage<T>): void;
46
+ onMessage<T>(_callback: (message: StateSyncMessage<T>) => void): () => void;
47
+ }
48
+ /**
49
+ * Chrome runtime sync adapter for Chrome extensions
50
+ * Uses chrome.runtime.sendMessage for cross-context messaging
51
+ */
52
+ export declare class ChromeRuntimeSyncAdapter implements SyncAdapter {
53
+ private listeners;
54
+ private port;
55
+ constructor();
56
+ broadcast<T>(message: StateSyncMessage<T>): void;
57
+ onMessage<T>(callback: (message: StateSyncMessage<T>) => void): () => void;
58
+ connect(): Promise<void>;
59
+ disconnect(): void;
60
+ isConnected(): boolean;
61
+ }
62
+ /**
63
+ * BroadcastChannel sync adapter for web apps (multi-tab)
64
+ * Uses BroadcastChannel API for cross-tab messaging
65
+ */
66
+ export declare class BroadcastChannelSyncAdapter implements SyncAdapter {
67
+ private channel;
68
+ private listeners;
69
+ constructor(channelName?: string);
70
+ broadcast<T>(message: StateSyncMessage<T>): void;
71
+ onMessage<T>(callback: (message: StateSyncMessage<T>) => void): () => void;
72
+ connect(): Promise<void>;
73
+ disconnect(): void;
74
+ isConnected(): boolean;
75
+ }
76
+ /**
77
+ * Detect available sync mechanisms and create appropriate adapter
78
+ */
79
+ export declare function createSyncAdapter(): SyncAdapter;