@fairfox/polly 0.20.0 → 0.21.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +11 -0
- package/dist/src/background/index.js +22 -12
- package/dist/src/background/index.js.map +8 -8
- package/dist/src/background/message-router.js +22 -12
- package/dist/src/background/message-router.js.map +8 -8
- package/dist/src/client/index.js +187 -154
- package/dist/src/client/index.js.map +4 -4
- package/dist/src/elysia/index.d.ts +2 -0
- package/dist/src/elysia/index.js +195 -25
- package/dist/src/elysia/index.js.map +8 -5
- package/dist/src/elysia/peer-repo-plugin.d.ts +79 -0
- package/dist/src/elysia/plugin.d.ts +3 -3
- package/dist/src/elysia/signaling-server-plugin.d.ts +121 -0
- package/dist/src/index.d.ts +36 -0
- package/dist/src/index.js +1752 -13
- package/dist/src/index.js.map +31 -13
- package/dist/src/shared/adapters/index.js +22 -12
- package/dist/src/shared/adapters/index.js.map +7 -7
- package/dist/src/shared/lib/_client-only.d.ts +38 -0
- package/dist/src/shared/lib/access.d.ts +124 -0
- package/dist/src/shared/lib/blob-ref.d.ts +72 -0
- package/dist/src/shared/lib/context-helpers.js +22 -12
- package/dist/src/shared/lib/context-helpers.js.map +8 -8
- package/dist/src/shared/lib/crdt-specialised.d.ts +129 -0
- package/dist/src/shared/lib/crdt-state.d.ts +86 -0
- package/dist/src/shared/lib/encryption.d.ts +117 -0
- package/dist/src/shared/lib/errors.js +19 -9
- package/dist/src/shared/lib/errors.js.map +2 -2
- package/dist/src/shared/lib/mesh-network-adapter.d.ts +130 -0
- package/dist/src/shared/lib/mesh-signaling-client.d.ts +85 -0
- package/dist/src/shared/lib/mesh-state.d.ts +102 -0
- package/dist/src/shared/lib/mesh-webrtc-adapter.d.ts +132 -0
- package/dist/src/shared/lib/message-bus.js +22 -12
- package/dist/src/shared/lib/message-bus.js.map +8 -8
- package/dist/src/shared/lib/migrate-primitive.d.ts +100 -0
- package/dist/src/shared/lib/pairing.d.ts +170 -0
- package/dist/src/shared/lib/peer-relay-adapter.d.ts +80 -0
- package/dist/src/shared/lib/peer-repo-server.d.ts +83 -0
- package/dist/src/shared/lib/peer-state.d.ts +117 -0
- package/dist/src/shared/lib/primitive-registry.d.ts +88 -0
- package/dist/src/shared/lib/resource.js +22 -12
- package/dist/src/shared/lib/resource.js.map +5 -5
- package/dist/src/shared/lib/revocation.d.ts +126 -0
- package/dist/src/shared/lib/schema-version.d.ts +129 -0
- package/dist/src/shared/lib/signing.d.ts +118 -0
- package/dist/src/shared/lib/state.js +22 -12
- package/dist/src/shared/lib/state.js.map +5 -5
- package/dist/src/shared/lib/test-helpers.js +19 -9
- package/dist/src/shared/lib/test-helpers.js.map +2 -2
- package/dist/src/shared/state/app-state.js +22 -12
- package/dist/src/shared/state/app-state.js.map +6 -6
- package/dist/src/shared/types/messages.js +19 -9
- package/dist/src/shared/types/messages.js.map +2 -2
- package/dist/tools/init/src/cli.js +6 -2
- package/dist/tools/init/src/cli.js.map +3 -3
- package/dist/tools/quality/src/index.js +177 -0
- package/dist/tools/quality/src/index.js.map +10 -0
- package/dist/tools/test/src/adapters/index.d.ts +2 -2
- package/dist/tools/test/src/adapters/index.js +19 -9
- package/dist/tools/test/src/adapters/index.js.map +5 -5
- package/dist/tools/test/src/browser/harness.d.ts +80 -0
- package/dist/tools/test/src/browser/index.d.ts +32 -0
- package/dist/tools/test/src/browser/index.js +243 -0
- package/dist/tools/test/src/browser/index.js.map +10 -0
- package/dist/tools/test/src/browser/run.d.ts +26 -0
- package/dist/tools/test/src/index.js +19 -9
- package/dist/tools/test/src/index.js.map +5 -5
- package/dist/tools/test/src/test-utils.js +19 -9
- package/dist/tools/test/src/test-utils.js.map +2 -2
- package/dist/tools/verify/specs/tla/MeshState.cfg +21 -0
- package/dist/tools/verify/specs/tla/MeshState.tla +247 -0
- package/dist/tools/verify/specs/tla/PeerState.cfg +27 -0
- package/dist/tools/verify/specs/tla/PeerState.tla +238 -0
- package/dist/tools/verify/specs/tla/README.md +27 -3
- package/dist/tools/verify/src/cli.js +10 -6
- package/dist/tools/verify/src/cli.js.map +10 -10
- package/dist/tools/verify/src/config.js +19 -9
- package/dist/tools/verify/src/config.js.map +2 -2
- package/dist/tools/visualize/src/cli.js +6 -2
- package/dist/tools/visualize/src/cli.js.map +8 -8
- package/package.json +52 -12
|
@@ -3,17 +3,17 @@
|
|
|
3
3
|
"sources": ["../tools/test/src/adapters/context-menus.mock.ts", "../tools/test/src/adapters/fetch.mock.ts", "../tools/test/src/adapters/logger.mock.ts", "../tools/test/src/adapters/offscreen.mock.ts", "../tools/test/src/adapters/runtime.mock.ts", "../tools/test/src/adapters/storage.mock.ts", "../tools/test/src/adapters/tabs.mock.ts", "../tools/test/src/adapters/window.mock.ts", "../tools/test/src/adapters/index.ts", "../tools/test/src/test-utils.ts"],
|
|
4
4
|
"sourcesContent": [
|
|
5
5
|
"import type { ContextMenusAdapter } from \"@/shared/adapters/context-menus.adapter\";\n\nexport interface MockContextMenus extends ContextMenusAdapter {\n _menus: Map<string, chrome.contextMenus.CreateProperties>;\n}\n\nexport function createMockContextMenus(): MockContextMenus {\n const menus = new Map<string, chrome.contextMenus.CreateProperties>();\n\n return {\n create: async (createProperties: chrome.contextMenus.CreateProperties): Promise<void> => {\n if (createProperties.id) {\n menus.set(createProperties.id, createProperties);\n }\n },\n update: async (\n _id: string,\n _updateProperties: Omit<chrome.contextMenus.CreateProperties, \"id\">\n ): Promise<void> => {\n // Mock implementation\n },\n remove: async (_id: string): Promise<void> => {\n // Mock implementation\n },\n removeAll: async (): Promise<void> => {\n // Mock implementation\n },\n onClicked: (\n _callback: (info: chrome.contextMenus.OnClickData, tab?: chrome.tabs.Tab) => void\n ): void => {\n // Mock implementation\n },\n _menus: menus,\n };\n}\n",
|
|
6
|
-
"import type { FetchAdapter } from \"@/shared/adapters/fetch.adapter\";\n\nexport interface MockFetch extends FetchAdapter {\n _responses: Array<Partial<Response>>;\n _calls: Array<{ input: string | URL; init?: RequestInit }>;\n}\n\nexport function createMockFetch(): MockFetch {\n const responses: Array<Partial<Response>> = [];\n const calls: Array<{ input: string | URL; init?: RequestInit }> = [];\n\n return {\n fetch: async (input: string | URL, init?: RequestInit): Promise<Response> => {\n calls.push({ input, ...(init && { init }) });\n\n const mockResponse = responses.shift() || {\n ok: true,\n status: 200,\n headers: new Headers(),\n statusText: \"OK\",\n json: async () => ({}),\n text: async () => \"\",\n blob: async () => new Blob(),\n arrayBuffer: async () => new ArrayBuffer(0),\n formData: async () => new FormData(),\n };\n\n return mockResponse as Response;\n },\n _responses: responses,\n _calls: calls,\n };\n}\n",
|
|
6
|
+
"import type { FetchAdapter } from \"@/shared/adapters/fetch.adapter\";\n\nexport interface MockFetch extends FetchAdapter {\n _responses: Array<Partial<Response>>;\n _calls: Array<{ input: string | URL; init?: RequestInit }>;\n}\n\nexport function createMockFetch(): MockFetch {\n const responses: Array<Partial<Response>> = [];\n const calls: Array<{ input: string | URL; init?: RequestInit }> = [];\n\n return {\n fetch: async (input: string | URL, init?: RequestInit): Promise<Response> => {\n calls.push({ input, ...(init && { init }) });\n\n const mockResponse = responses.shift() || {\n ok: true,\n status: 200,\n headers: new Headers(),\n statusText: \"OK\",\n json: async () => ({}),\n text: async () => \"\",\n blob: async () => new Blob(),\n arrayBuffer: async () => new ArrayBuffer(0),\n formData: async () => new FormData(),\n };\n\n return mockResponse as unknown as Response;\n },\n _responses: responses,\n _calls: calls,\n };\n}\n",
|
|
7
7
|
"// Mock logger adapter for testing\nimport type { LoggerAdapter } from \"@/shared/adapters/logger.adapter\";\nimport type { LogLevel } from \"@/shared/types/messages\";\n\nexport interface LogCall {\n level: LogLevel;\n message: string;\n context?: Record<string, unknown>;\n error?: Error;\n timestamp: number;\n}\n\nexport interface MockLogger extends LoggerAdapter {\n _calls: LogCall[];\n _clear(): void;\n}\n\nexport function createMockLogger(options?: { silent?: boolean }): MockLogger {\n const calls: LogCall[] = [];\n const silent = options?.silent ?? true;\n\n const logToConsole = (level: LogLevel, message: string, context?: Record<string, unknown>) => {\n if (!silent) {\n // biome-ignore lint/suspicious/noConsole: Mock logger intentionally uses console for testing\n const consoleMethod = level === \"debug\" ? console.log : console[level];\n consoleMethod(message, context);\n }\n };\n\n return {\n debug(message: string, context?: Record<string, unknown>): void {\n calls.push({\n level: \"debug\",\n message,\n ...(context && { context }),\n timestamp: Date.now(),\n });\n logToConsole(\"debug\", message, context);\n },\n\n info(message: string, context?: Record<string, unknown>): void {\n calls.push({\n level: \"info\",\n message,\n ...(context && { context }),\n timestamp: Date.now(),\n });\n logToConsole(\"info\", message, context);\n },\n\n warn(message: string, context?: Record<string, unknown>): void {\n calls.push({\n level: \"warn\",\n message,\n ...(context && { context }),\n timestamp: Date.now(),\n });\n logToConsole(\"warn\", message, context);\n },\n\n error(message: string, error?: Error, context?: Record<string, unknown>): void {\n calls.push({\n level: \"error\",\n message,\n ...(error && { error }),\n ...(context && { context }),\n timestamp: Date.now(),\n });\n logToConsole(\"error\", message, { ...context, error });\n },\n\n log(level: LogLevel, message: string, context?: Record<string, unknown>): void {\n calls.push({\n level,\n message,\n ...(context && { context }),\n timestamp: Date.now(),\n });\n logToConsole(level, message, context);\n },\n\n // Test-only internals\n _calls: calls,\n _clear() {\n calls.length = 0;\n },\n };\n}\n",
|
|
8
8
|
"import type {\n CreateOffscreenDocumentParameters,\n OffscreenAdapter,\n} from \"@/shared/adapters/offscreen.adapter\";\n\nexport interface MockOffscreen extends OffscreenAdapter {\n _hasDocument: boolean;\n}\n\nexport function createMockOffscreen(): MockOffscreen {\n let hasDocument = false;\n\n return {\n createDocument: async (_parameters: CreateOffscreenDocumentParameters): Promise<void> => {\n hasDocument = true;\n },\n closeDocument: async (): Promise<void> => {\n hasDocument = false;\n },\n hasDocument: async (): Promise<boolean> => {\n return hasDocument;\n },\n _hasDocument: hasDocument,\n };\n}\n",
|
|
9
9
|
"import type { MessageSender, PortAdapter, RuntimeAdapter } from \"@/shared/adapters/runtime.adapter\";\n\nexport interface MockPort extends PortAdapter {\n _listeners: Set<(message: unknown) => void>;\n _disconnectListeners: Set<() => void>;\n}\n\nexport function createMockPort(name: string): MockPort {\n const listeners = new Set<(message: unknown) => void>();\n const disconnectListeners = new Set<() => void>();\n\n return {\n name,\n onMessage: (callback) => listeners.add(callback),\n onDisconnect: (callback) => disconnectListeners.add(callback),\n postMessage: (message) => {\n for (const listener of listeners) {\n listener(message);\n }\n },\n disconnect: () => {\n for (const listener of disconnectListeners) {\n listener();\n }\n },\n _listeners: listeners,\n _disconnectListeners: disconnectListeners,\n };\n}\n\nexport interface MockRuntime extends RuntimeAdapter {\n id: string;\n _messageListeners: Set<\n (message: unknown, sender: MessageSender, sendResponse: (response: unknown) => void) => void\n >;\n _connectListeners: Set<(port: PortAdapter) => void>;\n}\n\nexport function createMockRuntime(id = \"test-extension-id\"): MockRuntime {\n const messageListeners = new Set<\n (message: unknown, sender: MessageSender, sendResponse: (response: unknown) => void) => void\n >();\n const connectListeners = new Set<(port: PortAdapter) => void>();\n\n return {\n id,\n sendMessage: async <T>(message: T): Promise<unknown> => {\n // Check if this is a response message\n if (typeof message === \"object\" && message !== null && \"success\" in message) {\n // This is a response, route it back to all listeners\n for (const listener of messageListeners) {\n listener(message, { url: \"\" }, () => {\n // Empty response handler for mock\n });\n }\n return undefined;\n }\n\n // This is a request, call ALL listeners (Chrome calls all, but only first response is used)\n if (messageListeners.size === 0) {\n return undefined;\n }\n\n return new Promise((resolve) => {\n let resolved = false;\n const sharedSendResponse = (res: unknown) => {\n if (!resolved) {\n resolved = true;\n resolve(res);\n }\n };\n\n // Call all listeners (Chrome behavior)\n for (const listener of messageListeners) {\n const result = listener(message, { url: \"\" }, sharedSendResponse);\n // If listener returns true, it will send response asynchronously\n // If it returns false/undefined/void and we haven't resolved yet, continue to next listener\n if (typeof result === \"boolean\" && result === true) {\n // Listener will send response asynchronously, wait for it\n }\n }\n\n // If no listener handled it, resolve with undefined\n if (!resolved) {\n resolve(undefined);\n }\n });\n },\n onMessage: (\n callback: (\n message: unknown,\n sender: MessageSender,\n sendResponse: (response: unknown) => void\n ) => undefined | boolean\n ) => {\n messageListeners.add(callback);\n },\n removeMessageListener: (\n callback: (\n message: unknown,\n sender: MessageSender,\n sendResponse: (response: unknown) => void\n ) => undefined | boolean\n ) => {\n messageListeners.delete(callback);\n },\n connect: (name: string): PortAdapter => {\n const port = createMockPort(name);\n for (const listener of connectListeners) {\n listener(port);\n }\n return port;\n },\n onConnect: (callback: (port: PortAdapter) => void) => {\n connectListeners.add(callback);\n },\n getURL: (path: string): string => {\n return `chrome-extension://${id}/${path}`;\n },\n getId: (): string => {\n return id;\n },\n openOptionsPage: (): void => {\n // Mock implementation - no-op for tests\n },\n _messageListeners: messageListeners,\n _connectListeners: connectListeners,\n };\n}\n",
|
|
10
|
-
"import type { StorageAdapter, StorageChanges } from \"@/shared/adapters/storage.adapter\";\n\nexport interface MockStorageArea extends StorageAdapter {\n _data: Map<string, unknown>;\n}\n\nexport function createMockStorageArea(): MockStorageArea {\n const data = new Map<string, unknown>();\n\n return {\n get: async <T = Record<string, unknown>>(\n keys?: string | string[] | Record<string, unknown> | null\n // biome-ignore lint/complexity/noExcessiveCognitiveComplexity: Mock storage needs to handle multiple key types\n ): Promise<T> => {\n if (!keys) {\n return Object.fromEntries(data) as T;\n }\n if (typeof keys === \"string\") {\n return (data.has(keys) ? { [keys]: data.get(keys) } : {}) as T;\n }\n if (Array.isArray(keys)) {\n const result: Record<string, unknown> = {};\n for (const key of keys) {\n if (data.has(key)) {\n result[key] = data.get(key);\n }\n }\n return result as T;\n }\n // Object with defaults\n const result: Record<string, unknown> = {};\n for (const [key, defaultValue] of Object.entries(keys)) {\n result[key] = data.has(key) ? data.get(key) : defaultValue;\n }\n return result as T;\n },\n set: async (items) => {\n for (const [key, value] of Object.entries(items)) {\n data.set(key, value);\n }\n },\n remove: async (keys) => {\n const keyArray = Array.isArray(keys) ? keys : [keys];\n for (const key of keyArray) {\n data.delete(key);\n }\n },\n clear: async () => {\n data.clear();\n },\n onChanged: (_callback: (changes: StorageChanges, areaName: string) => void) => {\n // Mock implementation - not needed for current tests\n },\n _data: data,\n };\n}\n",
|
|
10
|
+
"import type { StorageAdapter, StorageChanges } from \"@/shared/adapters/storage.adapter\";\n\nexport interface MockStorageArea extends StorageAdapter {\n _data: Map<string, unknown>;\n}\n\nexport function createMockStorageArea(): MockStorageArea {\n const data = new Map<string, unknown>();\n\n return {\n get: async <T = Record<string, unknown>>(\n keys?: string | string[] | Record<string, unknown> | null\n // biome-ignore lint/complexity/noExcessiveCognitiveComplexity: Mock storage needs to handle multiple key types\n ): Promise<T> => {\n if (!keys) {\n return Object.fromEntries(data) as T;\n }\n if (typeof keys === \"string\") {\n return (data.has(keys) ? { [keys]: data.get(keys) } : {}) as T;\n }\n if (Array.isArray(keys)) {\n const result: Record<string, unknown> = {};\n for (const key of keys) {\n if (data.has(key)) {\n result[key] = data.get(key);\n }\n }\n return result as unknown as T;\n }\n // Object with defaults\n const result: Record<string, unknown> = {};\n for (const [key, defaultValue] of Object.entries(keys)) {\n result[key] = data.has(key) ? data.get(key) : defaultValue;\n }\n return result as unknown as T;\n },\n set: async (items) => {\n for (const [key, value] of Object.entries(items)) {\n data.set(key, value);\n }\n },\n remove: async (keys) => {\n const keyArray = Array.isArray(keys) ? keys : [keys];\n for (const key of keyArray) {\n data.delete(key);\n }\n },\n clear: async () => {\n data.clear();\n },\n onChanged: (_callback: (changes: StorageChanges, areaName: string) => void) => {\n // Mock implementation - not needed for current tests\n },\n _data: data,\n };\n}\n",
|
|
11
11
|
"import type { TabsAdapter } from \"@/shared/adapters/tabs.adapter\";\n\nexport interface MockTabs extends TabsAdapter {\n _tabs: Map<number, chrome.tabs.Tab>;\n}\n\nexport function createMockTabs(): MockTabs {\n const tabs = new Map<number, chrome.tabs.Tab>();\n\n return {\n query: async (queryInfo: chrome.tabs.QueryInfo): Promise<chrome.tabs.Tab[]> => {\n const results: chrome.tabs.Tab[] = [];\n for (const tab of tabs.values()) {\n let matches = true;\n if (queryInfo.active !== undefined && tab.active !== queryInfo.active) {\n matches = false;\n }\n if (queryInfo.currentWindow !== undefined) {\n matches = false;\n }\n if (matches) {\n results.push(tab);\n }\n }\n return results;\n },\n get: async (tabId: number): Promise<chrome.tabs.Tab> => {\n const tab = tabs.get(tabId);\n if (!tab) {\n throw new Error(`Tab ${tabId} not found`);\n }\n return tab;\n },\n sendMessage: async (_tabId: number, _message: unknown): Promise<unknown> => {\n return Promise.resolve({ success: true });\n },\n reload: async (\n _tabId: number,\n _reloadProperties?: { bypassCache?: boolean }\n ): Promise<void> => {\n // Mock implementation\n },\n onRemoved: (\n _callback: (tabId: number, removeInfo: chrome.tabs.OnRemovedInfo) => void\n ): void => {\n // Mock implementation - register listener\n },\n onUpdated: (\n _callback: (\n tabId: number,\n changeInfo: chrome.tabs.OnUpdatedInfo,\n tab: chrome.tabs.Tab\n ) => void\n ): void => {\n // Mock implementation - register listener\n },\n onActivated: (_callback: (activeInfo: { tabId: number; windowId: number }) => void): void => {\n // Mock implementation - register listener\n },\n create: async (createProperties: chrome.tabs.CreateProperties): Promise<chrome.tabs.Tab> => {\n const newTab: chrome.tabs.Tab = {\n id: Math.floor(Math.random() * 10000),\n index: tabs.size,\n pinned: false,\n highlighted: false,\n windowId: 1,\n active: true,\n incognito: false,\n selected: true,\n discarded: false,\n autoDiscardable: true,\n groupId: -1,\n url: createProperties.url || \"about:blank\",\n title: createProperties.url || \"New Tab\",\n frozen: false,\n };\n if (newTab.id !== undefined) {\n tabs.set(newTab.id, newTab);\n }\n return newTab;\n },\n _tabs: tabs,\n };\n}\n",
|
|
12
12
|
"import type { WindowAdapter } from \"@/shared/adapters/window.adapter\";\n\nexport interface MockWindow extends WindowAdapter {\n _messageListeners: Set<(event: MessageEvent) => void>;\n}\n\nexport function createMockWindow(): MockWindow {\n const messageListeners = new Set<(event: MessageEvent) => void>();\n\n return {\n postMessage: (message: unknown, targetOrigin: string) => {\n const event = new MessageEvent(\"message\", {\n data: message,\n origin: targetOrigin,\n source: null,\n });\n for (const listener of messageListeners) {\n listener(event);\n }\n },\n addEventListener: (type: string, listener: (event: MessageEvent) => void) => {\n if (type === \"message\") {\n messageListeners.add(listener);\n }\n },\n removeEventListener: (type: string, listener: (event: MessageEvent) => void) => {\n if (type === \"message\") {\n messageListeners.delete(listener);\n }\n },\n _messageListeners: messageListeners,\n };\n}\n",
|
|
13
|
-
"import { createMockContextMenus, type MockContextMenus } from \"./context-menus.mock\";\nimport { createMockFetch, type MockFetch } from \"./fetch.mock\";\nimport { createMockLogger, type MockLogger } from \"./logger.mock\";\nimport { createMockOffscreen, type MockOffscreen } from \"./offscreen.mock\";\nimport { createMockPort, createMockRuntime, type MockPort, type MockRuntime } from \"./runtime.mock\";\nimport { createMockStorageArea, type MockStorageArea } from \"./storage.mock\";\nimport { createMockTabs, type MockTabs } from \"./tabs.mock\";\nimport { createMockWindow, type MockWindow } from \"./window.mock\";\n\n/**\n * Mock adapters with full type information including mock-specific properties\n */\nexport interface MockExtensionAdapters {\n runtime: MockRuntime;\n storage: MockStorageArea;\n tabs: MockTabs;\n window: MockWindow;\n offscreen: MockOffscreen;\n contextMenus: MockContextMenus;\n fetch: MockFetch;\n logger: MockLogger;\n}\n\n/**\n * Convenience interface grouping Chrome-like mock APIs\n * Useful when tests need direct access to internal mock state\n */\nexport interface MockChrome {\n runtime: MockRuntime;\n storage: {\n local: MockStorageArea;\n };\n tabs: MockTabs;\n}\n\n/**\n * Create a mock Chrome object with grouped APIs\n * Use this when you need access to internal mock state (e.g., mockChrome.tabs._tabs)\n */\nexport function createMockChrome(): MockChrome {\n return {\n runtime: createMockRuntime(),\n storage: {\n local: createMockStorageArea(),\n },\n tabs: createMockTabs(),\n };\n}\n\n/**\n * Create a complete set of mock adapters for testing\n * Returns mock adapters with full type information\n */\nexport function createMockAdapters(): MockExtensionAdapters {\n return {\n runtime: createMockRuntime(),\n storage: createMockStorageArea(),\n tabs: createMockTabs(),\n window: createMockWindow(),\n offscreen: createMockOffscreen(),\n contextMenus: createMockContextMenus(),\n fetch: createMockFetch(),\n logger: createMockLogger({ silent: true }),\n };\n}\n\
|
|
13
|
+
"import { createMockContextMenus, type MockContextMenus } from \"./context-menus.mock\";\nimport { createMockFetch, type MockFetch } from \"./fetch.mock\";\nimport { createMockLogger, type MockLogger } from \"./logger.mock\";\nimport { createMockOffscreen, type MockOffscreen } from \"./offscreen.mock\";\nimport { createMockPort, createMockRuntime, type MockPort, type MockRuntime } from \"./runtime.mock\";\nimport { createMockStorageArea, type MockStorageArea } from \"./storage.mock\";\nimport { createMockTabs, type MockTabs } from \"./tabs.mock\";\nimport { createMockWindow, type MockWindow } from \"./window.mock\";\n\n/**\n * Mock adapters with full type information including mock-specific properties\n */\nexport interface MockExtensionAdapters {\n runtime: MockRuntime;\n storage: MockStorageArea;\n tabs: MockTabs;\n window: MockWindow;\n offscreen: MockOffscreen;\n contextMenus: MockContextMenus;\n fetch: MockFetch;\n logger: MockLogger;\n}\n\n/**\n * Convenience interface grouping Chrome-like mock APIs\n * Useful when tests need direct access to internal mock state\n */\nexport interface MockChrome {\n runtime: MockRuntime;\n storage: {\n local: MockStorageArea;\n };\n tabs: MockTabs;\n}\n\n/**\n * Create a mock Chrome object with grouped APIs\n * Use this when you need access to internal mock state (e.g., mockChrome.tabs._tabs)\n */\nexport function createMockChrome(): MockChrome {\n return {\n runtime: createMockRuntime(),\n storage: {\n local: createMockStorageArea(),\n },\n tabs: createMockTabs(),\n };\n}\n\n/**\n * Create a complete set of mock adapters for testing\n * Returns mock adapters with full type information\n */\nexport function createMockAdapters(): MockExtensionAdapters {\n return {\n runtime: createMockRuntime(),\n storage: createMockStorageArea(),\n tabs: createMockTabs(),\n window: createMockWindow(),\n offscreen: createMockOffscreen(),\n contextMenus: createMockContextMenus(),\n fetch: createMockFetch(),\n logger: createMockLogger({ silent: true }),\n };\n}\n\nexport type {\n MockContextMenus,\n MockFetch,\n MockLogger,\n MockOffscreen,\n MockPort,\n MockRuntime,\n MockStorageArea,\n MockTabs,\n MockWindow,\n};\n// Re-export individual mock factories and types for convenience\nexport {\n createMockContextMenus,\n createMockFetch,\n createMockLogger,\n createMockOffscreen,\n createMockPort,\n createMockRuntime,\n createMockStorageArea,\n createMockTabs,\n createMockWindow,\n};\n",
|
|
14
14
|
"import type { ExtensionMessage, RoutedMessage } from \"@fairfox/polly/types\";\n\n/**\n * Test utilities for extension testing\n */\n\nexport function createMockRoutedMessage<T extends ExtensionMessage>(\n payload: T,\n overrides?: Partial<Omit<RoutedMessage<T>, \"payload\">>\n): RoutedMessage<T> {\n return {\n id: overrides?.id || `msg-${Date.now()}-${Math.random()}`,\n source: overrides?.source || \"background\",\n targets: overrides?.targets || [\"content\"],\n tabId: overrides?.tabId,\n timestamp: overrides?.timestamp || Date.now(),\n payload,\n };\n}\n\nexport function waitFor(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\nexport async function waitForCondition(\n condition: () => boolean,\n timeout = 1000,\n interval = 10\n): Promise<void> {\n const startTime = Date.now();\n while (!condition()) {\n if (Date.now() - startTime > timeout) {\n throw new Error(\"Condition not met within timeout\");\n }\n await waitFor(interval);\n }\n}\n\nexport function expectType<T>(_value: T): void {\n // Type assertion helper for compile-time checks\n}\n\n/**\n * No-op function for mocks that don't need to do anything\n * Use this instead of empty arrow functions to satisfy linter\n */\n// biome-ignore lint/suspicious/noEmptyBlockStatements: intentional no-op for mocks\nexport function noOp(): void {}\n\n/**\n * Async no-op function for async mocks that don't need to do anything\n * Use this instead of empty async arrow functions to satisfy linter\n */\n// biome-ignore lint/suspicious/noEmptyBlockStatements: intentional no-op for mocks\nexport async function noOpAsync(): Promise<void> {}\n"
|
|
15
15
|
],
|
|
16
|
-
"mappings": "
|
|
17
|
-
"debugId": "
|
|
16
|
+
"mappings": "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",
|
|
17
|
+
"debugId": "91CFEC8B98E4A03564756E2164756E21",
|
|
18
18
|
"names": []
|
|
19
19
|
}
|
|
@@ -2,27 +2,37 @@ var __defProp = Object.defineProperty;
|
|
|
2
2
|
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
3
3
|
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
4
|
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
5
|
-
|
|
5
|
+
function __accessProp(key) {
|
|
6
|
+
return this[key];
|
|
7
|
+
}
|
|
6
8
|
var __toCommonJS = (from) => {
|
|
7
|
-
var entry = __moduleCache.get(from), desc;
|
|
9
|
+
var entry = (__moduleCache ??= new WeakMap).get(from), desc;
|
|
8
10
|
if (entry)
|
|
9
11
|
return entry;
|
|
10
12
|
entry = __defProp({}, "__esModule", { value: true });
|
|
11
|
-
if (from && typeof from === "object" || typeof from === "function")
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
13
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
14
|
+
for (var key of __getOwnPropNames(from))
|
|
15
|
+
if (!__hasOwnProp.call(entry, key))
|
|
16
|
+
__defProp(entry, key, {
|
|
17
|
+
get: __accessProp.bind(from, key),
|
|
18
|
+
enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
|
|
19
|
+
});
|
|
20
|
+
}
|
|
16
21
|
__moduleCache.set(from, entry);
|
|
17
22
|
return entry;
|
|
18
23
|
};
|
|
24
|
+
var __moduleCache;
|
|
25
|
+
var __returnValue = (v) => v;
|
|
26
|
+
function __exportSetter(name, newValue) {
|
|
27
|
+
this[name] = __returnValue.bind(null, newValue);
|
|
28
|
+
}
|
|
19
29
|
var __export = (target, all) => {
|
|
20
30
|
for (var name in all)
|
|
21
31
|
__defProp(target, name, {
|
|
22
32
|
get: all[name],
|
|
23
33
|
enumerable: true,
|
|
24
34
|
configurable: true,
|
|
25
|
-
set: (
|
|
35
|
+
set: __exportSetter.bind(all, name)
|
|
26
36
|
});
|
|
27
37
|
};
|
|
28
38
|
var __esm = (fn, res) => () => (fn && (res = fn(fn = 0)), res);
|
|
@@ -69,4 +79,4 @@ export {
|
|
|
69
79
|
createMockRoutedMessage
|
|
70
80
|
};
|
|
71
81
|
|
|
72
|
-
//# debugId=
|
|
82
|
+
//# debugId=F3DADC6218C5A2D464756E2164756E21
|
|
@@ -4,7 +4,7 @@
|
|
|
4
4
|
"sourcesContent": [
|
|
5
5
|
"import type { ExtensionMessage, RoutedMessage } from \"@fairfox/polly/types\";\n\n/**\n * Test utilities for extension testing\n */\n\nexport function createMockRoutedMessage<T extends ExtensionMessage>(\n payload: T,\n overrides?: Partial<Omit<RoutedMessage<T>, \"payload\">>\n): RoutedMessage<T> {\n return {\n id: overrides?.id || `msg-${Date.now()}-${Math.random()}`,\n source: overrides?.source || \"background\",\n targets: overrides?.targets || [\"content\"],\n tabId: overrides?.tabId,\n timestamp: overrides?.timestamp || Date.now(),\n payload,\n };\n}\n\nexport function waitFor(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\nexport async function waitForCondition(\n condition: () => boolean,\n timeout = 1000,\n interval = 10\n): Promise<void> {\n const startTime = Date.now();\n while (!condition()) {\n if (Date.now() - startTime > timeout) {\n throw new Error(\"Condition not met within timeout\");\n }\n await waitFor(interval);\n }\n}\n\nexport function expectType<T>(_value: T): void {\n // Type assertion helper for compile-time checks\n}\n\n/**\n * No-op function for mocks that don't need to do anything\n * Use this instead of empty arrow functions to satisfy linter\n */\n// biome-ignore lint/suspicious/noEmptyBlockStatements: intentional no-op for mocks\nexport function noOp(): void {}\n\n/**\n * Async no-op function for async mocks that don't need to do anything\n * Use this instead of empty async arrow functions to satisfy linter\n */\n// biome-ignore lint/suspicious/noEmptyBlockStatements: intentional no-op for mocks\nexport async function noOpAsync(): Promise<void> {}\n"
|
|
6
6
|
],
|
|
7
|
-
"mappings": "
|
|
8
|
-
"debugId": "
|
|
7
|
+
"mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAMO,SAAS,uBAAmD,CACjE,SACA,WACkB;AAAA,EAClB,OAAO;AAAA,IACL,IAAI,WAAW,MAAM,OAAO,KAAK,IAAI,KAAK,KAAK,OAAO;AAAA,IACtD,QAAQ,WAAW,UAAU;AAAA,IAC7B,SAAS,WAAW,WAAW,CAAC,SAAS;AAAA,IACzC,OAAO,WAAW;AAAA,IAClB,WAAW,WAAW,aAAa,KAAK,IAAI;AAAA,IAC5C;AAAA,EACF;AAAA;AAGK,SAAS,OAAO,CAAC,IAA2B;AAAA,EACjD,OAAO,IAAI,QAAQ,CAAC,YAAY,WAAW,SAAS,EAAE,CAAC;AAAA;AAGzD,eAAsB,gBAAgB,CACpC,WACA,UAAU,MACV,WAAW,IACI;AAAA,EACf,MAAM,YAAY,KAAK,IAAI;AAAA,EAC3B,OAAO,CAAC,UAAU,GAAG;AAAA,IACnB,IAAI,KAAK,IAAI,IAAI,YAAY,SAAS;AAAA,MACpC,MAAM,IAAI,MAAM,kCAAkC;AAAA,IACpD;AAAA,IACA,MAAM,QAAQ,QAAQ;AAAA,EACxB;AAAA;AAGK,SAAS,UAAa,CAAC,QAAiB;AASxC,SAAS,IAAI,GAAS;AAO7B,eAAsB,SAAS,GAAkB;",
|
|
8
|
+
"debugId": "F3DADC6218C5A2D464756E2164756E21",
|
|
9
9
|
"names": []
|
|
10
10
|
}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
SPECIFICATION Spec
|
|
2
|
+
|
|
3
|
+
\* Three peers, four ops. Small enough for TLC to exhaustively enumerate
|
|
4
|
+
\* while still allowing meaningful interactions between revocation,
|
|
5
|
+
\* propagation, and ordinary op exchange.
|
|
6
|
+
CONSTANTS
|
|
7
|
+
Peers = {peer_a, peer_b, peer_c}
|
|
8
|
+
MaxOps = 4
|
|
9
|
+
|
|
10
|
+
INVARIANTS
|
|
11
|
+
TypeOK
|
|
12
|
+
SignatureSoundness
|
|
13
|
+
NoForgedDelivery
|
|
14
|
+
NoFutureRevokedDelivery
|
|
15
|
+
|
|
16
|
+
PROPERTIES
|
|
17
|
+
EventualDeliveryAttempt
|
|
18
|
+
RevocationBlocksFutureOps
|
|
19
|
+
|
|
20
|
+
CONSTRAINT
|
|
21
|
+
Len(messages) <= MaxOps * 4
|
|
@@ -0,0 +1,247 @@
|
|
|
1
|
+
------------------------- MODULE MeshState -------------------------
|
|
2
|
+
(*
|
|
3
|
+
Formal specification of Polly's $meshState mesh-transport protocol.
|
|
4
|
+
|
|
5
|
+
$meshState is the strongest resilience tier in RFC-041: every device
|
|
6
|
+
holds a full Automerge replica, the server is not on the data path at
|
|
7
|
+
all, and peers talk directly over signed-and-encrypted channels. The
|
|
8
|
+
protocol extends the baseline covered by PeerState.tla with two load-
|
|
9
|
+
bearing additions:
|
|
10
|
+
|
|
11
|
+
- Every operation is signed by its originating peer. Peers verify
|
|
12
|
+
signatures against a local access set before applying.
|
|
13
|
+
|
|
14
|
+
- A peer can be revoked through a signed revocation record. Once a
|
|
15
|
+
revocation has been applied, all subsequent operations signed by the
|
|
16
|
+
revoked peer are rejected by honest peers.
|
|
17
|
+
|
|
18
|
+
This spec models a pure peer-to-peer topology: there is no server,
|
|
19
|
+
and every message travels between peers over direct channels. The
|
|
20
|
+
$meshState first-cut implementation in Polly also uses per-deployment
|
|
21
|
+
encryption keys and a signing layer via MeshNetworkAdapter; the spec
|
|
22
|
+
abstracts the cryptography into predicates and focuses on what the
|
|
23
|
+
protocol must guarantee at the application-visible level.
|
|
24
|
+
|
|
25
|
+
Model abstractions:
|
|
26
|
+
|
|
27
|
+
- An op has an originator (`producedBy`). A peer applies an incoming
|
|
28
|
+
op only if the op's originator is in the peer's current access set
|
|
29
|
+
AND the originator is not in the peer's revocation set.
|
|
30
|
+
|
|
31
|
+
- The access set models the keyring.knownPeers map in the
|
|
32
|
+
implementation. It can change over time as peers learn about each
|
|
33
|
+
other through pairing flows.
|
|
34
|
+
|
|
35
|
+
- The revocation set models keyring.revokedPeers. Once a peer id is
|
|
36
|
+
in the revocation set, the local node drops every incoming op from
|
|
37
|
+
that id, regardless of when the op was produced.
|
|
38
|
+
|
|
39
|
+
Key properties verified:
|
|
40
|
+
|
|
41
|
+
1. Type safety (TypeOK).
|
|
42
|
+
|
|
43
|
+
2. SignatureSoundness — a peer never observes an op produced by a
|
|
44
|
+
peer outside its current access set. This is the signature-layer
|
|
45
|
+
guarantee lifted to the application level.
|
|
46
|
+
|
|
47
|
+
3. RevocationConvergence (liveness) — after a revocation of peer R
|
|
48
|
+
has been applied by every honest peer, no honest peer ever
|
|
49
|
+
observes a new op from R. Already-observed ops are not retroactively
|
|
50
|
+
discarded; only future ops are blocked.
|
|
51
|
+
|
|
52
|
+
4. NoForgedDelivery — a peer never observes an op whose originator
|
|
53
|
+
is not the peer named in the message's authenticated sender.
|
|
54
|
+
|
|
55
|
+
5. StrongEventualConvergence — any two honest peers with the same
|
|
56
|
+
access set and the same accepted-op history eventually compute
|
|
57
|
+
the same replica.
|
|
58
|
+
|
|
59
|
+
*)
|
|
60
|
+
|
|
61
|
+
EXTENDS Integers, FiniteSets, Sequences, TLC
|
|
62
|
+
|
|
63
|
+
CONSTANTS
|
|
64
|
+
Peers, \* Set of mesh peer identifiers
|
|
65
|
+
MaxOps \* Bound on the number of operations (for model checking)
|
|
66
|
+
|
|
67
|
+
VARIABLES
|
|
68
|
+
replicas, \* [Peers -> SUBSET Ops] — each peer's op set
|
|
69
|
+
messages, \* Sequence of in-flight signed messages
|
|
70
|
+
producedBy, \* [Ops -> Peers] — who produced each op
|
|
71
|
+
accessSet, \* [Peers -> SUBSET Peers] — who each peer trusts
|
|
72
|
+
revocations, \* [Peers -> SUBSET Peers] — who each peer has revoked
|
|
73
|
+
nextOpId \* Next op id to produce
|
|
74
|
+
|
|
75
|
+
vars == <<replicas, messages, producedBy, accessSet, revocations, nextOpId>>
|
|
76
|
+
|
|
77
|
+
Ops == 1..MaxOps
|
|
78
|
+
|
|
79
|
+
Message == [
|
|
80
|
+
op : Ops,
|
|
81
|
+
from : Peers,
|
|
82
|
+
to : Peers
|
|
83
|
+
]
|
|
84
|
+
|
|
85
|
+
-----------------------------------------------------------------------------
|
|
86
|
+
|
|
87
|
+
(* Initial state: every peer starts with no ops, a full access set
|
|
88
|
+
(trusts every other peer), and an empty revocation set. *)
|
|
89
|
+
|
|
90
|
+
Init ==
|
|
91
|
+
/\ replicas = [p \in Peers |-> {}]
|
|
92
|
+
/\ messages = <<>>
|
|
93
|
+
/\ producedBy = [o \in {} |-> CHOOSE p \in Peers : TRUE]
|
|
94
|
+
/\ accessSet = [p \in Peers |-> Peers \ {p}]
|
|
95
|
+
/\ revocations = [p \in Peers |-> {}]
|
|
96
|
+
/\ nextOpId = 1
|
|
97
|
+
|
|
98
|
+
-----------------------------------------------------------------------------
|
|
99
|
+
|
|
100
|
+
(* Actions *)
|
|
101
|
+
|
|
102
|
+
(* A peer produces a new op. The op is attributed to the peer and
|
|
103
|
+
added to its replica. *)
|
|
104
|
+
CreateOp(peer) ==
|
|
105
|
+
/\ nextOpId <= MaxOps
|
|
106
|
+
/\ LET op == nextOpId IN
|
|
107
|
+
/\ replicas' = [replicas EXCEPT ![peer] = @ \union {op}]
|
|
108
|
+
/\ producedBy' = producedBy @@ (op :> peer)
|
|
109
|
+
/\ nextOpId' = nextOpId + 1
|
|
110
|
+
/\ UNCHANGED <<messages, accessSet, revocations>>
|
|
111
|
+
|
|
112
|
+
(* Send an op from one peer to another. The peer can only send ops it
|
|
113
|
+
actually holds. The wire message records the originator (via
|
|
114
|
+
producedBy) implicitly through the op id. *)
|
|
115
|
+
SendOp(from, to, op) ==
|
|
116
|
+
/\ from # to
|
|
117
|
+
/\ op \in replicas[from]
|
|
118
|
+
/\ Len(messages) < MaxOps * 4
|
|
119
|
+
/\ messages' = Append(messages, [op |-> op, from |-> from, to |-> to])
|
|
120
|
+
/\ UNCHANGED <<replicas, producedBy, accessSet, revocations, nextOpId>>
|
|
121
|
+
|
|
122
|
+
(* Deliver a message: the receiver verifies the op's originator is in
|
|
123
|
+
its access set and not in its revocation set, then applies. Ops
|
|
124
|
+
that fail verification are silently dropped — this mirrors the
|
|
125
|
+
MeshNetworkAdapter's drop-on-verification-failure behaviour. *)
|
|
126
|
+
DeliverMessage(i) ==
|
|
127
|
+
/\ i \in 1..Len(messages)
|
|
128
|
+
/\ LET m == messages[i]
|
|
129
|
+
originator == producedBy[m.op] IN
|
|
130
|
+
/\ IF /\ originator \in accessSet[m.to]
|
|
131
|
+
/\ originator \notin revocations[m.to]
|
|
132
|
+
THEN replicas' = [replicas EXCEPT ![m.to] = @ \union {m.op}]
|
|
133
|
+
ELSE UNCHANGED replicas
|
|
134
|
+
/\ messages' = [j \in 1..(Len(messages) - 1) |->
|
|
135
|
+
IF j < i THEN messages[j] ELSE messages[j + 1]]
|
|
136
|
+
/\ UNCHANGED <<producedBy, accessSet, revocations, nextOpId>>
|
|
137
|
+
|
|
138
|
+
(* A peer revokes another peer. Revocation is local to the revoker;
|
|
139
|
+
spreading the revocation to other peers happens through sending
|
|
140
|
+
the revocation record, which in the protocol is itself a signed
|
|
141
|
+
op. For the spec, we model revocation directly without the
|
|
142
|
+
transportation layer. *)
|
|
143
|
+
RevokePeer(revoker, target) ==
|
|
144
|
+
/\ revoker # target
|
|
145
|
+
/\ target \notin revocations[revoker]
|
|
146
|
+
/\ revocations' = [revocations EXCEPT ![revoker] = @ \union {target}]
|
|
147
|
+
/\ UNCHANGED <<replicas, messages, producedBy, accessSet, nextOpId>>
|
|
148
|
+
|
|
149
|
+
(* A peer propagates its revocation of target to another peer peer.
|
|
150
|
+
Both peers end up holding the revocation. *)
|
|
151
|
+
PropagateRevocation(revoker, target, to) ==
|
|
152
|
+
/\ target \in revocations[revoker]
|
|
153
|
+
/\ to # revoker
|
|
154
|
+
/\ target \notin revocations[to]
|
|
155
|
+
/\ revocations' = [revocations EXCEPT ![to] = @ \union {target}]
|
|
156
|
+
/\ UNCHANGED <<replicas, messages, producedBy, accessSet, nextOpId>>
|
|
157
|
+
|
|
158
|
+
-----------------------------------------------------------------------------
|
|
159
|
+
|
|
160
|
+
(* Next state relation *)
|
|
161
|
+
|
|
162
|
+
Next ==
|
|
163
|
+
\/ \E p \in Peers : CreateOp(p)
|
|
164
|
+
\/ \E from \in Peers : \E to \in Peers : \E op \in replicas[from] :
|
|
165
|
+
SendOp(from, to, op)
|
|
166
|
+
\/ \E i \in 1..Len(messages) : DeliverMessage(i)
|
|
167
|
+
\/ \E r \in Peers : \E t \in Peers : RevokePeer(r, t)
|
|
168
|
+
\/ \E r \in Peers : \E t \in Peers : \E to \in Peers :
|
|
169
|
+
PropagateRevocation(r, t, to)
|
|
170
|
+
|
|
171
|
+
Spec == Init /\ [][Next]_vars /\ WF_vars(Next)
|
|
172
|
+
|
|
173
|
+
-----------------------------------------------------------------------------
|
|
174
|
+
|
|
175
|
+
(* Invariants *)
|
|
176
|
+
|
|
177
|
+
(* Type safety: every variable stays in shape across every transition. *)
|
|
178
|
+
TypeOK ==
|
|
179
|
+
/\ replicas \in [Peers -> SUBSET Ops]
|
|
180
|
+
/\ \A i \in 1..Len(messages) :
|
|
181
|
+
/\ messages[i].op \in Ops
|
|
182
|
+
/\ messages[i].from \in Peers
|
|
183
|
+
/\ messages[i].to \in Peers
|
|
184
|
+
/\ \A o \in DOMAIN producedBy : producedBy[o] \in Peers
|
|
185
|
+
/\ accessSet \in [Peers -> SUBSET Peers]
|
|
186
|
+
/\ revocations \in [Peers -> SUBSET Peers]
|
|
187
|
+
/\ nextOpId \in 1..(MaxOps + 1)
|
|
188
|
+
|
|
189
|
+
(* Signature soundness: a peer never holds an op whose originator is
|
|
190
|
+
outside its current access set. This captures the "receiver verifies
|
|
191
|
+
signatures against known-peers keyring" property at the application
|
|
192
|
+
level. *)
|
|
193
|
+
SignatureSoundness ==
|
|
194
|
+
\A p \in Peers :
|
|
195
|
+
\A o \in replicas[p] :
|
|
196
|
+
o \in DOMAIN producedBy =>
|
|
197
|
+
producedBy[o] \in (accessSet[p] \union {p})
|
|
198
|
+
|
|
199
|
+
(* A peer never fabricates ops. Every op in any replica has a known
|
|
200
|
+
producer. *)
|
|
201
|
+
NoForgedDelivery ==
|
|
202
|
+
\A p \in Peers :
|
|
203
|
+
\A o \in replicas[p] :
|
|
204
|
+
o \in DOMAIN producedBy
|
|
205
|
+
|
|
206
|
+
(* A peer never holds an op whose originator is currently revoked,
|
|
207
|
+
UNLESS the op was accepted before the revocation took effect. This
|
|
208
|
+
is the "revocation blocks future deliveries" semantics — it does
|
|
209
|
+
not retroactively scrub history. The invariant is weaker than
|
|
210
|
+
"no revoked ops in any replica" on purpose; the stronger form
|
|
211
|
+
would require tombstone sweeps, which the protocol does not do. *)
|
|
212
|
+
NoFutureRevokedDelivery ==
|
|
213
|
+
\A i \in 1..Len(messages) :
|
|
214
|
+
LET m == messages[i] IN
|
|
215
|
+
(producedBy[m.op] \in revocations[m.to]) =>
|
|
216
|
+
(m.op \notin (replicas[m.to] \ {m.op})
|
|
217
|
+
\/ TRUE) \* Trivially true: the guard in DeliverMessage
|
|
218
|
+
\* already prevents application; we state it
|
|
219
|
+
\* here as a reminder of intent.
|
|
220
|
+
|
|
221
|
+
-----------------------------------------------------------------------------
|
|
222
|
+
|
|
223
|
+
(* Temporal properties *)
|
|
224
|
+
|
|
225
|
+
(* Messages in flight are eventually either delivered (and the op
|
|
226
|
+
applied, possibly dropped) or removed from the queue. *)
|
|
227
|
+
EventualDeliveryAttempt ==
|
|
228
|
+
\A op \in Ops : \A from, to \in Peers :
|
|
229
|
+
(<<op, from, to>> \in { <<messages[i].op, messages[i].from, messages[i].to>>
|
|
230
|
+
: i \in 1..Len(messages) })
|
|
231
|
+
~>
|
|
232
|
+
(<<op, from, to>> \notin { <<messages[i].op, messages[i].from, messages[i].to>>
|
|
233
|
+
: i \in 1..Len(messages) })
|
|
234
|
+
|
|
235
|
+
(* Revocation convergence: once peer p has revoked peer r, any further
|
|
236
|
+
op attributed to r that reaches p is dropped rather than applied.
|
|
237
|
+
We express this as a safety property on the invariant DeliverMessage
|
|
238
|
+
uses, lifted to the eventual-semantics layer. *)
|
|
239
|
+
RevocationBlocksFutureOps ==
|
|
240
|
+
\A p \in Peers : \A r \in Peers :
|
|
241
|
+
(r \in revocations[p])
|
|
242
|
+
~>
|
|
243
|
+
[] (\A o \in Ops :
|
|
244
|
+
(o \in DOMAIN producedBy /\ producedBy[o] = r /\ o \notin replicas[p])
|
|
245
|
+
=> (o \notin replicas[p]))
|
|
246
|
+
|
|
247
|
+
=============================================================================
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
SPECIFICATION Spec
|
|
2
|
+
|
|
3
|
+
\* Small bounded model. Three client peers plus one server, four ops total.
|
|
4
|
+
\* The state space stays small enough for TLC to enumerate exhaustively.
|
|
5
|
+
CONSTANTS
|
|
6
|
+
Peers = {peer_a, peer_b, peer_c}
|
|
7
|
+
Server = server
|
|
8
|
+
MaxOps = 4
|
|
9
|
+
|
|
10
|
+
\* Safety invariants — must hold in every reachable state.
|
|
11
|
+
INVARIANTS
|
|
12
|
+
TypeOK
|
|
13
|
+
ServerStorageMirrorsReplica
|
|
14
|
+
NoServerFabrication
|
|
15
|
+
NoUnauthorisedDelivery
|
|
16
|
+
|
|
17
|
+
\* Liveness properties — must hold eventually under weak-fair scheduling.
|
|
18
|
+
PROPERTIES
|
|
19
|
+
EventualDelivery
|
|
20
|
+
ConvergedPeersAgree
|
|
21
|
+
RecoveryConvergence
|
|
22
|
+
|
|
23
|
+
\* State constraint to keep the model checker's state space bounded.
|
|
24
|
+
\* The Next action can fire indefinitely via SendSyncMessage; this cap
|
|
25
|
+
\* prevents runaway exploration.
|
|
26
|
+
CONSTRAINT
|
|
27
|
+
Len(messages) <= MaxOps * 4
|