@fairfox/polly 0.78.0 → 0.79.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli/polly.js +46 -3
- package/dist/cli/polly.js.map +3 -3
- package/dist/src/background/index.js.map +3 -3
- package/dist/src/background/message-router.js.map +3 -3
- package/dist/src/client/index.js +137 -32
- package/dist/src/client/index.js.map +6 -5
- package/dist/src/client/wrapper.d.ts +39 -2
- package/dist/src/elysia/index.js +22 -3
- package/dist/src/elysia/index.js.map +5 -5
- package/dist/src/elysia/route-match.d.ts +13 -0
- package/dist/src/index.d.ts +1 -1
- package/dist/src/index.js +12 -2
- package/dist/src/index.js.map +7 -7
- package/dist/src/mesh.js +28 -9
- package/dist/src/mesh.js.map +10 -9
- package/dist/src/peer.js +6 -2
- package/dist/src/peer.js.map +5 -5
- package/dist/src/polly-ui/Badge.d.ts +5 -0
- package/dist/src/polly-ui/Button.d.ts +31 -6
- package/dist/src/polly-ui/Dropdown.d.ts +6 -0
- package/dist/src/polly-ui/Select.d.ts +11 -1
- package/dist/src/polly-ui/TextInput.d.ts +30 -0
- package/dist/src/polly-ui/index.css +10 -0
- package/dist/src/polly-ui/index.js +81 -32
- package/dist/src/polly-ui/index.js.map +10 -10
- package/dist/src/polly-ui/styles.css +10 -0
- package/dist/src/shared/adapters/index.js.map +3 -3
- package/dist/src/shared/lib/context-helpers.js.map +3 -3
- package/dist/src/shared/lib/message-bus.js.map +3 -3
- package/dist/src/shared/lib/resource.js +11 -2
- package/dist/src/shared/lib/resource.js.map +6 -6
- package/dist/src/shared/lib/state.d.ts +20 -0
- package/dist/src/shared/lib/state.js +11 -1
- package/dist/src/shared/lib/state.js.map +5 -5
- package/dist/src/shared/state/app-state.js +10 -1
- package/dist/src/shared/state/app-state.js.map +5 -5
- package/dist/tools/init/src/cli.js +23 -2
- package/dist/tools/init/src/cli.js.map +4 -4
- package/dist/tools/init/templates/pwa/package.json.template +1 -1
- package/dist/tools/init/templates/pwa/src/service-worker.ts.template +26 -15
- package/dist/tools/init/templates/pwa/src/shared-worker.ts.template +13 -3
- package/dist/tools/init/templates/pwa/tsconfig.json.template +2 -2
- package/dist/tools/init/templates/pwa/tsconfig.worker.json.template +17 -0
- package/dist/tools/test/src/browser/index.js +5 -2
- package/dist/tools/test/src/browser/index.js.map +3 -3
- package/dist/tools/test/src/contrast/index.js +20 -15
- package/dist/tools/test/src/contrast/index.js.map +3 -3
- package/dist/tools/test/src/e2e-cli/index.d.ts +10 -0
- package/dist/tools/test/src/e2e-cli/run-cli.d.ts +25 -0
- package/dist/tools/test/src/e2e-cli/with-temp-dir.d.ts +15 -0
- package/dist/tools/test/src/e2e-mesh/index.js +12 -7
- package/dist/tools/test/src/e2e-mesh/index.js.map +4 -4
- package/dist/tools/test/src/e2e-mesh/launch-peer.d.ts +7 -1
- package/dist/tools/test/src/e2e-relay/index.d.ts +12 -0
- package/dist/tools/test/src/e2e-relay/wait-for-relay-convergence.d.ts +27 -0
- package/dist/tools/test/src/e2e-relay/with-repo-server.d.ts +24 -0
- package/dist/tools/test/src/e2e-shared/assert.d.ts +18 -0
- package/dist/tools/test/src/e2e-shared/contract.d.ts +40 -0
- package/dist/tools/test/src/e2e-shared/index.d.ts +2 -0
- package/dist/tools/test/src/tiers/args.d.ts +23 -0
- package/dist/tools/test/src/tiers/cli.d.ts +2 -0
- package/dist/tools/test/src/tiers/cli.js +490 -0
- package/dist/tools/test/src/tiers/cli.js.map +16 -0
- package/dist/tools/test/src/tiers/detect.d.ts +12 -0
- package/dist/tools/test/src/tiers/discover.d.ts +2 -0
- package/dist/tools/test/src/tiers/engine.d.ts +3 -0
- package/dist/tools/test/src/tiers/index.d.ts +14 -0
- package/dist/tools/test/src/tiers/protocol.d.ts +10 -0
- package/dist/tools/test/src/tiers/reporter.d.ts +12 -0
- package/dist/tools/test/src/tiers/types.d.ts +94 -0
- package/dist/tools/test/src/tiers/worker.d.ts +2 -0
- package/dist/tools/test/src/tiers/worker.js +60 -0
- package/dist/tools/test/src/tiers/worker.js.map +12 -0
- package/dist/tools/verify/src/cli.js +165 -30
- package/dist/tools/verify/src/cli.js.map +7 -6
- package/dist/tools/verify/src/stryker/index.js +20 -11
- package/dist/tools/verify/src/stryker/index.js.map +3 -3
- package/dist/tools/visualize/src/cli.js +8 -5
- package/dist/tools/visualize/src/cli.js.map +4 -4
- package/package.json +16 -6
|
@@ -2,14 +2,14 @@
|
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../src/shared/lib/idb-helpers.ts", "../src/shared/lib/storage-adapter.ts", "../src/shared/lib/sync-adapter.ts", "../src/shared/types/messages.ts", "../src/shared/lib/state.ts", "../src/shared/state/app-state.ts"],
|
|
4
4
|
"sourcesContent": [
|
|
5
|
-
"/**\n * Tiny internal helpers around the raw IndexedDB API.\n *\n * Wraps the patterns that `storage-adapter.ts` and `blob-cache.ts` both\n * re-implement: timed open with retry-on-failure caching, request→promise\n * wrapping, transaction-to-completion, and cursor iteration. Not exported\n * from the public package — internal to `src/shared/lib`.\n */\n\n/** Polly#107 post-v0.60: hard timeout on `indexedDB.open()`. Healthy\n * opens land in microseconds; the v0.60.0 fingerprint surfaced a zombie\n * cross-tab connection that left an open request firing no events at all.\n * 5s is two orders of magnitude beyond the normal upper bound and short\n * enough that the operator gets a named failure instead of a hung page\n * when the storage layer wedges. */\nexport const IDB_OPEN_TIMEOUT_MS = 5000;\n\nexport type IDBOpenFailureReason = \"timeout\" | \"blocked\" | \"error\";\n\n/** Structured failure for `openIDB`. `reason` distinguishes the three\n * recoveries: `timeout` (true zombie, browser restart), `blocked`\n * (sibling tab holds an older version open — closing it clears the\n * block immediately), and `error` (unexpected `onerror` from the\n * request itself). Stays `instanceof Error` for callers that just want\n * `console.warn(err)`; `instanceof IDBOpenError` + `err.reason` gives\n * structured handling without regexing the message. */\nexport class IDBOpenError extends Error {\n readonly reason: IDBOpenFailureReason;\n readonly dbName: string;\n readonly elapsedMs: number;\n\n constructor(reason: IDBOpenFailureReason, dbName: string, elapsedMs: number, cause?: unknown) {\n super(\n `Polly IndexedDB open of '${dbName}' ${reason} after ${elapsedMs}ms`,\n cause === undefined ? undefined : { cause }\n );\n this.name = \"IDBOpenError\";\n this.reason = reason;\n this.dbName = dbName;\n this.elapsedMs = elapsedMs;\n }\n}\n\nexport interface OpenIDBOptions {\n name: string;\n version: number;\n /** Invoked inside `onupgradeneeded`. Create object stores here. */\n upgrade: (db: IDBDatabase, event: IDBVersionChangeEvent) => void;\n}\n\n/** Open-request factory. Defaults to `indexedDB.open`; tests override\n * it to force the timeout, blocked, or error path without needing a\n * real zombie tab — `fake-indexeddb` cannot reproduce the v0.60.0\n * no-events fingerprint, so the regression that motivated this whole\n * module is otherwise uncoverable. */\nexport type IDBOpenFn = (name: string, version: number) => IDBOpenDBRequest;\n\nconst defaultOpenFn: IDBOpenFn = (name, version) => indexedDB.open(name, version);\n\n/** Open an IndexedDB database with the Polly#107 timeout guard. */\nexport function openIDB(\n options: OpenIDBOptions,\n openFn: IDBOpenFn = defaultOpenFn\n): Promise<IDBDatabase> {\n return new Promise<IDBDatabase>((resolve, reject) => {\n const start = Date.now();\n const request = openFn(options.name, options.version);\n let settled = false;\n const elapsed = () => Date.now() - start;\n const rejectWith = (reason: IDBOpenFailureReason, cause?: unknown) => {\n if (settled) return;\n settled = true;\n clearTimeout(timer);\n reject(new IDBOpenError(reason, options.name, elapsed(), cause));\n };\n const timer = setTimeout(() => rejectWith(\"timeout\"), IDB_OPEN_TIMEOUT_MS);\n\n request.onerror = () => rejectWith(\"error\", request.error);\n request.onblocked = () => rejectWith(\"blocked\");\n request.onsuccess = () => {\n if (settled) return;\n settled = true;\n clearTimeout(timer);\n resolve(request.result);\n };\n request.onupgradeneeded = (event) => {\n const db = (event.target as unknown as IDBOpenDBRequest).result;\n options.upgrade(db, event);\n };\n });\n}\n\n/** One-shot open caching with retry-on-failure. Caller owns a `{ promise }`\n * cell; first call opens, subsequent calls return the cached promise, and\n * a rejected open clears the cache so the next call can retry instead of\n * being poisoned by one transient failure. */\nexport function cachedOpen(\n ref: { promise: Promise<IDBDatabase> | null },\n options: OpenIDBOptions,\n openFn?: IDBOpenFn\n): Promise<IDBDatabase> {\n if (ref.promise) return ref.promise;\n const pending = openIDB(options, openFn);\n pending.catch(() => {\n if (ref.promise === pending) ref.promise = null;\n });\n ref.promise = pending;\n return pending;\n}\n\n/** Promise-wrap a single `IDBRequest` (`get`, `put`, `count`, `delete`…). */\nexport function runRequest<T>(request: IDBRequest<T>): Promise<T> {\n return new Promise<T>((resolve, reject) => {\n request.onsuccess = () => resolve(request.result);\n request.onerror = () => reject(request.error);\n });\n}\n\n/** Run a transaction to completion. Resolves on `tx.oncomplete` (write\n * durability), not on individual request success. Throws inside `fn`\n * abort the transaction and reject. */\nexport function runTx(\n db: IDBDatabase,\n storeName: string,\n mode: IDBTransactionMode,\n fn: (store: IDBObjectStore) => void\n): Promise<void> {\n return new Promise<void>((resolve, reject) => {\n const tx = db.transaction(storeName, mode);\n const store = tx.objectStore(storeName);\n tx.oncomplete = () => resolve();\n tx.onerror = () => reject(tx.error);\n tx.onabort = () => reject(tx.error);\n try {\n fn(store);\n } catch (err) {\n try {\n tx.abort();\n } catch {\n /* already aborted */\n }\n reject(err);\n }\n });\n}\n\n/** Walk every record in a store. */\nexport function iterateCursor<V>(\n db: IDBDatabase,\n storeName: string,\n visit: (key: IDBValidKey, value: V) => void\n): Promise<void> {\n return new Promise<void>((resolve, reject) => {\n const tx = db.transaction(storeName, \"readonly\");\n const store = tx.objectStore(storeName);\n const request = store.openCursor();\n request.onsuccess = () => {\n const cursor = request.result;\n if (!cursor) return resolve();\n visit(cursor.key, cursor.value
|
|
6
|
-
"// Storage adapters for different execution contexts\n// Automatically chooses the right storage mechanism based on environment\n\nimport { cachedOpen, runRequest, runTx } from \"./idb-helpers\";\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 readonly dbName: string;\n private readonly storeName = \"state\";\n private readonly dbRef: { promise: Promise<IDBDatabase> | null } = { promise: null };\n\n constructor(dbName = \"polly-state\") {\n this.dbName = dbName;\n }\n\n private getDB(): Promise<IDBDatabase> {\n return cachedOpen(this.dbRef, {\n name: this.dbName,\n version: 1,\n upgrade: (db) => {\n if (!db.objectStoreNames.contains(this.storeName)) {\n db.createObjectStore(this.storeName);\n }\n },\n });\n }\n\n async get<T = unknown>(keys: string[]): Promise<Record<string, T>> {\n try {\n const db = await this.getDB();\n const tx = db.transaction(this.storeName, \"readonly\");\n const store = tx.objectStore(this.storeName);\n const pairs = await Promise.all(\n keys.map(async (key) => [key, await runRequest(store.get(key))] as const)\n );\n const result: Record<string, T> = {};\n for (const [key, value] of pairs) {\n if (value !== undefined) result[key] = value
|
|
5
|
+
"/**\n * Tiny internal helpers around the raw IndexedDB API.\n *\n * Wraps the patterns that `storage-adapter.ts` and `blob-cache.ts` both\n * re-implement: timed open with retry-on-failure caching, request→promise\n * wrapping, transaction-to-completion, and cursor iteration. Not exported\n * from the public package — internal to `src/shared/lib`.\n */\n\n/** Polly#107 post-v0.60: hard timeout on `indexedDB.open()`. Healthy\n * opens land in microseconds; the v0.60.0 fingerprint surfaced a zombie\n * cross-tab connection that left an open request firing no events at all.\n * 5s is two orders of magnitude beyond the normal upper bound and short\n * enough that the operator gets a named failure instead of a hung page\n * when the storage layer wedges. */\nexport const IDB_OPEN_TIMEOUT_MS = 5000;\n\nexport type IDBOpenFailureReason = \"timeout\" | \"blocked\" | \"error\";\n\n/** Structured failure for `openIDB`. `reason` distinguishes the three\n * recoveries: `timeout` (true zombie, browser restart), `blocked`\n * (sibling tab holds an older version open — closing it clears the\n * block immediately), and `error` (unexpected `onerror` from the\n * request itself). Stays `instanceof Error` for callers that just want\n * `console.warn(err)`; `instanceof IDBOpenError` + `err.reason` gives\n * structured handling without regexing the message. */\nexport class IDBOpenError extends Error {\n readonly reason: IDBOpenFailureReason;\n readonly dbName: string;\n readonly elapsedMs: number;\n\n constructor(reason: IDBOpenFailureReason, dbName: string, elapsedMs: number, cause?: unknown) {\n super(\n `Polly IndexedDB open of '${dbName}' ${reason} after ${elapsedMs}ms`,\n cause === undefined ? undefined : { cause }\n );\n this.name = \"IDBOpenError\";\n this.reason = reason;\n this.dbName = dbName;\n this.elapsedMs = elapsedMs;\n }\n}\n\nexport interface OpenIDBOptions {\n name: string;\n version: number;\n /** Invoked inside `onupgradeneeded`. Create object stores here. */\n upgrade: (db: IDBDatabase, event: IDBVersionChangeEvent) => void;\n}\n\n/** Open-request factory. Defaults to `indexedDB.open`; tests override\n * it to force the timeout, blocked, or error path without needing a\n * real zombie tab — `fake-indexeddb` cannot reproduce the v0.60.0\n * no-events fingerprint, so the regression that motivated this whole\n * module is otherwise uncoverable. */\nexport type IDBOpenFn = (name: string, version: number) => IDBOpenDBRequest;\n\nconst defaultOpenFn: IDBOpenFn = (name, version) => indexedDB.open(name, version);\n\n/** Open an IndexedDB database with the Polly#107 timeout guard. */\nexport function openIDB(\n options: OpenIDBOptions,\n openFn: IDBOpenFn = defaultOpenFn\n): Promise<IDBDatabase> {\n return new Promise<IDBDatabase>((resolve, reject) => {\n const start = Date.now();\n const request = openFn(options.name, options.version);\n let settled = false;\n const elapsed = () => Date.now() - start;\n const rejectWith = (reason: IDBOpenFailureReason, cause?: unknown) => {\n if (settled) return;\n settled = true;\n clearTimeout(timer);\n reject(new IDBOpenError(reason, options.name, elapsed(), cause));\n };\n const timer = setTimeout(() => rejectWith(\"timeout\"), IDB_OPEN_TIMEOUT_MS);\n\n request.onerror = () => rejectWith(\"error\", request.error);\n request.onblocked = () => rejectWith(\"blocked\");\n request.onsuccess = () => {\n if (settled) return;\n settled = true;\n clearTimeout(timer);\n resolve(request.result);\n };\n request.onupgradeneeded = (event) => {\n const db = (event.target as unknown as IDBOpenDBRequest).result;\n options.upgrade(db, event);\n };\n });\n}\n\n/** One-shot open caching with retry-on-failure. Caller owns a `{ promise }`\n * cell; first call opens, subsequent calls return the cached promise, and\n * a rejected open clears the cache so the next call can retry instead of\n * being poisoned by one transient failure. */\nexport function cachedOpen(\n ref: { promise: Promise<IDBDatabase> | null },\n options: OpenIDBOptions,\n openFn?: IDBOpenFn\n): Promise<IDBDatabase> {\n if (ref.promise) return ref.promise;\n const pending = openIDB(options, openFn);\n pending.catch(() => {\n if (ref.promise === pending) ref.promise = null;\n });\n ref.promise = pending;\n return pending;\n}\n\n/** Promise-wrap a single `IDBRequest` (`get`, `put`, `count`, `delete`…). */\nexport function runRequest<T>(request: IDBRequest<T>): Promise<T> {\n return new Promise<T>((resolve, reject) => {\n request.onsuccess = () => resolve(request.result);\n request.onerror = () => reject(request.error);\n });\n}\n\n/** Run a transaction to completion. Resolves on `tx.oncomplete` (write\n * durability), not on individual request success. Throws inside `fn`\n * abort the transaction and reject. */\nexport function runTx(\n db: IDBDatabase,\n storeName: string,\n mode: IDBTransactionMode,\n fn: (store: IDBObjectStore) => void\n): Promise<void> {\n return new Promise<void>((resolve, reject) => {\n const tx = db.transaction(storeName, mode);\n const store = tx.objectStore(storeName);\n tx.oncomplete = () => resolve();\n tx.onerror = () => reject(tx.error);\n tx.onabort = () => reject(tx.error);\n try {\n fn(store);\n } catch (err) {\n try {\n tx.abort();\n } catch {\n /* already aborted */\n }\n reject(err);\n }\n });\n}\n\n/** Walk every record in a store. */\nexport function iterateCursor<V>(\n db: IDBDatabase,\n storeName: string,\n visit: (key: IDBValidKey, value: V) => void\n): Promise<void> {\n return new Promise<void>((resolve, reject) => {\n const tx = db.transaction(storeName, \"readonly\");\n const store = tx.objectStore(storeName);\n const request = store.openCursor();\n request.onsuccess = () => {\n const cursor = request.result;\n if (!cursor) return resolve();\n visit(cursor.key, cursor.value);\n cursor.continue();\n };\n request.onerror = () => reject(request.error);\n });\n}\n",
|
|
6
|
+
"// Storage adapters for different execution contexts\n// Automatically chooses the right storage mechanism based on environment\n\nimport { cachedOpen, runRequest, runTx } from \"./idb-helpers\";\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 readonly dbName: string;\n private readonly storeName = \"state\";\n private readonly dbRef: { promise: Promise<IDBDatabase> | null } = { promise: null };\n\n constructor(dbName = \"polly-state\") {\n this.dbName = dbName;\n }\n\n private getDB(): Promise<IDBDatabase> {\n return cachedOpen(this.dbRef, {\n name: this.dbName,\n version: 1,\n upgrade: (db) => {\n if (!db.objectStoreNames.contains(this.storeName)) {\n db.createObjectStore(this.storeName);\n }\n },\n });\n }\n\n async get<T = unknown>(keys: string[]): Promise<Record<string, T>> {\n try {\n const db = await this.getDB();\n const tx = db.transaction(this.storeName, \"readonly\");\n const store = tx.objectStore(this.storeName);\n const pairs = await Promise.all(\n keys.map(async (key) => [key, await runRequest<T | undefined>(store.get(key))] as const)\n );\n const result: Record<string, T> = {};\n for (const [key, value] of pairs) {\n if (value !== undefined) result[key] = value;\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 runTx(db, this.storeName, \"readwrite\", (store) => {\n for (const [key, value] of Object.entries(items)) {\n store.put(value, key);\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 runTx(db, this.storeName, \"readwrite\", (store) => {\n for (const key of keys) store.delete(key);\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 unknown 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 unknown 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",
|
|
7
7
|
"// 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// 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// 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 unknown as (message: StateSyncMessage<unknown>) => void);\n\n // Return cleanup function\n return () => {\n const index = this.listeners.indexOf(\n callback as unknown 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 console.warn(\"[SyncAdapter] BroadcastChannel not available\");\n } else {\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 }\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 unknown as (message: StateSyncMessage<unknown>) => void);\n\n // Return cleanup function\n return () => {\n const index = this.listeners.indexOf(\n callback as unknown 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",
|
|
8
8
|
"// 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_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_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_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",
|
|
9
|
-
"// 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 unknown as SignalWithLoaded<T>).loaded = entry.loaded;\n }\n\n return sig as unknown 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 unknown as SignalWithLoaded<T>).loaded = entry.loaded;\n }\n\n return sig as unknown 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\nexport function 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 (\n !deepEqual(\n (a as unknown as Record<string, unknown>)[key],\n (b as unknown as Record<string, unknown>)[key]\n )\n )\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 // 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 unknown 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 unknown as T;\n (sig as unknown 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 unknown as SignalWithVerify<T>;\n if (verifySignal.verify) {\n // nosemgrep: javascript.lang.security.insecure-object-assign.insecure-object-assign — internal mirror; value is a deep JSON clone, not arbitrary external input.\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 unknown as T, message.clock);\n }\n });\n }\n\n stateRegistry.set(key, entry as unknown as StateEntry<unknown>);\n return sig;\n}\n\nfunction isPlainObject(value: unknown): value is Record<string, unknown> {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\n/**\n * Backfill keys added to the state shape since a value was persisted.\n *\n * Stored plain objects are shallow-merged over the current defaults so that a\n * field added in a later release rehydrates to its default rather than\n * `undefined`. Primitives and arrays are replaced wholesale — merging those\n * would corrupt them.\n */\nfunction reconcileWithDefaults<T>(defaults: T, stored: unknown): T {\n if (isPlainObject(defaults) && isPlainObject(stored)) {\n return { ...defaults, ...stored } as T;\n }\n return stored as T;\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 // sig still holds the initial value here, so it is the default base\n // for backfilling keys the persisted blob predates.\n const storedValue = reconcileWithDefaults(sig.peek(), 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 unknown as T;\n }\n }\n\n if (result[`${key}:clock`] !== undefined) {\n entry.clock = result[`${key}:clock`] as unknown 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 unknown as Signal<T> | undefined;\n}\n\n/**\n * Clear state registry (useful for testing)\n */\nexport function clearStateRegistry(): void {\n stateRegistry.clear();\n}\n",
|
|
9
|
+
"// 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 unknown as SignalWithLoaded<T>).loaded = entry.loaded;\n }\n\n return sig as unknown 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 unknown as SignalWithLoaded<T>).loaded = entry.loaded;\n }\n\n return sig as unknown 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\n/** Server-side authoritative state, keyed for stable identity across the\n * process. Unlike the browser primitives it does NOT sync across contexts or\n * persist to chrome.storage — the server holds the canonical replica and is\n * the always-on peer; connected clients receive derived updates through the\n * Elysia `polly()` plugin's effect/broadcast layer rather than through this\n * signal directly. Used inside `polly({ state: { server: { ... } } })` and\n * read on the request context as `pollyState.server.<key>.value`. */\nconst serverStateRegistry = new Map<string, Signal<unknown>>();\n\n/**\n * Reactive server-only state. The first call for a `key` creates the signal;\n * later calls with the same `key` return that same signal, so a module that\n * declares its server state once gets a stable reference everywhere it is\n * read.\n *\n * @param key - Stable identifier for this server state (e.g. \"db\")\n * @param initialValue - Value held until the server mutates it\n * @returns A reactive signal holding the authoritative server value\n *\n * @example\n * ```typescript\n * import { $serverState } from \"@fairfox/polly\";\n *\n * const db = { todos: signal([]), nextId: 1 };\n * polly({ state: { server: { db: $serverState(\"db\", db) } } });\n * // in a handler: ctx.pollyState.server.db.value.todos.value = [...]\n * ```\n */\nexport function $serverState<T>(key: string, initialValue: T): Signal<T> {\n const existing = serverStateRegistry.get(key);\n // The registry erases T (it stores Signal<unknown>); the key is the\n // caller's contract that the stored signal carries their T.\n if (existing) return existing as unknown as Signal<T>;\n const sig = signal(initialValue);\n serverStateRegistry.set(key, sig as unknown as Signal<unknown>);\n return sig;\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\nexport function 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 (\n !deepEqual(\n (a as unknown as Record<string, unknown>)[key],\n (b as unknown as Record<string, unknown>)[key]\n )\n )\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 // 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 unknown 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 unknown as T;\n (sig as unknown 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 unknown as SignalWithVerify<T>;\n if (verifySignal.verify) {\n // nosemgrep: javascript.lang.security.insecure-object-assign.insecure-object-assign — internal mirror; value is a deep JSON clone, not arbitrary external input.\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 unknown as T, message.clock);\n }\n });\n }\n\n stateRegistry.set(key, entry as unknown as StateEntry<unknown>);\n return sig;\n}\n\nfunction isPlainObject(value: unknown): value is Record<string, unknown> {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\n/**\n * Backfill keys added to the state shape since a value was persisted.\n *\n * Stored plain objects are shallow-merged over the current defaults so that a\n * field added in a later release rehydrates to its default rather than\n * `undefined`. Primitives and arrays are replaced wholesale — merging those\n * would corrupt them.\n */\nfunction reconcileWithDefaults<T>(defaults: T, stored: unknown): T {\n // Persisted values are trusted to be the T this key was registered with;\n // schema drift is handled by the merge, not by runtime validation.\n if (isPlainObject(defaults) && isPlainObject(stored)) {\n return { ...defaults, ...stored } as unknown as T;\n }\n return stored as unknown as T;\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 // sig still holds the initial value here, so it is the default base\n // for backfilling keys the persisted blob predates.\n const storedValue = reconcileWithDefaults(sig.peek(), 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 unknown as T;\n }\n }\n\n if (result[`${key}:clock`] !== undefined) {\n entry.clock = result[`${key}:clock`] as unknown 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 unknown as Signal<T> | undefined;\n}\n\n/**\n * Clear state registry (useful for testing)\n */\nexport function clearStateRegistry(): void {\n stateRegistry.clear();\n}\n",
|
|
10
10
|
"// Shared application state\n\nimport { signal } from \"@preact/signals\";\nimport { $sharedState, $syncedState } from \"../lib/state\";\nimport type { Settings } from \"../types/messages\";\nimport { defaultSettings as _defaultSettings } from \"../types/messages\";\n\n// Re-export for convenience\nexport { defaultSettings } from \"../types/messages\";\n\n// Synced across all contexts, persisted to storage\nexport const settings = $sharedState<Settings>(\"app-settings\", _defaultSettings);\n\n// Synced but not persisted\nexport const currentTab = $syncedState<number | null>(\"current-tab\", null);\n\n// Local to each context (not synced) - use regular signal\nexport const uiState = signal({\n sidebarOpen: false,\n selectedPanel: \"main\" as unknown as \"main\" | \"settings\" | \"debug\",\n});\n"
|
|
11
11
|
],
|
|
12
|
-
"mappings": "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",
|
|
13
|
-
"debugId": "
|
|
12
|
+
"mappings": "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",
|
|
13
|
+
"debugId": "31635B335DEF6E3D64756E2164756E21",
|
|
14
14
|
"names": []
|
|
15
15
|
}
|
|
@@ -54,6 +54,17 @@ function copyTemplateFiles(sourceDir, targetDir, projectName, relativePath = "")
|
|
|
54
54
|
}
|
|
55
55
|
}
|
|
56
56
|
}
|
|
57
|
+
var ALL_PROJECT_TYPES = ["extension", "pwa", "websocket", "generic"];
|
|
58
|
+
function getAvailableTypes(baseDir = import.meta.dir) {
|
|
59
|
+
const templatesRoot = join(baseDir, "..", "templates");
|
|
60
|
+
let present;
|
|
61
|
+
try {
|
|
62
|
+
present = readdirSync(templatesRoot, { withFileTypes: true }).filter((entry) => entry.isDirectory()).map((entry) => entry.name);
|
|
63
|
+
} catch {
|
|
64
|
+
return [];
|
|
65
|
+
}
|
|
66
|
+
return ALL_PROJECT_TYPES.filter((type) => present.includes(type));
|
|
67
|
+
}
|
|
57
68
|
function getTemplateDir(projectType, baseDir) {
|
|
58
69
|
return join(baseDir, "..", "templates", projectType);
|
|
59
70
|
}
|
|
@@ -77,11 +88,21 @@ function validateProjectName(name) {
|
|
|
77
88
|
}
|
|
78
89
|
|
|
79
90
|
// tools/init/src/cli.ts
|
|
91
|
+
function isAvailableType(value, available) {
|
|
92
|
+
const names = available;
|
|
93
|
+
return names.includes(value);
|
|
94
|
+
}
|
|
80
95
|
async function main() {
|
|
81
96
|
const args = process.argv.slice(2);
|
|
82
97
|
const cwd = process.cwd();
|
|
83
98
|
const projectName = args[0] || "my-project";
|
|
84
|
-
const typeArg = args.find((arg) => arg.startsWith("--type="))?.split("=")[1] || "
|
|
99
|
+
const typeArg = args.find((arg) => arg.startsWith("--type="))?.split("=")[1] || "pwa";
|
|
100
|
+
const available = getAvailableTypes(import.meta.dir);
|
|
101
|
+
if (!isAvailableType(typeArg, available)) {
|
|
102
|
+
console.log(`\x1B[31m✗ Unknown or unavailable project type '${typeArg}'. Available: ${available.join(", ")}\x1B[0m
|
|
103
|
+
`);
|
|
104
|
+
process.exit(1);
|
|
105
|
+
}
|
|
85
106
|
const projectType = typeArg;
|
|
86
107
|
const validation = validateProjectName(projectName);
|
|
87
108
|
if (!validation.valid) {
|
|
@@ -105,4 +126,4 @@ async function main() {
|
|
|
105
126
|
}
|
|
106
127
|
main();
|
|
107
128
|
|
|
108
|
-
//# debugId=
|
|
129
|
+
//# debugId=0981B588805418F764756E2164756E21
|
|
@@ -2,10 +2,10 @@
|
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../tools/init/src/cli.ts", "../tools/init/src/template-utils.ts"],
|
|
4
4
|
"sourcesContent": [
|
|
5
|
-
"#!/usr/bin/env bun\n// CLI for project initialization\n\nimport { existsSync } from \"node:fs\";\nimport {\n getTemplateDir,\n type ProjectType,\n scaffoldFromTemplate,\n validateProjectName,\n} from \"./template-utils.ts\";\n\nasync function main() {\n const args = process.argv.slice(2);\n const cwd = process.cwd();\n\n // Parse arguments\n const projectName = args[0] || \"my-project\";\n const typeArg = args.find((arg) => arg.startsWith(\"--type=\"))?.split(\"=\")[1] || \"
|
|
6
|
-
"/**\n * Template utilities for project initialization\n */\n\nimport { cpSync, existsSync, mkdirSync, readdirSync, readFileSync, writeFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\n\nexport type ProjectType = \"extension\" | \"pwa\" | \"websocket\" | \"generic\";\n\nexport interface TemplateOptions {\n projectName: string;\n projectPath: string;\n projectType: ProjectType;\n templateDir: string;\n}\n\n/**\n * Process template file and replace placeholders\n */\nexport function processTemplate(content: string, projectName: string): string {\n return content\n .replace(/\\{\\{PROJECT_NAME\\}\\}/g, projectName)\n .replace(/\\{\\{PROJECT_NAME_UPPER\\}\\}/g, projectName.toUpperCase())\n .replace(/\\{\\{PROJECT_NAME_CAMEL\\}\\}/g, toCamelCase(projectName));\n}\n\n/**\n * Convert string to camelCase\n */\nfunction toCamelCase(str: string): string {\n return str\n .replace(/[-_](.)/g, (_, char) => char.toUpperCase())\n .replace(/^(.)/, (char) => char.toLowerCase());\n}\n\n/**\n * Copy template directory to project\n */\nexport async function scaffoldFromTemplate(options: TemplateOptions): Promise<void> {\n const { projectName, projectPath, templateDir } = options;\n\n // Create project directory\n mkdirSync(projectPath, { recursive: true });\n\n // Copy all template files\n copyTemplateFiles(templateDir, projectPath, projectName);\n}\n\n/**\n * Recursively copy template files and process them\n */\nfunction copyTemplateFiles(\n sourceDir: string,\n targetDir: string,\n projectName: string,\n relativePath = \"\"\n): void {\n if (!existsSync(sourceDir)) {\n throw new Error(`Template directory not found: ${sourceDir}`);\n }\n\n const entries = readdirSync(sourceDir, { withFileTypes: true });\n\n for (const entry of entries) {\n const sourcePath = join(sourceDir, entry.name);\n const targetPath = join(targetDir, relativePath, entry.name.replace(\".template\", \"\"));\n\n if (entry.isDirectory()) {\n // Create directory and recurse\n mkdirSync(targetPath, { recursive: true });\n copyTemplateFiles(sourcePath, targetDir, projectName, join(relativePath, entry.name));\n } else if (entry.name.endsWith(\".template\")) {\n // Process template file\n const content = readFileSync(sourcePath, \"utf-8\");\n const processed = processTemplate(content, projectName);\n writeFileSync(targetPath, processed, \"utf-8\");\n } else {\n // Copy binary files as-is\n cpSync(sourcePath, targetPath);\n }\n }\n}\n\n/**\n * Get
|
|
5
|
+
"#!/usr/bin/env bun\n// CLI for project initialization\n\nimport { existsSync } from \"node:fs\";\nimport {\n getAvailableTypes,\n getTemplateDir,\n type ProjectType,\n scaffoldFromTemplate,\n validateProjectName,\n} from \"./template-utils.ts\";\n\nfunction isAvailableType(value: string, available: ProjectType[]): value is ProjectType {\n const names: readonly string[] = available;\n return names.includes(value);\n}\n\nasync function main() {\n const args = process.argv.slice(2);\n const cwd = process.cwd();\n\n // Parse arguments\n const projectName = args[0] || \"my-project\";\n const typeArg = args.find((arg) => arg.startsWith(\"--type=\"))?.split(\"=\")[1] || \"pwa\";\n\n // Validate the requested type against the templates that actually ship,\n // before creating any directory. Asking for a type with no template used\n // to fail deep in the copy with a raw \"Template directory not found\" after\n // the project dir had already been made.\n const available = getAvailableTypes(import.meta.dir);\n if (!isAvailableType(typeArg, available)) {\n console.log(\n `\\x1b[31m✗ Unknown or unavailable project type '${typeArg}'. Available: ${available.join(\", \")}\\x1b[0m\\n`\n );\n process.exit(1);\n }\n const projectType = typeArg;\n\n // Validate project name\n const validation = validateProjectName(projectName);\n if (!validation.valid) {\n console.log(`\\x1b[31m✗ ${validation.error}\\x1b[0m\\n`);\n process.exit(1);\n }\n\n const projectPath = `${cwd}/${projectName}`;\n\n // Check if directory already exists\n if (existsSync(projectPath)) {\n console.log(`\\x1b[31m✗ Directory '${projectName}' already exists\\x1b[0m\\n`);\n process.exit(1);\n }\n\n // Get template directory\n const templateDir = getTemplateDir(projectType, import.meta.dir);\n\n // Scaffold project\n await scaffoldFromTemplate({\n projectName,\n projectPath,\n projectType,\n templateDir,\n });\n}\n\nmain();\n",
|
|
6
|
+
"/**\n * Template utilities for project initialization\n */\n\nimport { cpSync, existsSync, mkdirSync, readdirSync, readFileSync, writeFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\n\nexport type ProjectType = \"extension\" | \"pwa\" | \"websocket\" | \"generic\";\n\nexport interface TemplateOptions {\n projectName: string;\n projectPath: string;\n projectType: ProjectType;\n templateDir: string;\n}\n\n/**\n * Process template file and replace placeholders\n */\nexport function processTemplate(content: string, projectName: string): string {\n return content\n .replace(/\\{\\{PROJECT_NAME\\}\\}/g, projectName)\n .replace(/\\{\\{PROJECT_NAME_UPPER\\}\\}/g, projectName.toUpperCase())\n .replace(/\\{\\{PROJECT_NAME_CAMEL\\}\\}/g, toCamelCase(projectName));\n}\n\n/**\n * Convert string to camelCase\n */\nfunction toCamelCase(str: string): string {\n return str\n .replace(/[-_](.)/g, (_, char) => char.toUpperCase())\n .replace(/^(.)/, (char) => char.toLowerCase());\n}\n\n/**\n * Copy template directory to project\n */\nexport async function scaffoldFromTemplate(options: TemplateOptions): Promise<void> {\n const { projectName, projectPath, templateDir } = options;\n\n // Create project directory\n mkdirSync(projectPath, { recursive: true });\n\n // Copy all template files\n copyTemplateFiles(templateDir, projectPath, projectName);\n}\n\n/**\n * Recursively copy template files and process them\n */\nfunction copyTemplateFiles(\n sourceDir: string,\n targetDir: string,\n projectName: string,\n relativePath = \"\"\n): void {\n if (!existsSync(sourceDir)) {\n throw new Error(`Template directory not found: ${sourceDir}`);\n }\n\n const entries = readdirSync(sourceDir, { withFileTypes: true });\n\n for (const entry of entries) {\n const sourcePath = join(sourceDir, entry.name);\n const targetPath = join(targetDir, relativePath, entry.name.replace(\".template\", \"\"));\n\n if (entry.isDirectory()) {\n // Create directory and recurse\n mkdirSync(targetPath, { recursive: true });\n copyTemplateFiles(sourcePath, targetDir, projectName, join(relativePath, entry.name));\n } else if (entry.name.endsWith(\".template\")) {\n // Process template file\n const content = readFileSync(sourcePath, \"utf-8\");\n const processed = processTemplate(content, projectName);\n writeFileSync(targetPath, processed, \"utf-8\");\n } else {\n // Copy binary files as-is\n cpSync(sourcePath, targetPath);\n }\n }\n}\n\n/** Every project type the scaffolder knows how to name. A type is only\n * *available* once a matching template directory ships next to this\n * module — see {@link getAvailableTypes}. */\nconst ALL_PROJECT_TYPES: ProjectType[] = [\"extension\", \"pwa\", \"websocket\", \"generic\"];\n\n/**\n * Get the project types that can actually be scaffolded — the subset of\n * {@link ALL_PROJECT_TYPES} that has a template directory on disk. Derived\n * from the filesystem so the list cannot drift from reality: advertising a\n * type whose template was never written is exactly the gap that left\n * `polly init` (default `extension`) pointing at a missing directory.\n */\nexport function getAvailableTypes(baseDir: string = import.meta.dir): ProjectType[] {\n const templatesRoot = join(baseDir, \"..\", \"templates\");\n let present: string[];\n try {\n present = readdirSync(templatesRoot, { withFileTypes: true })\n .filter((entry) => entry.isDirectory())\n .map((entry) => entry.name);\n } catch {\n return [];\n }\n return ALL_PROJECT_TYPES.filter((type) => present.includes(type));\n}\n\n/**\n * Get template directory for project type\n */\nexport function getTemplateDir(projectType: ProjectType, baseDir: string): string {\n return join(baseDir, \"..\", \"templates\", projectType);\n}\n\n/**\n * Validate project name\n */\nexport function validateProjectName(name: string): {\n valid: boolean;\n error?: string;\n} {\n if (!name || name.trim().length === 0) {\n return { valid: false, error: \"Project name cannot be empty\" };\n }\n\n if (!/^[a-z0-9-_]+$/i.test(name)) {\n return {\n valid: false,\n error: \"Project name can only contain letters, numbers, hyphens, and underscores\",\n };\n }\n\n if (name.startsWith(\"-\") || name.startsWith(\"_\")) {\n return {\n valid: false,\n error: \"Project name cannot start with a hyphen or underscore\",\n };\n }\n\n return { valid: true };\n}\n"
|
|
7
7
|
],
|
|
8
|
-
"mappings": ";;;;;;;;;;;;;;;;;;;;AAGA,uBAAS;;;ACCT;AACA;AAcO,SAAS,eAAe,CAAC,SAAiB,aAA6B;AAAA,EAC5E,OAAO,QACJ,QAAQ,yBAAyB,WAAW,EAC5C,QAAQ,+BAA+B,YAAY,YAAY,CAAC,EAChE,QAAQ,+BAA+B,YAAY,WAAW,CAAC;AAAA;AAMpE,SAAS,WAAW,CAAC,KAAqB;AAAA,EACxC,OAAO,IACJ,QAAQ,YAAY,CAAC,GAAG,SAAS,KAAK,YAAY,CAAC,EACnD,QAAQ,QAAQ,CAAC,SAAS,KAAK,YAAY,CAAC;AAAA;AAMjD,eAAsB,oBAAoB,CAAC,SAAyC;AAAA,EAClF,QAAQ,aAAa,aAAa,gBAAgB;AAAA,EAGlD,UAAU,aAAa,EAAE,WAAW,KAAK,CAAC;AAAA,EAG1C,kBAAkB,aAAa,aAAa,WAAW;AAAA;AAMzD,SAAS,iBAAiB,CACxB,WACA,WACA,aACA,eAAe,IACT;AAAA,EACN,IAAI,CAAC,WAAW,SAAS,GAAG;AAAA,IAC1B,MAAM,IAAI,MAAM,iCAAiC,WAAW;AAAA,EAC9D;AAAA,EAEA,MAAM,UAAU,YAAY,WAAW,EAAE,eAAe,KAAK,CAAC;AAAA,EAE9D,WAAW,SAAS,SAAS;AAAA,IAC3B,MAAM,aAAa,KAAK,WAAW,MAAM,IAAI;AAAA,IAC7C,MAAM,aAAa,KAAK,WAAW,cAAc,MAAM,KAAK,QAAQ,aAAa,EAAE,CAAC;AAAA,IAEpF,IAAI,MAAM,YAAY,GAAG;AAAA,MAEvB,UAAU,YAAY,EAAE,WAAW,KAAK,CAAC;AAAA,MACzC,kBAAkB,YAAY,WAAW,aAAa,KAAK,cAAc,MAAM,IAAI,CAAC;AAAA,IACtF,EAAO,SAAI,MAAM,KAAK,SAAS,WAAW,GAAG;AAAA,MAE3C,MAAM,UAAU,aAAa,YAAY,OAAO;AAAA,MAChD,MAAM,YAAY,gBAAgB,SAAS,WAAW;AAAA,MACtD,cAAc,YAAY,WAAW,OAAO;AAAA,IAC9C,EAAO;AAAA,MAEL,OAAO,YAAY,UAAU;AAAA;AAAA,EAEjC;AAAA;
|
|
9
|
-
"debugId": "
|
|
8
|
+
"mappings": "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",
|
|
9
|
+
"debugId": "0981B588805418F764756E2164756E21",
|
|
10
10
|
"names": []
|
|
11
11
|
}
|
|
@@ -7,7 +7,7 @@
|
|
|
7
7
|
"build": "bun build.ts",
|
|
8
8
|
"serve": "bun --hot server.ts",
|
|
9
9
|
"test": "bun test",
|
|
10
|
-
"typecheck": "bun tsc --noEmit",
|
|
10
|
+
"typecheck": "bun tsc --noEmit -p tsconfig.json && bun tsc --noEmit -p tsconfig.worker.json",
|
|
11
11
|
"verify": "polly verify",
|
|
12
12
|
"verify:setup": "polly verify --setup",
|
|
13
13
|
"visualize": "polly visualize",
|
|
@@ -3,8 +3,19 @@
|
|
|
3
3
|
* Handles background tasks, caching, and push notifications
|
|
4
4
|
*/
|
|
5
5
|
|
|
6
|
-
/// <reference lib="
|
|
7
|
-
|
|
6
|
+
/// <reference lib="webworker" />
|
|
7
|
+
|
|
8
|
+
// `export {}` makes this file a module so its top-level `sw` is module-scoped
|
|
9
|
+
// rather than a global that collides with the shared worker's `sw` when both
|
|
10
|
+
// are type-checked together. The bundler emits a standalone worker regardless.
|
|
11
|
+
export {};
|
|
12
|
+
|
|
13
|
+
// In a service worker `self` is the ServiceWorkerGlobalScope, but the
|
|
14
|
+
// webworker lib types the ambient `self` as the base WorkerGlobalScope. Narrow
|
|
15
|
+
// it once here with a cast (a `declare const self` redeclaration collides with
|
|
16
|
+
// the lib's own `self`); `sw` then carries skipWaiting/clients/registration and
|
|
17
|
+
// the correctly-typed event map (install/fetch/push/...).
|
|
18
|
+
const sw = self as unknown as ServiceWorkerGlobalScope;
|
|
8
19
|
|
|
9
20
|
console.log("[SW] Service Worker loading...");
|
|
10
21
|
|
|
@@ -13,7 +24,7 @@ const CACHE_NAME = "{{PROJECT_NAME}}-v1";
|
|
|
13
24
|
const urlsToCache = ["/", "/index.html", "/src/main.ts"];
|
|
14
25
|
|
|
15
26
|
// Install event - cache resources
|
|
16
|
-
|
|
27
|
+
sw.addEventListener("install", (event) => {
|
|
17
28
|
console.log("[SW] Installing...");
|
|
18
29
|
|
|
19
30
|
event.waitUntil(
|
|
@@ -24,11 +35,11 @@ self.addEventListener("install", (event) => {
|
|
|
24
35
|
);
|
|
25
36
|
|
|
26
37
|
// Activate immediately
|
|
27
|
-
void
|
|
38
|
+
void sw.skipWaiting();
|
|
28
39
|
});
|
|
29
40
|
|
|
30
41
|
// Activate event - clean up old caches
|
|
31
|
-
|
|
42
|
+
sw.addEventListener("activate", (event) => {
|
|
32
43
|
console.log("[SW] Activating...");
|
|
33
44
|
|
|
34
45
|
event.waitUntil(
|
|
@@ -42,11 +53,11 @@ self.addEventListener("activate", (event) => {
|
|
|
42
53
|
);
|
|
43
54
|
|
|
44
55
|
// Take control immediately
|
|
45
|
-
void
|
|
56
|
+
void sw.clients.claim();
|
|
46
57
|
});
|
|
47
58
|
|
|
48
59
|
// Fetch event - serve from cache, fallback to network
|
|
49
|
-
|
|
60
|
+
sw.addEventListener("fetch", (event) => {
|
|
50
61
|
event.respondWith(
|
|
51
62
|
caches.match(event.request).then((response) => {
|
|
52
63
|
// Cache hit - return response
|
|
@@ -81,7 +92,7 @@ self.addEventListener("fetch", (event) => {
|
|
|
81
92
|
});
|
|
82
93
|
|
|
83
94
|
// Message handling
|
|
84
|
-
|
|
95
|
+
sw.addEventListener("message", (event) => {
|
|
85
96
|
console.log("[SW] Received message:", event.data);
|
|
86
97
|
|
|
87
98
|
switch (event.data.type) {
|
|
@@ -94,7 +105,7 @@ self.addEventListener("message", (event) => {
|
|
|
94
105
|
});
|
|
95
106
|
|
|
96
107
|
// Also broadcast to all clients
|
|
97
|
-
void
|
|
108
|
+
void sw.clients.matchAll().then((clients) => {
|
|
98
109
|
for (const client of clients) {
|
|
99
110
|
client.postMessage({
|
|
100
111
|
type: "PONG",
|
|
@@ -107,7 +118,7 @@ self.addEventListener("message", (event) => {
|
|
|
107
118
|
|
|
108
119
|
case "BROADCAST":
|
|
109
120
|
// Broadcast to all clients
|
|
110
|
-
void
|
|
121
|
+
void sw.clients.matchAll().then((clients) => {
|
|
111
122
|
console.log(`[SW] Broadcasting to ${clients.length} clients`);
|
|
112
123
|
for (const client of clients) {
|
|
113
124
|
client.postMessage({
|
|
@@ -121,7 +132,7 @@ self.addEventListener("message", (event) => {
|
|
|
121
132
|
break;
|
|
122
133
|
|
|
123
134
|
case "SKIP_WAITING":
|
|
124
|
-
void
|
|
135
|
+
void sw.skipWaiting();
|
|
125
136
|
break;
|
|
126
137
|
|
|
127
138
|
default:
|
|
@@ -130,7 +141,7 @@ self.addEventListener("message", (event) => {
|
|
|
130
141
|
});
|
|
131
142
|
|
|
132
143
|
// Push notification event (example)
|
|
133
|
-
|
|
144
|
+
sw.addEventListener("push", (event) => {
|
|
134
145
|
console.log("[SW] Push received");
|
|
135
146
|
|
|
136
147
|
const data = event.data?.json() ?? {};
|
|
@@ -142,16 +153,16 @@ self.addEventListener("push", (event) => {
|
|
|
142
153
|
data: data,
|
|
143
154
|
};
|
|
144
155
|
|
|
145
|
-
event.waitUntil(
|
|
156
|
+
event.waitUntil(sw.registration.showNotification(title, options));
|
|
146
157
|
});
|
|
147
158
|
|
|
148
159
|
// Notification click event
|
|
149
|
-
|
|
160
|
+
sw.addEventListener("notificationclick", (event) => {
|
|
150
161
|
console.log("[SW] Notification clicked");
|
|
151
162
|
|
|
152
163
|
event.notification.close();
|
|
153
164
|
|
|
154
|
-
event.waitUntil(
|
|
165
|
+
event.waitUntil(sw.clients.openWindow("/"));
|
|
155
166
|
});
|
|
156
167
|
|
|
157
168
|
console.log("[SW] Service Worker loaded");
|
|
@@ -3,8 +3,18 @@
|
|
|
3
3
|
* Shared state and coordination across multiple tabs/windows
|
|
4
4
|
*/
|
|
5
5
|
|
|
6
|
-
/// <reference lib="
|
|
7
|
-
|
|
6
|
+
/// <reference lib="webworker" />
|
|
7
|
+
|
|
8
|
+
// `export {}` makes this file a module so its top-level `sw` is module-scoped
|
|
9
|
+
// rather than a global that collides with the service worker's `sw` when both
|
|
10
|
+
// are type-checked together. The bundler emits a standalone worker regardless.
|
|
11
|
+
export {};
|
|
12
|
+
|
|
13
|
+
// In a shared worker `self` is the SharedWorkerGlobalScope, but the webworker
|
|
14
|
+
// lib types the ambient `self` as the base WorkerGlobalScope. Narrow it once
|
|
15
|
+
// here with a cast (a `declare const self` redeclaration collides with the
|
|
16
|
+
// lib's own `self`); `sw` then carries the correctly-typed "connect" event.
|
|
17
|
+
const sw = self as unknown as SharedWorkerGlobalScope;
|
|
8
18
|
|
|
9
19
|
console.log("[Shared Worker] Starting...");
|
|
10
20
|
|
|
@@ -26,7 +36,7 @@ const state: SharedState = {
|
|
|
26
36
|
};
|
|
27
37
|
|
|
28
38
|
// Connection event - when a new tab connects
|
|
29
|
-
|
|
39
|
+
sw.addEventListener("connect", (event) => {
|
|
30
40
|
const port = event.ports[0];
|
|
31
41
|
ports.push(port);
|
|
32
42
|
state.connectedTabs = ports.length;
|
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
"compilerOptions": {
|
|
3
3
|
"target": "ES2022",
|
|
4
4
|
"module": "ESNext",
|
|
5
|
-
"lib": ["ES2022", "DOM", "DOM.Iterable"
|
|
5
|
+
"lib": ["ES2022", "DOM", "DOM.Iterable"],
|
|
6
6
|
"moduleResolution": "bundler",
|
|
7
7
|
"strict": true,
|
|
8
8
|
"esModuleInterop": true,
|
|
@@ -14,5 +14,5 @@
|
|
|
14
14
|
"noEmit": true
|
|
15
15
|
},
|
|
16
16
|
"include": ["src/**/*"],
|
|
17
|
-
"exclude": ["node_modules", "dist"]
|
|
17
|
+
"exclude": ["node_modules", "dist", "src/service-worker.ts", "src/shared-worker.ts"]
|
|
18
18
|
}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
{
|
|
2
|
+
"compilerOptions": {
|
|
3
|
+
"target": "ES2022",
|
|
4
|
+
"module": "ESNext",
|
|
5
|
+
"lib": ["ES2022", "WebWorker"],
|
|
6
|
+
"moduleResolution": "bundler",
|
|
7
|
+
"strict": true,
|
|
8
|
+
"esModuleInterop": true,
|
|
9
|
+
"skipLibCheck": true,
|
|
10
|
+
"forceConsistentCasingInFileNames": true,
|
|
11
|
+
"resolveJsonModule": true,
|
|
12
|
+
"allowSyntheticDefaultImports": true,
|
|
13
|
+
"isolatedModules": true,
|
|
14
|
+
"noEmit": true
|
|
15
|
+
},
|
|
16
|
+
"include": ["src/service-worker.ts", "src/shared-worker.ts"]
|
|
17
|
+
}
|
|
@@ -244,11 +244,14 @@ async function done() {
|
|
|
244
244
|
}
|
|
245
245
|
}
|
|
246
246
|
const report = window["__pollyReport"];
|
|
247
|
-
if (
|
|
247
|
+
if (!isReportFunction(report)) {
|
|
248
248
|
throw new Error("harness.done(): window.__pollyReport is not defined. " + "This harness must be driven by the Polly browser test runner.");
|
|
249
249
|
}
|
|
250
250
|
report(results);
|
|
251
251
|
}
|
|
252
|
+
function isReportFunction(value) {
|
|
253
|
+
return typeof value === "function";
|
|
254
|
+
}
|
|
252
255
|
async function waitFor(predicate, timeoutMs = 5000, intervalMs = 25) {
|
|
253
256
|
const deadline = Date.now() + timeoutMs;
|
|
254
257
|
while (Date.now() < deadline) {
|
|
@@ -268,4 +271,4 @@ export {
|
|
|
268
271
|
cleanup
|
|
269
272
|
};
|
|
270
273
|
|
|
271
|
-
//# debugId=
|
|
274
|
+
//# debugId=24287A6E9F4D05B664756E2164756E21
|