@naeemo/capnp 0.8.1 → 0.9.1

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.
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cli-bench-bkR1vGK8.js","names":["isNode","EventEmitter","http","createHash","Duplex","EventEmitter","WebSocket","WebSocketServer"],"sources":["../src/debug/config.ts","../src/debug/index.ts","../src/rpc/stream.ts","../src/rpc/realtime.ts","../src/rpc/stream-manager.ts","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/lib/constants.js","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/lib/buffer-util.js","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/lib/limiter.js","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/lib/permessage-deflate.js","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/lib/validation.js","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/lib/receiver.js","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/lib/sender.js","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/lib/event-target.js","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/lib/extension.js","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/lib/websocket.js","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/lib/stream.js","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/lib/subprotocol.js","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/lib/websocket-server.js","../node_modules/.pnpm/ws@8.19.0/node_modules/ws/wrapper.mjs","../src/proxy/websocket-proxy.ts","../src/cli-bench.ts"],"sourcesContent":["/**\n * Global Debug Configuration for Cap'n Proto TypeScript\n *\n * Provides runtime debug controls and logging configuration.\n */\n\n/**\n * Debug configuration options\n */\nexport interface DebugOptions {\n /** Enable colored output in debug logs */\n colors?: boolean;\n /** Maximum bytes to display when logging binary data */\n maxBytes?: number;\n /** Filter by message type or peer (string or RegExp) */\n filter?: string | RegExp;\n}\n\n/**\n * Internal debug state\n */\ninterface DebugState {\n enabled: boolean;\n options: Required<DebugOptions>;\n}\n\n/**\n * Global debug state\n */\nconst debugState: DebugState = {\n enabled: false,\n options: {\n colors: true,\n maxBytes: 256,\n filter: '',\n },\n};\n\n/**\n * Check if running in Node.js environment\n */\nfunction isNode(): boolean {\n return (\n typeof process !== 'undefined' && process.versions != null && process.versions.node != null\n );\n}\n\n/**\n * Check CAPNP_DEBUG environment variable in Node.js\n */\nfunction checkEnvVar(): boolean {\n if (isNode()) {\n try {\n const envValue = process.env.CAPNP_DEBUG;\n return envValue === '1' || envValue === 'true';\n } catch {\n // Environment access failed, assume false\n return false;\n }\n }\n return false;\n}\n\n// Initialize from environment variable on module load\nif (checkEnvVar()) {\n debugState.enabled = true;\n}\n\n/**\n * Enable debug mode with optional configuration\n *\n * @param options - Debug configuration options\n * @example\n * ```typescript\n * enableDebug({ colors: true, maxBytes: 512 });\n * ```\n */\nexport function enableDebug(options?: DebugOptions): void {\n debugState.enabled = true;\n if (options) {\n debugState.options = {\n ...debugState.options,\n ...options,\n };\n }\n}\n\n/**\n * Disable debug mode\n *\n * @example\n * ```typescript\n * disableDebug();\n * ```\n */\nexport function disableDebug(): void {\n debugState.enabled = false;\n}\n\n/**\n * Check if debug mode is currently enabled\n *\n * @returns True if debug mode is enabled\n * @example\n * ```typescript\n * if (isDebugEnabled()) {\n * console.log('Debug is on');\n * }\n * ```\n */\nexport function isDebugEnabled(): boolean {\n return debugState.enabled;\n}\n\n/**\n * Get current debug options (internal use)\n * @internal\n */\nexport function getDebugOptions(): Readonly<Required<DebugOptions>> {\n return Object.freeze({ ...debugState.options });\n}\n\n/**\n * Check if a message/peer matches the current filter\n * @internal\n */\nexport function matchesFilter(messageType: string, peer?: string): boolean {\n const { filter } = debugState.options;\n\n if (!filter || filter === '') {\n return true;\n }\n\n const target = peer ? `${messageType}:${peer}` : messageType;\n\n if (filter instanceof RegExp) {\n return filter.test(target);\n }\n\n return target.includes(filter);\n}\n\n/**\n * Format bytes for debug output\n * @internal\n */\nexport function formatBytes(data: Uint8Array, maxBytes?: number): string {\n const limit = maxBytes ?? debugState.options.maxBytes;\n const slice = data.length > limit ? data.slice(0, limit) : data;\n const hex = Array.from(slice)\n .map((b) => b.toString(16).padStart(2, '0'))\n .join(' ');\n\n if (data.length > limit) {\n return `${hex}... (${data.length} bytes total)`;\n }\n return hex;\n}\n","/**\n * Debug/Trace module for Cap'n Proto RPC\n * Provides formatted logging of binary messages with hexdump-style output\n */\n\n/**\n * Configuration options for debug logging\n */\nexport interface DebugConfig {\n /** Whether debugging is enabled */\n enabled: boolean;\n /** Whether to use colors in output */\n colors: boolean;\n /** Maximum number of bytes to log per message */\n maxBytesToLog: number;\n}\n\n/**\n * Default configuration for debug logging\n */\nconst DEFAULT_CONFIG: DebugConfig = {\n enabled: false,\n colors: true,\n maxBytesToLog: 1024,\n};\n\n/**\n * ANSI color codes for Node.js output\n */\nconst ANSI_COLORS = {\n reset: '\\x1b[0m',\n bright: '\\x1b[1m',\n dim: '\\x1b[2m',\n red: '\\x1b[31m',\n green: '\\x1b[32m',\n yellow: '\\x1b[33m',\n blue: '\\x1b[34m',\n magenta: '\\x1b[35m',\n cyan: '\\x1b[36m',\n white: '\\x1b[37m',\n gray: '\\x1b[90m',\n};\n\n/**\n * CSS styles for browser console output\n */\nconst BROWSER_STYLES = {\n header: 'color: #6c757d; font-weight: bold;',\n send: 'color: #28a745; font-weight: bold;',\n recv: 'color: #007bff; font-weight: bold;',\n hex: 'color: #6c757d;',\n ascii: 'color: #495057;',\n arrow: 'color: #6c757d;',\n parsed: 'color: #17a2b8;',\n};\n\n/**\n * Check if running in Node.js environment\n */\nfunction isNode(): boolean {\n return (\n typeof process !== 'undefined' && process.versions != null && process.versions.node != null\n );\n}\n\n/**\n * Format a byte as two-digit hex string\n */\nfunction byteToHex(byte: number): string {\n return byte.toString(16).padStart(2, '0');\n}\n\n/**\n * Check if a byte is printable ASCII\n */\nfunction isPrintable(byte: number): boolean {\n return byte >= 32 && byte < 127;\n}\n\n/**\n * Format binary data as hex + ASCII (like hexdump -C)\n *\n * Output format:\n * 00000000: 00 00 00 00 02 00 00 00 │........│\n *\n * @param data - The binary data to format\n * @param maxBytes - Maximum number of bytes to format\n * @param useColors - Whether to include ANSI colors\n * @returns Array of formatted lines\n */\nexport function formatHexDump(data: Uint8Array, maxBytes = 1024, useColors = false): string[] {\n const lines: string[] = [];\n const bytesToFormat = Math.min(data.length, maxBytes);\n const bytesPerLine = 16;\n\n for (let offset = 0; offset < bytesToFormat; offset += bytesPerLine) {\n const chunk = data.slice(offset, Math.min(offset + bytesPerLine, bytesToFormat));\n\n // Address prefix\n let line = `${byteToHex((offset >> 24) & 0xff)}${byteToHex((offset >> 16) & 0xff)}${byteToHex((offset >> 8) & 0xff)}${byteToHex(offset & 0xff)}: `;\n\n // Hex bytes\n const hexParts: string[] = [];\n for (let i = 0; i < bytesPerLine; i++) {\n if (i < chunk.length) {\n hexParts.push(byteToHex(chunk[i]));\n } else {\n hexParts.push(' ');\n }\n // Add extra space after 8 bytes for readability\n if (i === 7) {\n hexParts.push('');\n }\n }\n line += hexParts.join(' ');\n\n // ASCII representation with delimiter\n let ascii = ' │';\n for (let i = 0; i < chunk.length; i++) {\n ascii += isPrintable(chunk[i]) ? String.fromCharCode(chunk[i]) : '.';\n }\n ascii += '│';\n\n line += ascii;\n\n // Add ANSI colors if enabled (dim the hex/ascii parts)\n if (useColors && isNode()) {\n // Find where hex starts (after address)\n const hexStart = line.indexOf(':') + 2;\n const delimiterIndex = line.indexOf('│');\n const hexPart = line.slice(hexStart, delimiterIndex);\n const asciiPart = line.slice(delimiterIndex);\n line =\n line.slice(0, hexStart) +\n ANSI_COLORS.gray +\n hexPart +\n ANSI_COLORS.reset +\n ANSI_COLORS.dim +\n asciiPart +\n ANSI_COLORS.reset;\n }\n\n lines.push(line);\n }\n\n // Add truncation indicator if data was truncated\n if (data.length > maxBytes) {\n const remaining = data.length - maxBytes;\n lines.push(`... (${remaining} more bytes)`);\n }\n\n return lines;\n}\n\n/**\n * Debug logger for Cap'n Proto RPC messages\n */\nexport class DebugLogger {\n private config: DebugConfig;\n\n /**\n * Create a new DebugLogger instance\n * @param config - Partial configuration to override defaults\n */\n constructor(config: Partial<DebugConfig> = {}) {\n this.config = { ...DEFAULT_CONFIG, ...config };\n }\n\n /**\n * Update the logger configuration\n * @param config - Partial configuration to merge\n */\n setConfig(config: Partial<DebugConfig>): void {\n this.config = { ...this.config, ...config };\n }\n\n /**\n * Get current configuration\n */\n getConfig(): DebugConfig {\n return { ...this.config };\n }\n\n /**\n * Check if debugging is enabled\n */\n isEnabled(): boolean {\n return this.config.enabled;\n }\n\n /**\n * Enable debug logging\n */\n enable(): void {\n this.config.enabled = true;\n }\n\n /**\n * Disable debug logging\n */\n disable(): void {\n this.config.enabled = false;\n }\n\n /**\n * Format the header for a log message\n * @param direction - 'send' or 'recv'\n * @param byteLength - Number of bytes\n * @returns Formatted header string\n */\n private formatHeader(direction: 'send' | 'recv', byteLength: number): string {\n const prefix = direction === 'send' ? 'CAPNP:SEND' : 'CAPNP:RECV';\n const _directionStr = direction === 'send' ? 'SEND' : 'RECV';\n\n if (isNode() && this.config.colors) {\n const color = direction === 'send' ? ANSI_COLORS.green : ANSI_COLORS.blue;\n return `${color}[${prefix}]${ANSI_COLORS.reset} ${byteLength} bytes`;\n }\n\n return `[${prefix}] ${byteLength} bytes`;\n }\n\n /**\n * Log a Cap'n Proto message\n *\n * Output format:\n * [CAPNP:SEND] 64 bytes\n * 00000000: 00 00 00 00 02 00 00 00 │........│\n * → { messageType: 'Bootstrap', ... }\n *\n * @param direction - 'send' for outgoing, 'recv' for incoming\n * @param data - The raw binary message data\n * @param parsed - Optional parsed message object to display\n */\n logMessage(direction: 'send' | 'recv', data: Uint8Array, parsed?: object): void {\n if (!this.config.enabled) {\n return;\n }\n\n const isNodeEnv = isNode();\n const useColors = this.config.colors;\n\n // Format and log header\n const header = this.formatHeader(direction, data.length);\n\n // Format hex dump\n const hexLines = formatHexDump(data, this.config.maxBytesToLog, useColors);\n\n if (isNodeEnv) {\n // Node.js output with ANSI colors\n console.log(header);\n for (const line of hexLines) {\n console.log(line);\n }\n\n if (parsed !== undefined) {\n const arrow = useColors ? `${ANSI_COLORS.gray}→${ANSI_COLORS.reset}` : '→';\n const parsedStr = JSON.stringify(parsed, null, 2);\n const coloredParsed = useColors\n ? `${ANSI_COLORS.cyan}${parsedStr}${ANSI_COLORS.reset}`\n : parsedStr;\n console.log(`${arrow} ${coloredParsed}`);\n }\n } else {\n // Browser output with CSS\n if (useColors) {\n const style = direction === 'send' ? BROWSER_STYLES.send : BROWSER_STYLES.recv;\n console.log(\n `%c[CAPNP:${direction.toUpperCase()}]%c ${data.length} bytes`,\n style,\n 'color: inherit;'\n );\n\n for (const line of hexLines) {\n console.log(`%c${line}`, BROWSER_STYLES.hex);\n }\n\n if (parsed !== undefined) {\n console.log('%c→%c %o', BROWSER_STYLES.arrow, BROWSER_STYLES.parsed, parsed);\n }\n } else {\n console.log(header);\n for (const line of hexLines) {\n console.log(line);\n }\n if (parsed !== undefined) {\n console.log('→', parsed);\n }\n }\n }\n }\n\n /**\n * Log a generic debug message (only if enabled)\n * @param message - Message to log\n * @param args - Additional arguments\n */\n log(message: string, ...args: unknown[]): void {\n if (!this.config.enabled) {\n return;\n }\n\n if (isNode() && this.config.colors) {\n console.log(`${ANSI_COLORS.gray}[CAPNP:DEBUG]${ANSI_COLORS.reset} ${message}`, ...args);\n } else {\n console.log(`[CAPNP:DEBUG] ${message}`, ...args);\n }\n }\n\n /**\n * Log an error message (always shown if debug is enabled)\n * @param message - Error message\n * @param error - Optional error object\n */\n error(message: string, error?: unknown): void {\n if (!this.config.enabled) {\n return;\n }\n\n if (isNode() && this.config.colors) {\n console.error(`${ANSI_COLORS.red}[CAPNP:ERROR]${ANSI_COLORS.reset} ${message}`, error ?? '');\n } else {\n console.error(`[CAPNP:ERROR] ${message}`, error ?? '');\n }\n }\n}\n\n/**\n * Create a default debug logger instance\n * @param config - Optional configuration\n * @returns DebugLogger instance\n */\nexport function createDebugLogger(config?: Partial<DebugConfig>): DebugLogger {\n return new DebugLogger(config);\n}\n\n/**\n * Global debug logger instance for convenience\n */\nexport const debug = new DebugLogger();\n\n// Export default for convenience\nexport default DebugLogger;\n","/**\n * Stream Abstraction for Cap'n Proto RPC\n *\n * Phase 5: Flow Control and Realtime Communication\n *\n * Provides Stream interface for bidirectional streaming with:\n * - Backpressure handling\n * - Flow control window management\n * - Chunked transfer\n * - Progress notifications\n */\n\nimport type { RpcMessage } from './rpc-types.js';\nimport type { RpcTransport } from './transport.js';\n\n/** Stream state */\nexport type StreamState = 'connecting' | 'open' | 'closing' | 'closed' | 'error';\n\n/** Stream direction */\nexport type StreamDirection = 'inbound' | 'outbound' | 'bidirectional';\n\n/** Stream priority levels */\nexport enum StreamPriority {\n CRITICAL = 0, // Must not drop, immediate delivery\n HIGH = 1, // High priority, minimal delay\n NORMAL = 2, // Normal priority (default)\n LOW = 3, // Low priority, can be delayed\n BACKGROUND = 4, // Background tasks, lowest priority\n}\n\n/** Flow control window configuration */\nexport interface FlowControlConfig {\n /** Initial window size in bytes */\n initialWindowSize: number;\n /** Maximum window size in bytes */\n maxWindowSize: number;\n /** Minimum window size before backpressure */\n minWindowSize: number;\n /** Window update threshold (update when window drops below this) */\n windowUpdateThreshold: number;\n /** Window update increment size */\n windowUpdateIncrement: number;\n}\n\n/** Default flow control configuration */\nexport const DEFAULT_FLOW_CONTROL: FlowControlConfig = {\n initialWindowSize: 65536, // 64KB\n maxWindowSize: 1048576, // 1MB\n minWindowSize: 4096, // 4KB\n windowUpdateThreshold: 16384, // 16KB\n windowUpdateIncrement: 32768, // 32KB\n};\n\n/** Stream configuration options */\nexport interface StreamOptions {\n /** Stream ID (unique within connection) */\n streamId: number;\n /** Stream direction */\n direction: StreamDirection;\n /** Stream priority */\n priority?: StreamPriority;\n /** Flow control configuration */\n flowControl?: Partial<FlowControlConfig>;\n /** Enable progress notifications */\n enableProgress?: boolean;\n /** Progress notification interval in bytes */\n progressInterval?: number;\n /** Stream metadata */\n metadata?: Record<string, string>;\n}\n\n/** Chunk of data in a stream */\nexport interface StreamChunk {\n /** Chunk data */\n data: Uint8Array;\n /** Whether this is the final chunk */\n endOfStream?: boolean;\n /** Chunk sequence number (for ordering) */\n sequenceNumber?: number;\n /** Timestamp when chunk was sent */\n timestamp?: number;\n}\n\n/** Progress notification */\nexport interface StreamProgress {\n /** Stream ID */\n streamId: number;\n /** Total bytes sent/received */\n bytesTransferred: number;\n /** Total bytes expected (if known) */\n totalBytes?: number;\n /** Progress percentage (0-100) */\n percentage?: number;\n /** Transfer rate in bytes per second */\n transferRate?: number;\n /** Estimated time remaining in milliseconds */\n estimatedTimeRemaining?: number;\n}\n\n/** Flow control window state */\ninterface FlowControlWindow {\n /** Current window size */\n currentSize: number;\n /** Maximum window size */\n maxSize: number;\n /** Bytes sent/received in current window */\n bytesInWindow: number;\n /** Whether backpressure is active */\n backpressureActive: boolean;\n}\n\n/** Stream event handlers */\nexport interface StreamEventHandlers {\n /** Called when data is received */\n onData?: (chunk: StreamChunk) => void | Promise<void>;\n /** Called when stream is opened */\n onOpen?: () => void;\n /** Called when stream is closed */\n onClose?: () => void;\n /** Called when progress updates */\n onProgress?: (progress: StreamProgress) => void;\n /** Called when backpressure state changes */\n onBackpressure?: (active: boolean) => void;\n /** Called when an error occurs */\n onError?: (error: Error) => void;\n}\n\n/**\n * Stream abstraction for Cap'n Proto RPC\n *\n * Manages bidirectional streaming with flow control and backpressure.\n */\nexport class Stream {\n private options: StreamOptions;\n private handlers: StreamEventHandlers;\n private state: StreamState = 'connecting';\n private error?: Error;\n\n // Flow control\n private sendWindow: FlowControlWindow;\n private receiveWindow: FlowControlWindow;\n private flowControlConfig: FlowControlConfig;\n\n // Data buffering\n private sendBuffer: StreamChunk[] = [];\n private receiveBuffer: StreamChunk[] = [];\n private maxBufferSize = 1048576; // 1MB max buffer\n\n // Progress tracking\n private bytesSent = 0;\n private bytesReceived = 0;\n private totalBytesExpected?: number;\n private lastProgressUpdate = 0;\n private progressUpdateInterval: number;\n private transferStartTime?: number;\n\n // Chunk sequencing\n private nextSendSequence = 0;\n private nextExpectedSequence = 0;\n\n // Promise resolvers for async operations\n private openResolver?: () => void;\n private openRejector?: (error: Error) => void;\n private closeResolver?: () => void;\n\n constructor(options: StreamOptions, handlers: StreamEventHandlers = {}) {\n this.options = options;\n this.handlers = handlers;\n\n // Initialize flow control\n this.flowControlConfig = {\n ...DEFAULT_FLOW_CONTROL,\n ...options.flowControl,\n };\n\n this.sendWindow = {\n currentSize: this.flowControlConfig.initialWindowSize,\n maxSize: this.flowControlConfig.maxWindowSize,\n bytesInWindow: 0,\n backpressureActive: false,\n };\n\n this.receiveWindow = {\n currentSize: this.flowControlConfig.initialWindowSize,\n maxSize: this.flowControlConfig.maxWindowSize,\n bytesInWindow: 0,\n backpressureActive: false,\n };\n\n this.progressUpdateInterval = options.progressInterval ?? 65536; // 64KB default\n }\n\n /** Get stream ID */\n get id(): number {\n return this.options.streamId;\n }\n\n /** Get stream direction */\n get direction(): StreamDirection {\n return this.options.direction;\n }\n\n /** Get stream priority */\n get priority(): StreamPriority {\n return this.options.priority ?? StreamPriority.NORMAL;\n }\n\n /** Get current stream state */\n get currentState(): StreamState {\n return this.state;\n }\n\n /** Get whether stream is open */\n get isOpen(): boolean {\n return this.state === 'open';\n }\n\n /** Get whether backpressure is active for sending */\n get isBackpressureActive(): boolean {\n return this.sendWindow.backpressureActive;\n }\n\n /** Get bytes sent */\n get bytesSentCount(): number {\n return this.bytesSent;\n }\n\n /** Get bytes received */\n get bytesReceivedCount(): number {\n return this.bytesReceived;\n }\n\n /** Get metadata */\n get metadata(): Record<string, string> | undefined {\n return this.options.metadata;\n }\n\n /**\n * Open the stream\n */\n async open(): Promise<void> {\n if (this.state !== 'connecting') {\n throw new Error(`Cannot open stream in state: ${this.state}`);\n }\n\n return new Promise((resolve, reject) => {\n this.openResolver = resolve;\n this.openRejector = reject;\n\n // Transition to open state\n this.transitionState('open');\n this.transferStartTime = Date.now();\n\n // Notify handlers\n this.handlers.onOpen?.();\n });\n }\n\n /**\n * Send data through the stream\n *\n * Respects flow control and handles backpressure.\n */\n async send(data: Uint8Array, endOfStream = false): Promise<void> {\n if (this.state !== 'open') {\n throw new Error(`Cannot send in state: ${this.state}`);\n }\n\n // Check if we need to wait for window update\n if (this.sendWindow.bytesInWindow + data.length > this.sendWindow.currentSize) {\n // Wait for window update\n await this.waitForWindowUpdate();\n }\n\n // Create chunk\n const chunk: StreamChunk = {\n data,\n endOfStream,\n sequenceNumber: this.nextSendSequence++,\n timestamp: Date.now(),\n };\n\n // Update flow control\n this.sendWindow.bytesInWindow += data.length;\n this.bytesSent += data.length;\n\n // Check for backpressure\n this.checkBackpressure();\n\n // Send the chunk\n await this.sendChunk(chunk);\n\n // Report progress\n this.reportProgress();\n }\n\n /**\n * Send a chunk of data\n *\n * Override in subclasses to implement actual transport.\n */\n protected async sendChunk(chunk: StreamChunk): Promise<void> {\n // To be implemented by transport-specific subclass\n // This base class just buffers\n this.sendBuffer.push(chunk);\n }\n\n /**\n * Receive data from the stream\n *\n * Returns buffered data or waits for new data.\n */\n async receive(): Promise<StreamChunk | null> {\n if (this.receiveBuffer.length > 0) {\n return this.receiveBuffer.shift()!;\n }\n\n if (this.state === 'closed') {\n return null;\n }\n\n // Wait for data\n return new Promise((resolve, reject) => {\n const checkBuffer = () => {\n if (this.receiveBuffer.length > 0) {\n resolve(this.receiveBuffer.shift()!);\n } else if (this.state === 'closed') {\n resolve(null);\n } else if (this.state === 'error') {\n reject(this.error ?? new Error('Stream error'));\n } else {\n setTimeout(checkBuffer, 10);\n }\n };\n checkBuffer();\n });\n }\n\n /**\n * Handle incoming chunk from transport\n */\n handleIncomingChunk(chunk: StreamChunk): void {\n if (this.state !== 'open' && this.state !== 'closing') {\n return;\n }\n\n // Update receive window\n this.receiveWindow.bytesInWindow += chunk.data.length;\n this.bytesReceived += chunk.data.length;\n\n // Check if we need to send window update\n this.checkReceiveWindow();\n\n // Buffer or deliver chunk\n if (this.handlers.onData) {\n // Deliver immediately if handler is registered\n Promise.resolve(this.handlers.onData(chunk)).catch((err) => {\n this.handleError(err);\n });\n } else {\n // Buffer for later\n this.receiveBuffer.push(chunk);\n }\n\n // Report progress\n this.reportProgress();\n\n // Handle end of stream\n if (chunk.endOfStream) {\n this.transitionState('closing');\n }\n }\n\n /**\n * Update the send window (called when receiving window update from peer)\n */\n updateSendWindow(increment: number): void {\n this.sendWindow.currentSize = Math.min(\n this.sendWindow.currentSize + increment,\n this.sendWindow.maxSize\n );\n\n // Check if we can release backpressure\n if (this.sendWindow.backpressureActive) {\n const available = this.sendWindow.currentSize - this.sendWindow.bytesInWindow;\n if (available >= this.flowControlConfig.minWindowSize) {\n this.sendWindow.backpressureActive = false;\n this.handlers.onBackpressure?.(false);\n }\n }\n }\n\n /**\n * Acknowledge received bytes (called to update peer's send window)\n */\n acknowledgeBytes(bytes: number): void {\n this.receiveWindow.bytesInWindow = Math.max(0, this.receiveWindow.bytesInWindow - bytes);\n\n // Check if we should send window update\n if (this.receiveWindow.bytesInWindow < this.flowControlConfig.windowUpdateThreshold) {\n this.sendWindowUpdate();\n }\n }\n\n /**\n * Close the stream gracefully\n */\n async close(): Promise<void> {\n if (this.state === 'closed' || this.state === 'closing') {\n return;\n }\n\n this.transitionState('closing');\n\n // Wait for any pending sends to complete\n await this.drainSendBuffer();\n\n // Send end-of-stream marker\n await this.sendChunk({ data: new Uint8Array(0), endOfStream: true });\n\n this.transitionState('closed');\n this.handlers.onClose?.();\n }\n\n /**\n * Abort the stream with an error\n */\n abort(error: Error): void {\n this.error = error;\n this.transitionState('error');\n this.handlers.onError?.(error);\n }\n\n /**\n * Set total bytes expected (for progress calculation)\n */\n setTotalBytesExpected(total: number): void {\n this.totalBytesExpected = total;\n }\n\n /**\n * Wait for the stream to be ready for sending\n */\n async ready(): Promise<void> {\n if (this.state === 'open' && !this.sendWindow.backpressureActive) {\n return;\n }\n\n if (this.state !== 'open') {\n throw new Error(`Stream not open: ${this.state}`);\n }\n\n return this.waitForWindowUpdate();\n }\n\n // =============================================================================\n // Private Methods\n // =============================================================================\n\n private transitionState(newState: StreamState): void {\n const _oldState = this.state;\n this.state = newState;\n\n // Resolve/reject pending promises\n if (newState === 'open' && this.openResolver) {\n this.openResolver();\n this.openResolver = undefined;\n this.openRejector = undefined;\n } else if (newState === 'error' && this.openRejector) {\n this.openRejector(this.error ?? new Error('Stream error'));\n this.openResolver = undefined;\n this.openRejector = undefined;\n }\n\n if (newState === 'closed' && this.closeResolver) {\n this.closeResolver();\n this.closeResolver = undefined;\n }\n }\n\n private checkBackpressure(): void {\n const available = this.sendWindow.currentSize - this.sendWindow.bytesInWindow;\n\n if (available < this.flowControlConfig.minWindowSize && !this.sendWindow.backpressureActive) {\n this.sendWindow.backpressureActive = true;\n this.handlers.onBackpressure?.(true);\n }\n }\n\n private checkReceiveWindow(): void {\n if (this.receiveWindow.bytesInWindow < this.flowControlConfig.windowUpdateThreshold) {\n this.sendWindowUpdate();\n }\n }\n\n private async waitForWindowUpdate(): Promise<void> {\n return new Promise((resolve, reject) => {\n const checkWindow = () => {\n const available = this.sendWindow.currentSize - this.sendWindow.bytesInWindow;\n\n if (this.state === 'error') {\n reject(this.error ?? new Error('Stream error'));\n return;\n }\n\n if (this.state !== 'open') {\n reject(new Error('Stream closed'));\n return;\n }\n\n if (available >= this.flowControlConfig.minWindowSize) {\n resolve();\n return;\n }\n\n setTimeout(checkWindow, 10);\n };\n\n checkWindow();\n });\n }\n\n private async drainSendBuffer(): Promise<void> {\n // In the base Stream class, sendBuffer is just for buffering\n // Subclasses that actually send data should override this\n // For now, just clear the buffer\n this.sendBuffer.length = 0;\n }\n\n private sendWindowUpdate(): void {\n // To be implemented by transport layer\n // This would send a window update message to the peer\n }\n\n private reportProgress(): void {\n if (!this.options.enableProgress || !this.handlers.onProgress) {\n return;\n }\n\n const now = Date.now();\n const bytesTransferred = Math.max(this.bytesSent, this.bytesReceived);\n\n // Check if it's time to report progress\n if (bytesTransferred - this.lastProgressUpdate < this.progressUpdateInterval) {\n return;\n }\n\n this.lastProgressUpdate = bytesTransferred;\n\n // Calculate transfer rate\n let transferRate: number | undefined;\n if (this.transferStartTime) {\n const elapsed = (now - this.transferStartTime) / 1000;\n if (elapsed > 0) {\n transferRate = bytesTransferred / elapsed;\n }\n }\n\n // Calculate percentage and ETA\n let percentage: number | undefined;\n let estimatedTimeRemaining: number | undefined;\n\n if (this.totalBytesExpected && this.totalBytesExpected > 0) {\n percentage = Math.min(100, (bytesTransferred / this.totalBytesExpected) * 100);\n\n if (transferRate && transferRate > 0) {\n const remaining = this.totalBytesExpected - bytesTransferred;\n estimatedTimeRemaining = (remaining / transferRate) * 1000;\n }\n }\n\n const progress: StreamProgress = {\n streamId: this.id,\n bytesTransferred,\n totalBytes: this.totalBytesExpected,\n percentage,\n transferRate,\n estimatedTimeRemaining,\n };\n\n this.handlers.onProgress(progress);\n }\n\n private handleError(error: Error): void {\n this.error = error;\n this.transitionState('error');\n this.handlers.onError?.(error);\n }\n}\n\n/**\n * Create a new stream\n */\nexport function createStream(options: StreamOptions, handlers?: StreamEventHandlers): Stream {\n return new Stream(options, handlers);\n}\n\n/**\n * Check if an object is a Stream\n */\nexport function isStream(obj: unknown): obj is Stream {\n return obj instanceof Stream;\n}\n","/**\n * Realtime API - Real-time communication with prioritization\n *\n * Phase 5: Flow Control and Realtime Communication\n *\n * Features:\n * - Message priority queues\n * - Message drop policies for latency-sensitive scenarios\n * - Bandwidth adaptation\n * - Jitter buffer management\n */\n\nimport {\n type FlowControlConfig,\n type Stream,\n type StreamChunk,\n type StreamOptions,\n StreamPriority,\n} from './stream.js';\n\n/** Message priority levels (extends StreamPriority) */\nexport { StreamPriority as MessagePriority };\n\n/** Message drop policy for latency-sensitive scenarios */\nexport enum DropPolicy {\n /** Never drop messages */\n NEVER = 'never',\n /** Drop oldest messages when queue is full */\n DROP_OLDEST = 'drop_oldest',\n /** Drop newest messages when queue is full */\n DROP_NEWEST = 'drop_newest',\n /** Drop low priority messages first */\n DROP_LOW_PRIORITY = 'drop_low_priority',\n /** Drop messages that exceed max latency */\n DROP_STALE = 'drop_stale',\n}\n\n/** Realtime stream configuration */\nexport interface RealtimeConfig {\n /** Target latency in milliseconds */\n targetLatencyMs: number;\n /** Maximum acceptable latency in milliseconds */\n maxLatencyMs: number;\n /** Jitter buffer size in milliseconds */\n jitterBufferMs: number;\n /** Message queue size limit */\n maxQueueSize: number;\n /** Drop policy for queue management */\n dropPolicy: DropPolicy;\n /** Enable adaptive bitrate */\n adaptiveBitrate: boolean;\n /** Minimum bitrate in bytes per second */\n minBitrate: number;\n /** Maximum bitrate in bytes per second */\n maxBitrate: number;\n /** Bandwidth measurement window in milliseconds */\n bandwidthWindowMs: number;\n}\n\n/** Default realtime configuration */\nexport const DEFAULT_REALTIME_CONFIG: RealtimeConfig = {\n targetLatencyMs: 50,\n maxLatencyMs: 200,\n jitterBufferMs: 30,\n maxQueueSize: 1000,\n dropPolicy: DropPolicy.DROP_STALE,\n adaptiveBitrate: true,\n minBitrate: 16000, // 16 KB/s\n maxBitrate: 10485760, // 10 MB/s\n bandwidthWindowMs: 1000,\n};\n\n/** Realtime message */\nexport interface RealtimeMessage {\n /** Message ID */\n id: string;\n /** Message priority */\n priority: StreamPriority;\n /** Message timestamp */\n timestamp: number;\n /** Message data */\n data: Uint8Array;\n /** Message type/category */\n type?: string;\n /** Sequence number for ordering */\n sequenceNumber: number;\n /** Whether message is critical (cannot be dropped) */\n critical?: boolean;\n}\n\n/** Bandwidth statistics */\nexport interface BandwidthStats {\n /** Current bitrate in bytes per second */\n currentBitrate: number;\n /** Measured bandwidth in bytes per second */\n measuredBandwidth: number;\n /** Packet loss rate (0-1) */\n packetLossRate: number;\n /** Average latency in milliseconds */\n averageLatencyMs: number;\n /** Jitter in milliseconds */\n jitterMs: number;\n /** Congestion level (0-1) */\n congestionLevel: number;\n}\n\n/** Jitter buffer entry */\ninterface JitterBufferEntry {\n message: RealtimeMessage;\n receivedAt: number;\n playoutTime: number;\n}\n\n/** Realtime stream event handlers */\nexport interface RealtimeStreamHandlers {\n /** Called when a message is received */\n onMessage?: (message: RealtimeMessage) => void | Promise<void>;\n /** Called when messages are dropped */\n onDrop?: (messages: RealtimeMessage[], reason: string) => void;\n /** Called when bandwidth is adapted */\n onBandwidthAdapt?: (newBitrate: number, stats: BandwidthStats) => void;\n /** Called when latency changes */\n onLatencyChange?: (latencyMs: number) => void;\n /** Called when stream is ready to play */\n onReady?: () => void;\n /** Called on error */\n onError?: (error: Error) => void;\n}\n\n/**\n * Priority queue for realtime messages\n */\nclass PriorityMessageQueue {\n private queues: Map<StreamPriority, RealtimeMessage[]> = new Map();\n private totalSize = 0;\n private maxSize: number;\n private dropPolicy: DropPolicy;\n private maxLatencyMs: number;\n\n constructor(maxSize: number, dropPolicy: DropPolicy, maxLatencyMs: number) {\n this.maxSize = maxSize;\n this.dropPolicy = dropPolicy;\n this.maxLatencyMs = maxLatencyMs;\n\n // Initialize queues for each priority level\n for (let i = 0; i <= 4; i++) {\n this.queues.set(i as StreamPriority, []);\n }\n }\n\n /** Get total queue size */\n get size(): number {\n return this.totalSize;\n }\n\n /** Check if queue is empty */\n get isEmpty(): boolean {\n return this.totalSize === 0;\n }\n\n /** Enqueue a message */\n enqueue(message: RealtimeMessage): boolean {\n // Check if message is stale\n if (this.dropPolicy === DropPolicy.DROP_STALE) {\n const age = Date.now() - message.timestamp;\n if (age > this.maxLatencyMs && !message.critical) {\n return false; // Message dropped\n }\n }\n\n // Check if queue is full\n if (this.totalSize >= this.maxSize) {\n if (!this.handleQueueFull(message)) {\n return false;\n }\n }\n\n const queue = this.queues.get(message.priority)!;\n queue.push(message);\n this.totalSize++;\n\n return true;\n }\n\n /** Dequeue the highest priority message */\n dequeue(): RealtimeMessage | undefined {\n // Check queues from highest to lowest priority\n for (let priority = 0; priority <= 4; priority++) {\n const queue = this.queues.get(priority as StreamPriority)!;\n if (queue.length > 0) {\n this.totalSize--;\n return queue.shift()!;\n }\n }\n return undefined;\n }\n\n /** Peek at the highest priority message without removing */\n peek(): RealtimeMessage | undefined {\n for (let priority = 0; priority <= 4; priority++) {\n const queue = this.queues.get(priority as StreamPriority)!;\n if (queue.length > 0) {\n return queue[0];\n }\n }\n return undefined;\n }\n\n /** Remove stale messages */\n removeStale(): RealtimeMessage[] {\n const now = Date.now();\n const removed: RealtimeMessage[] = [];\n\n for (const [priority, queue] of this.queues) {\n const remaining: RealtimeMessage[] = [];\n for (const msg of queue) {\n if (now - msg.timestamp <= this.maxLatencyMs || msg.critical) {\n remaining.push(msg);\n } else {\n removed.push(msg);\n this.totalSize--;\n }\n }\n this.queues.set(priority, remaining);\n }\n\n return removed;\n }\n\n /** Clear all messages */\n clear(): RealtimeMessage[] {\n const all: RealtimeMessage[] = [];\n for (const queue of this.queues.values()) {\n all.push(...queue);\n }\n for (const queue of this.queues.values()) {\n queue.length = 0;\n }\n this.totalSize = 0;\n return all;\n }\n\n private handleQueueFull(newMessage: RealtimeMessage): boolean {\n switch (this.dropPolicy) {\n case DropPolicy.NEVER:\n return false;\n\n case DropPolicy.DROP_OLDEST:\n // Remove oldest message from lowest priority queue\n for (let priority = 4; priority >= 0; priority--) {\n const queue = this.queues.get(priority as StreamPriority)!;\n if (queue.length > 0 && priority >= newMessage.priority) {\n queue.shift();\n this.totalSize--;\n return true;\n }\n }\n return false;\n\n case DropPolicy.DROP_NEWEST:\n // Don't add the new message if it's lower priority\n if (newMessage.priority >= StreamPriority.NORMAL) {\n return false;\n }\n return true;\n\n case DropPolicy.DROP_LOW_PRIORITY:\n // Remove a message from lower priority than the new message\n for (let priority = 4; priority > newMessage.priority; priority--) {\n const queue = this.queues.get(priority as StreamPriority)!;\n if (queue.length > 0) {\n queue.shift();\n this.totalSize--;\n return true;\n }\n }\n return false;\n\n case DropPolicy.DROP_STALE: {\n // Remove stale messages first\n const stale = this.removeStale();\n if (stale.length > 0) {\n return true;\n }\n // Fall through to DROP_OLDEST\n for (let priority = 4; priority >= 0; priority--) {\n const queue = this.queues.get(priority as StreamPriority)!;\n if (queue.length > 0 && priority >= newMessage.priority) {\n queue.shift();\n this.totalSize--;\n return true;\n }\n }\n return false;\n }\n\n default:\n return false;\n }\n }\n}\n\n/**\n * Realtime stream for low-latency communication\n *\n * Manages message prioritization, jitter buffering, and bandwidth adaptation.\n */\nexport class RealtimeStream {\n private stream: Stream;\n private config: RealtimeConfig;\n private handlers: RealtimeStreamHandlers;\n\n // Message queues\n private sendQueue: PriorityMessageQueue;\n private receiveQueue: PriorityMessageQueue;\n\n // Jitter buffer\n private jitterBuffer: JitterBufferEntry[] = [];\n private jitterBufferTargetSize: number;\n\n // Bandwidth adaptation\n private bandwidthStats: BandwidthStats = {\n currentBitrate: 0,\n measuredBandwidth: 0,\n packetLossRate: 0,\n averageLatencyMs: 0,\n jitterMs: 0,\n congestionLevel: 0,\n };\n private bitrateHistory: number[] = [];\n private latencyHistory: number[] = [];\n private lastBandwidthUpdate = 0;\n\n // Sequence numbers\n private nextSendSequence = 0;\n private nextExpectedSequence = 0;\n private receivedSequences: Set<number> = new Set();\n\n // State\n private isRunning = false;\n private sendInterval?: ReturnType<typeof setInterval>;\n private jitterInterval?: ReturnType<typeof setInterval>;\n private bandwidthInterval?: ReturnType<typeof setInterval>;\n\n constructor(\n stream: Stream,\n config: Partial<RealtimeConfig> = {},\n handlers: RealtimeStreamHandlers = {}\n ) {\n this.stream = stream;\n this.config = { ...DEFAULT_REALTIME_CONFIG, ...config };\n this.handlers = handlers;\n\n this.sendQueue = new PriorityMessageQueue(\n this.config.maxQueueSize,\n this.config.dropPolicy,\n this.config.maxLatencyMs\n );\n\n this.receiveQueue = new PriorityMessageQueue(\n this.config.maxQueueSize,\n this.config.dropPolicy,\n this.config.maxLatencyMs\n );\n\n // Calculate jitter buffer target size based on jitter buffer time\n this.jitterBufferTargetSize = Math.ceil(\n this.config.jitterBufferMs / this.config.targetLatencyMs\n );\n\n this.setupStreamHandlers();\n }\n\n /** Get current bandwidth statistics */\n get stats(): BandwidthStats {\n return { ...this.bandwidthStats };\n }\n\n /** Get current send queue size */\n get sendQueueSize(): number {\n return this.sendQueue.size;\n }\n\n /** Get current receive queue size */\n get receiveQueueSize(): number {\n return this.receiveQueue.size;\n }\n\n /** Get jitter buffer size */\n get jitterBufferSize(): number {\n return this.jitterBuffer.length;\n }\n\n /**\n * Start the realtime stream\n */\n start(): void {\n if (this.isRunning) return;\n\n this.isRunning = true;\n\n // Start send loop\n this.sendInterval = setInterval(() => {\n this.processSendQueue();\n }, this.config.targetLatencyMs / 2);\n\n // Start jitter buffer processing\n this.jitterInterval = setInterval(() => {\n this.processJitterBuffer();\n }, this.config.targetLatencyMs / 4);\n\n // Start bandwidth monitoring\n if (this.config.adaptiveBitrate) {\n this.bandwidthInterval = setInterval(() => {\n this.updateBandwidthStats();\n }, this.config.bandwidthWindowMs);\n }\n\n this.handlers.onReady?.();\n }\n\n /**\n * Stop the realtime stream\n */\n stop(): void {\n this.isRunning = false;\n\n if (this.sendInterval) {\n clearInterval(this.sendInterval);\n this.sendInterval = undefined;\n }\n\n if (this.jitterInterval) {\n clearInterval(this.jitterInterval);\n this.jitterInterval = undefined;\n }\n\n if (this.bandwidthInterval) {\n clearInterval(this.bandwidthInterval);\n this.bandwidthInterval = undefined;\n }\n\n // Clear queues\n const dropped = this.sendQueue.clear();\n if (dropped.length > 0) {\n this.handlers.onDrop?.(dropped, 'stream stopped');\n }\n }\n\n /**\n * Send a realtime message\n */\n sendMessage(\n data: Uint8Array,\n priority: StreamPriority = StreamPriority.NORMAL,\n options: { type?: string; critical?: boolean } = {}\n ): boolean {\n if (!this.isRunning) {\n return false;\n }\n\n const message: RealtimeMessage = {\n id: this.generateMessageId(),\n priority,\n timestamp: Date.now(),\n data,\n type: options.type,\n sequenceNumber: this.nextSendSequence++,\n critical: options.critical,\n };\n\n const enqueued = this.sendQueue.enqueue(message);\n\n if (!enqueued) {\n this.handlers.onDrop?.([message], 'queue full');\n return false;\n }\n\n return true;\n }\n\n /**\n * Receive the next message (blocking)\n */\n async receiveMessage(): Promise<RealtimeMessage | undefined> {\n return new Promise((resolve) => {\n const checkQueue = () => {\n const message = this.receiveQueue.dequeue();\n if (message) {\n resolve(message);\n } else if (!this.isRunning) {\n resolve(undefined);\n } else {\n setTimeout(checkQueue, 5);\n }\n };\n checkQueue();\n });\n }\n\n /**\n * Set target bitrate (for manual bitrate control)\n */\n setTargetBitrate(bitrate: number): void {\n this.bandwidthStats.currentBitrate = Math.max(\n this.config.minBitrate,\n Math.min(this.config.maxBitrate, bitrate)\n );\n }\n\n // =============================================================================\n // Private Methods\n // =============================================================================\n\n private setupStreamHandlers(): void {\n // Set up stream data handler\n const originalOnData = (\n this.stream as unknown as { handlers?: { onData?: (chunk: StreamChunk) => void } }\n ).handlers?.onData;\n\n (this.stream as unknown as { handlers: { onData?: (chunk: StreamChunk) => void } }).handlers = {\n ...(this.stream as unknown as { handlers: { onData?: (chunk: StreamChunk) => void } })\n .handlers,\n onData: (chunk: StreamChunk) => {\n this.handleIncomingData(chunk);\n originalOnData?.(chunk);\n },\n };\n }\n\n private handleIncomingData(chunk: StreamChunk): void {\n try {\n // Parse message from chunk\n const message = this.deserializeMessage(chunk.data);\n\n // Track sequence number for packet loss calculation\n if (message.sequenceNumber > this.nextExpectedSequence) {\n // Packet loss detected\n const lost = message.sequenceNumber - this.nextExpectedSequence;\n this.bandwidthStats.packetLossRate = this.bandwidthStats.packetLossRate * 0.9 + lost * 0.1;\n }\n this.nextExpectedSequence = message.sequenceNumber + 1;\n\n // Calculate latency\n const latency = Date.now() - message.timestamp;\n this.latencyHistory.push(latency);\n if (this.latencyHistory.length > 100) {\n this.latencyHistory.shift();\n }\n\n // Add to jitter buffer\n const playoutTime = Date.now() + this.config.jitterBufferMs;\n this.jitterBuffer.push({\n message,\n receivedAt: Date.now(),\n playoutTime,\n });\n\n // Sort jitter buffer by sequence number\n this.jitterBuffer.sort((a, b) => a.message.sequenceNumber - b.message.sequenceNumber);\n } catch (error) {\n this.handlers.onError?.(error as Error);\n }\n }\n\n private processSendQueue(): void {\n if (!this.isRunning || this.sendQueue.isEmpty) {\n return;\n }\n\n // Check bandwidth limit\n if (this.config.adaptiveBitrate) {\n const currentBitrate = this.bandwidthStats.currentBitrate;\n const maxBytesPerInterval = (currentBitrate * this.config.targetLatencyMs) / 1000 / 2;\n\n let bytesSent = 0;\n while (bytesSent < maxBytesPerInterval) {\n const message = this.sendQueue.dequeue();\n if (!message) break;\n\n this.sendMessageToStream(message);\n bytesSent += message.data.length;\n }\n } else {\n // Send one message per interval\n const message = this.sendQueue.dequeue();\n if (message) {\n this.sendMessageToStream(message);\n }\n }\n\n // Remove stale messages from queue\n if (this.config.dropPolicy === DropPolicy.DROP_STALE) {\n const stale = this.sendQueue.removeStale();\n if (stale.length > 0) {\n this.handlers.onDrop?.(stale, 'stale');\n }\n }\n }\n\n private sendMessageToStream(message: RealtimeMessage): void {\n try {\n const data = this.serializeMessage(message);\n this.stream.send(data).catch((err) => {\n this.handlers.onError?.(err);\n });\n\n // Track bitrate\n this.bitrateHistory.push(data.length);\n if (this.bitrateHistory.length > 100) {\n this.bitrateHistory.shift();\n }\n } catch (error) {\n this.handlers.onError?.(error as Error);\n }\n }\n\n private processJitterBuffer(): void {\n if (!this.isRunning || this.jitterBuffer.length === 0) {\n return;\n }\n\n const now = Date.now();\n\n // Process messages that have reached their playout time\n while (this.jitterBuffer.length > 0) {\n const entry = this.jitterBuffer[0];\n\n // Wait until we have enough messages in the buffer\n if (this.jitterBuffer.length < this.jitterBufferTargetSize && entry.playoutTime > now) {\n break;\n }\n\n if (entry.playoutTime <= now) {\n this.jitterBuffer.shift();\n this.receiveQueue.enqueue(entry.message);\n this.handlers.onMessage?.(entry.message);\n } else {\n break;\n }\n }\n\n // Remove stale messages from jitter buffer\n const maxAge = this.config.maxLatencyMs * 2;\n const stale: RealtimeMessage[] = [];\n this.jitterBuffer = this.jitterBuffer.filter((entry) => {\n if (now - entry.receivedAt > maxAge && !entry.message.critical) {\n stale.push(entry.message);\n return false;\n }\n return true;\n });\n\n if (stale.length > 0) {\n this.handlers.onDrop?.(stale, 'jitter buffer stale');\n }\n }\n\n private updateBandwidthStats(): void {\n const now = Date.now();\n const windowMs = this.config.bandwidthWindowMs;\n\n // Calculate measured bandwidth\n if (this.bitrateHistory.length >= 2) {\n const totalBytes = this.bitrateHistory.reduce((a, b) => a + b, 0);\n this.bandwidthStats.measuredBandwidth = (totalBytes / windowMs) * 1000;\n }\n\n // Calculate average latency\n if (this.latencyHistory.length > 0) {\n const avgLatency =\n this.latencyHistory.reduce((a, b) => a + b, 0) / this.latencyHistory.length;\n this.bandwidthStats.averageLatencyMs = avgLatency;\n\n // Calculate jitter (standard deviation of latency)\n const variance =\n this.latencyHistory.reduce((sum, lat) => sum + (lat - avgLatency) ** 2, 0) /\n this.latencyHistory.length;\n this.bandwidthStats.jitterMs = Math.sqrt(variance);\n\n this.handlers.onLatencyChange?.(avgLatency);\n }\n\n // Calculate congestion level\n const queueUtilization = this.sendQueue.size / this.config.maxQueueSize;\n const latencyRatio = this.bandwidthStats.averageLatencyMs / this.config.targetLatencyMs;\n this.bandwidthStats.congestionLevel = Math.min(1, (queueUtilization + latencyRatio) / 2);\n\n // Adapt bitrate\n if (this.config.adaptiveBitrate) {\n this.adaptBitrate();\n }\n\n this.lastBandwidthUpdate = now;\n }\n\n private adaptBitrate(): void {\n const { congestionLevel, packetLossRate } = this.bandwidthStats;\n let newBitrate = this.bandwidthStats.currentBitrate;\n\n if (congestionLevel > 0.7 || packetLossRate > 0.05) {\n // Decrease bitrate\n newBitrate = newBitrate * 0.8;\n } else if (congestionLevel < 0.3 && packetLossRate < 0.01) {\n // Increase bitrate\n newBitrate = newBitrate * 1.05;\n }\n\n // Clamp to min/max\n newBitrate = Math.max(this.config.minBitrate, Math.min(this.config.maxBitrate, newBitrate));\n\n if (newBitrate !== this.bandwidthStats.currentBitrate) {\n this.bandwidthStats.currentBitrate = newBitrate;\n this.handlers.onBandwidthAdapt?.(newBitrate, this.bandwidthStats);\n }\n }\n\n private serializeMessage(message: RealtimeMessage): Uint8Array {\n // Simple serialization: JSON header + binary data\n const header = JSON.stringify({\n id: message.id,\n priority: message.priority,\n timestamp: message.timestamp,\n type: message.type,\n sequenceNumber: message.sequenceNumber,\n critical: message.critical,\n dataLength: message.data.length,\n });\n\n const headerBytes = new TextEncoder().encode(header);\n const headerLength = new Uint8Array(4);\n new DataView(headerLength.buffer).setUint32(0, headerBytes.length, true);\n\n const result = new Uint8Array(4 + headerBytes.length + message.data.length);\n result.set(headerLength, 0);\n result.set(headerBytes, 4);\n result.set(message.data, 4 + headerBytes.length);\n\n return result;\n }\n\n private deserializeMessage(data: Uint8Array): RealtimeMessage {\n const headerLength = new DataView(data.buffer, data.byteOffset, 4).getUint32(0, true);\n const headerBytes = data.slice(4, 4 + headerLength);\n const header = JSON.parse(new TextDecoder().decode(headerBytes));\n\n return {\n id: header.id,\n priority: header.priority,\n timestamp: header.timestamp,\n type: header.type,\n sequenceNumber: header.sequenceNumber,\n critical: header.critical,\n data: data.slice(4 + headerLength, 4 + headerLength + header.dataLength),\n };\n }\n\n private generateMessageId(): string {\n return `${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n }\n}\n\n/**\n * Realtime stream manager\n */\nexport class RealtimeStreamManager {\n private streams: Map<number, RealtimeStream> = new Map();\n private nextStreamId = 1;\n\n /**\n * Create a new realtime stream\n */\n createStream(\n baseStream: Stream,\n config?: Partial<RealtimeConfig>,\n handlers?: RealtimeStreamHandlers\n ): RealtimeStream {\n const streamId = this.nextStreamId++;\n const stream = new RealtimeStream(baseStream, config, handlers);\n this.streams.set(streamId, stream);\n return stream;\n }\n\n /**\n * Get a stream by ID\n */\n getStream(id: number): RealtimeStream | undefined {\n return this.streams.get(id);\n }\n\n /**\n * Remove a stream\n */\n removeStream(id: number): boolean {\n const stream = this.streams.get(id);\n if (stream) {\n stream.stop();\n this.streams.delete(id);\n return true;\n }\n return false;\n }\n\n /**\n * Get all active streams\n */\n getActiveStreams(): RealtimeStream[] {\n return Array.from(this.streams.values());\n }\n\n /**\n * Stop all streams\n */\n stopAll(): void {\n for (const stream of this.streams.values()) {\n stream.stop();\n }\n this.streams.clear();\n }\n}\n\n/**\n * Create a realtime stream manager\n */\nexport function createRealtimeStreamManager(): RealtimeStreamManager {\n return new RealtimeStreamManager();\n}\n","/**\n * Stream Management - Unified stream lifecycle management\n *\n * Phase 5: Flow Control and Realtime Communication\n *\n * Manages:\n * - Stream creation and registration\n * - Bidirectional stream support\n * - Stream lifecycle (open, close, error handling)\n * - Stream multiplexing over a single connection\n */\n\nimport {\n type BulkTransfer,\n type BulkTransferConfig,\n BulkTransferManager,\n type BulkTransferMetadata,\n} from './bulk.js';\nimport {\n type RealtimeConfig,\n type RealtimeStream,\n type RealtimeStreamHandlers,\n RealtimeStreamManager,\n} from './realtime.js';\nimport type { RpcConnection } from './rpc-connection.js';\nimport {\n Stream,\n type StreamDirection,\n type StreamOptions,\n StreamPriority,\n type StreamState,\n} from './stream.js';\nimport type { RpcTransport } from './transport.js';\n\n/** Stream type */\nexport enum StreamType {\n /** Standard stream */\n STANDARD = 'standard',\n /** Bulk transfer stream */\n BULK = 'bulk',\n /** Realtime stream */\n REALTIME = 'realtime',\n}\n\n/** Stream info */\nexport interface StreamInfo {\n /** Stream ID */\n id: number;\n /** Stream type */\n type: StreamType;\n /** Stream direction */\n direction: StreamDirection;\n /** Stream priority */\n priority: StreamPriority;\n /** Current state */\n state: StreamState;\n /** Creation timestamp */\n createdAt: number;\n /** Bytes transferred */\n bytesTransferred: number;\n /** Metadata */\n metadata?: Record<string, string>;\n}\n\n/** Stream manager configuration */\nexport interface StreamManagerConfig {\n /** Maximum number of concurrent streams */\n maxStreams: number;\n /** Default stream priority */\n defaultPriority: StreamPriority;\n /** Enable stream multiplexing */\n enableMultiplexing: boolean;\n /** Idle timeout in milliseconds */\n idleTimeoutMs: number;\n}\n\n/** Default stream manager configuration */\nexport const DEFAULT_STREAM_MANAGER_CONFIG: StreamManagerConfig = {\n maxStreams: 100,\n defaultPriority: StreamPriority.NORMAL,\n enableMultiplexing: true,\n idleTimeoutMs: 300000, // 5 minutes\n};\n\n/** Stream creation options */\nexport interface CreateStreamOptions {\n /** Stream type */\n type?: StreamType;\n /** Stream direction */\n direction?: StreamDirection;\n /** Stream priority */\n priority?: StreamPriority;\n /** Stream metadata */\n metadata?: Record<string, string>;\n /** Flow control configuration */\n flowControl?: Partial<import('./stream.js').FlowControlConfig>;\n}\n\n/** Stream manager event handlers */\nexport interface StreamManagerHandlers {\n /** Called when a stream is created */\n onStreamCreate?: (info: StreamInfo) => void;\n /** Called when a stream is opened */\n onStreamOpen?: (info: StreamInfo) => void;\n /** Called when a stream is closed */\n onStreamClose?: (info: StreamInfo) => void;\n /** Called when a stream errors */\n onStreamError?: (info: StreamInfo, error: Error) => void;\n /** Called when stream count changes */\n onStreamCountChange?: (count: number) => void;\n}\n\n/**\n * Stream manager\n *\n * Manages all streams for an RPC connection.\n */\nexport class StreamManager {\n private config: StreamManagerConfig;\n private handlers: StreamManagerHandlers;\n private connection?: RpcConnection;\n private transport?: RpcTransport;\n\n // Stream registries\n private streams: Map<number, Stream> = new Map();\n private streamTypes: Map<number, StreamType> = new Map();\n private streamInfos: Map<number, StreamInfo> = new Map();\n private nextStreamId = 1;\n\n // Sub-managers\n private bulkManager: BulkTransferManager;\n private realtimeManager: RealtimeStreamManager;\n\n // Lifecycle\n private idleTimeout?: ReturnType<typeof setTimeout>;\n private isRunning = false;\n\n constructor(config: Partial<StreamManagerConfig> = {}, handlers: StreamManagerHandlers = {}) {\n this.config = { ...DEFAULT_STREAM_MANAGER_CONFIG, ...config };\n this.handlers = handlers;\n this.bulkManager = new BulkTransferManager();\n this.realtimeManager = new RealtimeStreamManager();\n }\n\n /** Get the number of active streams */\n get streamCount(): number {\n return this.streams.size;\n }\n\n /** Get the maximum number of streams */\n get maxStreams(): number {\n return this.config.maxStreams;\n }\n\n /** Get all stream infos */\n get allStreamInfos(): StreamInfo[] {\n return Array.from(this.streamInfos.values());\n }\n\n /**\n * Attach to an RPC connection\n */\n attach(connection: RpcConnection, transport?: RpcTransport): void {\n this.connection = connection;\n this.transport = transport;\n this.isRunning = true;\n this.resetIdleTimeout();\n }\n\n /**\n * Detach from connection\n */\n detach(): void {\n this.closeAllStreams();\n this.isRunning = false;\n this.connection = undefined;\n this.transport = undefined;\n\n if (this.idleTimeout) {\n clearTimeout(this.idleTimeout);\n this.idleTimeout = undefined;\n }\n }\n\n /**\n * Create a new stream\n */\n createStream(options: CreateStreamOptions = {}): Stream {\n if (this.streams.size >= this.config.maxStreams) {\n throw new Error(`Maximum number of streams (${this.config.maxStreams}) reached`);\n }\n\n const streamId = this.nextStreamId++;\n const type = options.type ?? StreamType.STANDARD;\n\n const streamOptions: StreamOptions = {\n streamId,\n direction: options.direction ?? 'bidirectional',\n priority: options.priority ?? this.config.defaultPriority ?? StreamPriority.NORMAL,\n metadata: options.metadata,\n flowControl: options.flowControl,\n };\n\n const stream = new Stream(streamOptions, {\n onOpen: () => {\n this.updateStreamState(streamId, 'open');\n this.handlers.onStreamOpen?.(this.getStreamInfo(streamId)!);\n },\n onClose: () => {\n this.updateStreamState(streamId, 'closed');\n this.handlers.onStreamClose?.(this.getStreamInfo(streamId)!);\n this.removeStream(streamId);\n },\n onError: (error) => {\n this.updateStreamState(streamId, 'error');\n const info = this.getStreamInfo(streamId);\n if (info) {\n this.handlers.onStreamError?.(info, error);\n }\n },\n });\n\n this.streams.set(streamId, stream);\n this.streamTypes.set(streamId, type);\n\n const info: StreamInfo = {\n id: streamId,\n type,\n direction: streamOptions.direction,\n priority: streamOptions.priority ?? StreamPriority.NORMAL,\n state: 'connecting',\n createdAt: Date.now(),\n bytesTransferred: 0,\n metadata: options.metadata,\n };\n this.streamInfos.set(streamId, info);\n\n this.handlers.onStreamCreate?.(info);\n this.handlers.onStreamCountChange?.(this.streams.size);\n\n this.resetIdleTimeout();\n\n return stream;\n }\n\n /**\n * Create a bulk transfer stream\n */\n createBulkStream(\n direction: 'upload' | 'download',\n metadata: BulkTransferMetadata,\n config?: Partial<BulkTransferConfig>,\n handlers?: import('./bulk.js').BulkTransferHandlers\n ): BulkTransfer {\n const _stream = this.createStream({\n type: StreamType.BULK,\n direction: direction === 'upload' ? 'outbound' : 'inbound',\n priority: StreamPriority.NORMAL,\n metadata: metadata.custom,\n });\n\n return this.bulkManager.createTransfer(direction, metadata, config, handlers);\n }\n\n /**\n * Create a realtime stream\n */\n createRealtimeStream(\n config?: Partial<RealtimeConfig>,\n handlers?: RealtimeStreamHandlers\n ): RealtimeStream {\n const stream = this.createStream({\n type: StreamType.REALTIME,\n direction: 'bidirectional',\n priority: StreamPriority.HIGH,\n });\n\n return this.realtimeManager.createStream(stream, config, handlers);\n }\n\n /**\n * Get a stream by ID\n */\n getStream(id: number): Stream | undefined {\n return this.streams.get(id);\n }\n\n /**\n * Get stream info by ID\n */\n getStreamInfo(id: number): StreamInfo | undefined {\n return this.streamInfos.get(id);\n }\n\n /**\n * Get stream type by ID\n */\n getStreamType(id: number): StreamType | undefined {\n return this.streamTypes.get(id);\n }\n\n /**\n * Get bulk transfer by ID\n */\n getBulkTransfer(id: string): BulkTransfer | undefined {\n return this.bulkManager.getTransfer(id);\n }\n\n /**\n * Get realtime stream by ID\n */\n getRealtimeStream(id: number): RealtimeStream | undefined {\n return this.realtimeManager.getStream(id);\n }\n\n /**\n * Close a specific stream\n */\n async closeStream(id: number): Promise<boolean> {\n const stream = this.streams.get(id);\n if (!stream) {\n return false;\n }\n\n await stream.close();\n this.removeStream(id);\n return true;\n }\n\n /**\n * Close all streams\n */\n async closeAllStreams(): Promise<void> {\n const closePromises: Promise<void>[] = [];\n\n for (const [_id, stream] of this.streams) {\n closePromises.push(\n stream.close().catch(() => {\n // Ignore errors during bulk close\n })\n );\n }\n\n await Promise.all(closePromises);\n\n this.streams.clear();\n this.streamTypes.clear();\n this.streamInfos.clear();\n this.bulkManager.closeAll();\n this.realtimeManager.stopAll();\n\n this.handlers.onStreamCountChange?.(0);\n }\n\n /**\n * Get streams by type\n */\n getStreamsByType(type: StreamType): StreamInfo[] {\n return this.allStreamInfos.filter((info) => info.type === type);\n }\n\n /**\n * Get streams by state\n */\n getStreamsByState(state: StreamState): StreamInfo[] {\n return this.allStreamInfos.filter((info) => info.state === state);\n }\n\n /**\n * Get statistics\n */\n getStatistics(): {\n totalStreams: number;\n activeStreams: number;\n streamsByType: Record<StreamType, number>;\n streamsByState: Record<StreamState, number>;\n totalBytesTransferred: number;\n } {\n const infos = this.allStreamInfos;\n const active = infos.filter((i) => i.state === 'open');\n\n const byType: Record<StreamType, number> = {\n [StreamType.STANDARD]: 0,\n [StreamType.BULK]: 0,\n [StreamType.REALTIME]: 0,\n };\n\n const byState: Record<StreamState, number> = {\n connecting: 0,\n open: 0,\n closing: 0,\n closed: 0,\n error: 0,\n };\n\n let totalBytes = 0;\n\n for (const info of infos) {\n byType[info.type]++;\n byState[info.state]++;\n totalBytes += info.bytesTransferred;\n }\n\n return {\n totalStreams: infos.length,\n activeStreams: active.length,\n streamsByType: byType,\n streamsByState: byState,\n totalBytesTransferred: totalBytes,\n };\n }\n\n /**\n * Update stream priority\n */\n updatePriority(id: number, priority: StreamPriority): boolean {\n const info = this.streamInfos.get(id);\n if (!info) {\n return false;\n }\n\n info.priority = priority;\n return true;\n }\n\n /**\n * Pause all streams (backpressure)\n */\n pauseAll(): void {\n for (const _stream of this.streams.values()) {\n // Streams don't have a direct pause method, but we can use backpressure\n // This is a simplified implementation\n }\n }\n\n /**\n * Resume all streams\n */\n resumeAll(): void {\n for (const _stream of this.streams.values()) {\n // Resume from backpressure\n }\n }\n\n // =============================================================================\n // Private Methods\n // =============================================================================\n\n private removeStream(id: number): void {\n this.streams.delete(id);\n this.streamTypes.delete(id);\n this.streamInfos.delete(id);\n\n this.handlers.onStreamCountChange?.(this.streams.size);\n this.resetIdleTimeout();\n }\n\n private updateStreamState(id: number, state: StreamState): void {\n const info = this.streamInfos.get(id);\n if (info) {\n info.state = state;\n }\n }\n\n private resetIdleTimeout(): void {\n if (this.idleTimeout) {\n clearTimeout(this.idleTimeout);\n }\n\n if (!this.isRunning || this.streams.size > 0) {\n return;\n }\n\n this.idleTimeout = setTimeout(() => {\n if (this.streams.size === 0) {\n // No active streams, could emit an event or take action\n }\n }, this.config.idleTimeoutMs);\n }\n}\n\n/**\n * Create a stream manager\n */\nexport function createStreamManager(\n config?: Partial<StreamManagerConfig>,\n handlers?: StreamManagerHandlers\n): StreamManager {\n return new StreamManager(config, handlers);\n}\n\n// Re-export types\nexport type { Stream, StreamOptions, StreamState, StreamDirection } from './stream.js';\nexport { StreamPriority } from './stream.js';\nexport type {\n BulkTransfer,\n BulkTransferConfig,\n BulkTransferMetadata,\n} from './bulk.js';\nexport { BulkTransferManager } from './bulk.js';\nexport type {\n RealtimeStream,\n RealtimeConfig,\n RealtimeStreamHandlers,\n} from './realtime.js';\nexport { RealtimeStreamManager } from './realtime.js';\n","'use strict';\n\nconst BINARY_TYPES = ['nodebuffer', 'arraybuffer', 'fragments'];\nconst hasBlob = typeof Blob !== 'undefined';\n\nif (hasBlob) BINARY_TYPES.push('blob');\n\nmodule.exports = {\n BINARY_TYPES,\n CLOSE_TIMEOUT: 30000,\n EMPTY_BUFFER: Buffer.alloc(0),\n GUID: '258EAFA5-E914-47DA-95CA-C5AB0DC85B11',\n hasBlob,\n kForOnEventAttribute: Symbol('kIsForOnEventAttribute'),\n kListener: Symbol('kListener'),\n kStatusCode: Symbol('status-code'),\n kWebSocket: Symbol('websocket'),\n NOOP: () => {}\n};\n","'use strict';\n\nconst { EMPTY_BUFFER } = require('./constants');\n\nconst FastBuffer = Buffer[Symbol.species];\n\n/**\n * Merges an array of buffers into a new buffer.\n *\n * @param {Buffer[]} list The array of buffers to concat\n * @param {Number} totalLength The total length of buffers in the list\n * @return {Buffer} The resulting buffer\n * @public\n */\nfunction concat(list, totalLength) {\n if (list.length === 0) return EMPTY_BUFFER;\n if (list.length === 1) return list[0];\n\n const target = Buffer.allocUnsafe(totalLength);\n let offset = 0;\n\n for (let i = 0; i < list.length; i++) {\n const buf = list[i];\n target.set(buf, offset);\n offset += buf.length;\n }\n\n if (offset < totalLength) {\n return new FastBuffer(target.buffer, target.byteOffset, offset);\n }\n\n return target;\n}\n\n/**\n * Masks a buffer using the given mask.\n *\n * @param {Buffer} source The buffer to mask\n * @param {Buffer} mask The mask to use\n * @param {Buffer} output The buffer where to store the result\n * @param {Number} offset The offset at which to start writing\n * @param {Number} length The number of bytes to mask.\n * @public\n */\nfunction _mask(source, mask, output, offset, length) {\n for (let i = 0; i < length; i++) {\n output[offset + i] = source[i] ^ mask[i & 3];\n }\n}\n\n/**\n * Unmasks a buffer using the given mask.\n *\n * @param {Buffer} buffer The buffer to unmask\n * @param {Buffer} mask The mask to use\n * @public\n */\nfunction _unmask(buffer, mask) {\n for (let i = 0; i < buffer.length; i++) {\n buffer[i] ^= mask[i & 3];\n }\n}\n\n/**\n * Converts a buffer to an `ArrayBuffer`.\n *\n * @param {Buffer} buf The buffer to convert\n * @return {ArrayBuffer} Converted buffer\n * @public\n */\nfunction toArrayBuffer(buf) {\n if (buf.length === buf.buffer.byteLength) {\n return buf.buffer;\n }\n\n return buf.buffer.slice(buf.byteOffset, buf.byteOffset + buf.length);\n}\n\n/**\n * Converts `data` to a `Buffer`.\n *\n * @param {*} data The data to convert\n * @return {Buffer} The buffer\n * @throws {TypeError}\n * @public\n */\nfunction toBuffer(data) {\n toBuffer.readOnly = true;\n\n if (Buffer.isBuffer(data)) return data;\n\n let buf;\n\n if (data instanceof ArrayBuffer) {\n buf = new FastBuffer(data);\n } else if (ArrayBuffer.isView(data)) {\n buf = new FastBuffer(data.buffer, data.byteOffset, data.byteLength);\n } else {\n buf = Buffer.from(data);\n toBuffer.readOnly = false;\n }\n\n return buf;\n}\n\nmodule.exports = {\n concat,\n mask: _mask,\n toArrayBuffer,\n toBuffer,\n unmask: _unmask\n};\n\n/* istanbul ignore else */\nif (!process.env.WS_NO_BUFFER_UTIL) {\n try {\n const bufferUtil = require('bufferutil');\n\n module.exports.mask = function (source, mask, output, offset, length) {\n if (length < 48) _mask(source, mask, output, offset, length);\n else bufferUtil.mask(source, mask, output, offset, length);\n };\n\n module.exports.unmask = function (buffer, mask) {\n if (buffer.length < 32) _unmask(buffer, mask);\n else bufferUtil.unmask(buffer, mask);\n };\n } catch (e) {\n // Continue regardless of the error.\n }\n}\n","'use strict';\n\nconst kDone = Symbol('kDone');\nconst kRun = Symbol('kRun');\n\n/**\n * A very simple job queue with adjustable concurrency. Adapted from\n * https://github.com/STRML/async-limiter\n */\nclass Limiter {\n /**\n * Creates a new `Limiter`.\n *\n * @param {Number} [concurrency=Infinity] The maximum number of jobs allowed\n * to run concurrently\n */\n constructor(concurrency) {\n this[kDone] = () => {\n this.pending--;\n this[kRun]();\n };\n this.concurrency = concurrency || Infinity;\n this.jobs = [];\n this.pending = 0;\n }\n\n /**\n * Adds a job to the queue.\n *\n * @param {Function} job The job to run\n * @public\n */\n add(job) {\n this.jobs.push(job);\n this[kRun]();\n }\n\n /**\n * Removes a job from the queue and runs it if possible.\n *\n * @private\n */\n [kRun]() {\n if (this.pending === this.concurrency) return;\n\n if (this.jobs.length) {\n const job = this.jobs.shift();\n\n this.pending++;\n job(this[kDone]);\n }\n }\n}\n\nmodule.exports = Limiter;\n","'use strict';\n\nconst zlib = require('zlib');\n\nconst bufferUtil = require('./buffer-util');\nconst Limiter = require('./limiter');\nconst { kStatusCode } = require('./constants');\n\nconst FastBuffer = Buffer[Symbol.species];\nconst TRAILER = Buffer.from([0x00, 0x00, 0xff, 0xff]);\nconst kPerMessageDeflate = Symbol('permessage-deflate');\nconst kTotalLength = Symbol('total-length');\nconst kCallback = Symbol('callback');\nconst kBuffers = Symbol('buffers');\nconst kError = Symbol('error');\n\n//\n// We limit zlib concurrency, which prevents severe memory fragmentation\n// as documented in https://github.com/nodejs/node/issues/8871#issuecomment-250915913\n// and https://github.com/websockets/ws/issues/1202\n//\n// Intentionally global; it's the global thread pool that's an issue.\n//\nlet zlibLimiter;\n\n/**\n * permessage-deflate implementation.\n */\nclass PerMessageDeflate {\n /**\n * Creates a PerMessageDeflate instance.\n *\n * @param {Object} [options] Configuration options\n * @param {(Boolean|Number)} [options.clientMaxWindowBits] Advertise support\n * for, or request, a custom client window size\n * @param {Boolean} [options.clientNoContextTakeover=false] Advertise/\n * acknowledge disabling of client context takeover\n * @param {Number} [options.concurrencyLimit=10] The number of concurrent\n * calls to zlib\n * @param {(Boolean|Number)} [options.serverMaxWindowBits] Request/confirm the\n * use of a custom server window size\n * @param {Boolean} [options.serverNoContextTakeover=false] Request/accept\n * disabling of server context takeover\n * @param {Number} [options.threshold=1024] Size (in bytes) below which\n * messages should not be compressed if context takeover is disabled\n * @param {Object} [options.zlibDeflateOptions] Options to pass to zlib on\n * deflate\n * @param {Object} [options.zlibInflateOptions] Options to pass to zlib on\n * inflate\n * @param {Boolean} [isServer=false] Create the instance in either server or\n * client mode\n * @param {Number} [maxPayload=0] The maximum allowed message length\n */\n constructor(options, isServer, maxPayload) {\n this._maxPayload = maxPayload | 0;\n this._options = options || {};\n this._threshold =\n this._options.threshold !== undefined ? this._options.threshold : 1024;\n this._isServer = !!isServer;\n this._deflate = null;\n this._inflate = null;\n\n this.params = null;\n\n if (!zlibLimiter) {\n const concurrency =\n this._options.concurrencyLimit !== undefined\n ? this._options.concurrencyLimit\n : 10;\n zlibLimiter = new Limiter(concurrency);\n }\n }\n\n /**\n * @type {String}\n */\n static get extensionName() {\n return 'permessage-deflate';\n }\n\n /**\n * Create an extension negotiation offer.\n *\n * @return {Object} Extension parameters\n * @public\n */\n offer() {\n const params = {};\n\n if (this._options.serverNoContextTakeover) {\n params.server_no_context_takeover = true;\n }\n if (this._options.clientNoContextTakeover) {\n params.client_no_context_takeover = true;\n }\n if (this._options.serverMaxWindowBits) {\n params.server_max_window_bits = this._options.serverMaxWindowBits;\n }\n if (this._options.clientMaxWindowBits) {\n params.client_max_window_bits = this._options.clientMaxWindowBits;\n } else if (this._options.clientMaxWindowBits == null) {\n params.client_max_window_bits = true;\n }\n\n return params;\n }\n\n /**\n * Accept an extension negotiation offer/response.\n *\n * @param {Array} configurations The extension negotiation offers/reponse\n * @return {Object} Accepted configuration\n * @public\n */\n accept(configurations) {\n configurations = this.normalizeParams(configurations);\n\n this.params = this._isServer\n ? this.acceptAsServer(configurations)\n : this.acceptAsClient(configurations);\n\n return this.params;\n }\n\n /**\n * Releases all resources used by the extension.\n *\n * @public\n */\n cleanup() {\n if (this._inflate) {\n this._inflate.close();\n this._inflate = null;\n }\n\n if (this._deflate) {\n const callback = this._deflate[kCallback];\n\n this._deflate.close();\n this._deflate = null;\n\n if (callback) {\n callback(\n new Error(\n 'The deflate stream was closed while data was being processed'\n )\n );\n }\n }\n }\n\n /**\n * Accept an extension negotiation offer.\n *\n * @param {Array} offers The extension negotiation offers\n * @return {Object} Accepted configuration\n * @private\n */\n acceptAsServer(offers) {\n const opts = this._options;\n const accepted = offers.find((params) => {\n if (\n (opts.serverNoContextTakeover === false &&\n params.server_no_context_takeover) ||\n (params.server_max_window_bits &&\n (opts.serverMaxWindowBits === false ||\n (typeof opts.serverMaxWindowBits === 'number' &&\n opts.serverMaxWindowBits > params.server_max_window_bits))) ||\n (typeof opts.clientMaxWindowBits === 'number' &&\n !params.client_max_window_bits)\n ) {\n return false;\n }\n\n return true;\n });\n\n if (!accepted) {\n throw new Error('None of the extension offers can be accepted');\n }\n\n if (opts.serverNoContextTakeover) {\n accepted.server_no_context_takeover = true;\n }\n if (opts.clientNoContextTakeover) {\n accepted.client_no_context_takeover = true;\n }\n if (typeof opts.serverMaxWindowBits === 'number') {\n accepted.server_max_window_bits = opts.serverMaxWindowBits;\n }\n if (typeof opts.clientMaxWindowBits === 'number') {\n accepted.client_max_window_bits = opts.clientMaxWindowBits;\n } else if (\n accepted.client_max_window_bits === true ||\n opts.clientMaxWindowBits === false\n ) {\n delete accepted.client_max_window_bits;\n }\n\n return accepted;\n }\n\n /**\n * Accept the extension negotiation response.\n *\n * @param {Array} response The extension negotiation response\n * @return {Object} Accepted configuration\n * @private\n */\n acceptAsClient(response) {\n const params = response[0];\n\n if (\n this._options.clientNoContextTakeover === false &&\n params.client_no_context_takeover\n ) {\n throw new Error('Unexpected parameter \"client_no_context_takeover\"');\n }\n\n if (!params.client_max_window_bits) {\n if (typeof this._options.clientMaxWindowBits === 'number') {\n params.client_max_window_bits = this._options.clientMaxWindowBits;\n }\n } else if (\n this._options.clientMaxWindowBits === false ||\n (typeof this._options.clientMaxWindowBits === 'number' &&\n params.client_max_window_bits > this._options.clientMaxWindowBits)\n ) {\n throw new Error(\n 'Unexpected or invalid parameter \"client_max_window_bits\"'\n );\n }\n\n return params;\n }\n\n /**\n * Normalize parameters.\n *\n * @param {Array} configurations The extension negotiation offers/reponse\n * @return {Array} The offers/response with normalized parameters\n * @private\n */\n normalizeParams(configurations) {\n configurations.forEach((params) => {\n Object.keys(params).forEach((key) => {\n let value = params[key];\n\n if (value.length > 1) {\n throw new Error(`Parameter \"${key}\" must have only a single value`);\n }\n\n value = value[0];\n\n if (key === 'client_max_window_bits') {\n if (value !== true) {\n const num = +value;\n if (!Number.isInteger(num) || num < 8 || num > 15) {\n throw new TypeError(\n `Invalid value for parameter \"${key}\": ${value}`\n );\n }\n value = num;\n } else if (!this._isServer) {\n throw new TypeError(\n `Invalid value for parameter \"${key}\": ${value}`\n );\n }\n } else if (key === 'server_max_window_bits') {\n const num = +value;\n if (!Number.isInteger(num) || num < 8 || num > 15) {\n throw new TypeError(\n `Invalid value for parameter \"${key}\": ${value}`\n );\n }\n value = num;\n } else if (\n key === 'client_no_context_takeover' ||\n key === 'server_no_context_takeover'\n ) {\n if (value !== true) {\n throw new TypeError(\n `Invalid value for parameter \"${key}\": ${value}`\n );\n }\n } else {\n throw new Error(`Unknown parameter \"${key}\"`);\n }\n\n params[key] = value;\n });\n });\n\n return configurations;\n }\n\n /**\n * Decompress data. Concurrency limited.\n *\n * @param {Buffer} data Compressed data\n * @param {Boolean} fin Specifies whether or not this is the last fragment\n * @param {Function} callback Callback\n * @public\n */\n decompress(data, fin, callback) {\n zlibLimiter.add((done) => {\n this._decompress(data, fin, (err, result) => {\n done();\n callback(err, result);\n });\n });\n }\n\n /**\n * Compress data. Concurrency limited.\n *\n * @param {(Buffer|String)} data Data to compress\n * @param {Boolean} fin Specifies whether or not this is the last fragment\n * @param {Function} callback Callback\n * @public\n */\n compress(data, fin, callback) {\n zlibLimiter.add((done) => {\n this._compress(data, fin, (err, result) => {\n done();\n callback(err, result);\n });\n });\n }\n\n /**\n * Decompress data.\n *\n * @param {Buffer} data Compressed data\n * @param {Boolean} fin Specifies whether or not this is the last fragment\n * @param {Function} callback Callback\n * @private\n */\n _decompress(data, fin, callback) {\n const endpoint = this._isServer ? 'client' : 'server';\n\n if (!this._inflate) {\n const key = `${endpoint}_max_window_bits`;\n const windowBits =\n typeof this.params[key] !== 'number'\n ? zlib.Z_DEFAULT_WINDOWBITS\n : this.params[key];\n\n this._inflate = zlib.createInflateRaw({\n ...this._options.zlibInflateOptions,\n windowBits\n });\n this._inflate[kPerMessageDeflate] = this;\n this._inflate[kTotalLength] = 0;\n this._inflate[kBuffers] = [];\n this._inflate.on('error', inflateOnError);\n this._inflate.on('data', inflateOnData);\n }\n\n this._inflate[kCallback] = callback;\n\n this._inflate.write(data);\n if (fin) this._inflate.write(TRAILER);\n\n this._inflate.flush(() => {\n const err = this._inflate[kError];\n\n if (err) {\n this._inflate.close();\n this._inflate = null;\n callback(err);\n return;\n }\n\n const data = bufferUtil.concat(\n this._inflate[kBuffers],\n this._inflate[kTotalLength]\n );\n\n if (this._inflate._readableState.endEmitted) {\n this._inflate.close();\n this._inflate = null;\n } else {\n this._inflate[kTotalLength] = 0;\n this._inflate[kBuffers] = [];\n\n if (fin && this.params[`${endpoint}_no_context_takeover`]) {\n this._inflate.reset();\n }\n }\n\n callback(null, data);\n });\n }\n\n /**\n * Compress data.\n *\n * @param {(Buffer|String)} data Data to compress\n * @param {Boolean} fin Specifies whether or not this is the last fragment\n * @param {Function} callback Callback\n * @private\n */\n _compress(data, fin, callback) {\n const endpoint = this._isServer ? 'server' : 'client';\n\n if (!this._deflate) {\n const key = `${endpoint}_max_window_bits`;\n const windowBits =\n typeof this.params[key] !== 'number'\n ? zlib.Z_DEFAULT_WINDOWBITS\n : this.params[key];\n\n this._deflate = zlib.createDeflateRaw({\n ...this._options.zlibDeflateOptions,\n windowBits\n });\n\n this._deflate[kTotalLength] = 0;\n this._deflate[kBuffers] = [];\n\n this._deflate.on('data', deflateOnData);\n }\n\n this._deflate[kCallback] = callback;\n\n this._deflate.write(data);\n this._deflate.flush(zlib.Z_SYNC_FLUSH, () => {\n if (!this._deflate) {\n //\n // The deflate stream was closed while data was being processed.\n //\n return;\n }\n\n let data = bufferUtil.concat(\n this._deflate[kBuffers],\n this._deflate[kTotalLength]\n );\n\n if (fin) {\n data = new FastBuffer(data.buffer, data.byteOffset, data.length - 4);\n }\n\n //\n // Ensure that the callback will not be called again in\n // `PerMessageDeflate#cleanup()`.\n //\n this._deflate[kCallback] = null;\n\n this._deflate[kTotalLength] = 0;\n this._deflate[kBuffers] = [];\n\n if (fin && this.params[`${endpoint}_no_context_takeover`]) {\n this._deflate.reset();\n }\n\n callback(null, data);\n });\n }\n}\n\nmodule.exports = PerMessageDeflate;\n\n/**\n * The listener of the `zlib.DeflateRaw` stream `'data'` event.\n *\n * @param {Buffer} chunk A chunk of data\n * @private\n */\nfunction deflateOnData(chunk) {\n this[kBuffers].push(chunk);\n this[kTotalLength] += chunk.length;\n}\n\n/**\n * The listener of the `zlib.InflateRaw` stream `'data'` event.\n *\n * @param {Buffer} chunk A chunk of data\n * @private\n */\nfunction inflateOnData(chunk) {\n this[kTotalLength] += chunk.length;\n\n if (\n this[kPerMessageDeflate]._maxPayload < 1 ||\n this[kTotalLength] <= this[kPerMessageDeflate]._maxPayload\n ) {\n this[kBuffers].push(chunk);\n return;\n }\n\n this[kError] = new RangeError('Max payload size exceeded');\n this[kError].code = 'WS_ERR_UNSUPPORTED_MESSAGE_LENGTH';\n this[kError][kStatusCode] = 1009;\n this.removeListener('data', inflateOnData);\n\n //\n // The choice to employ `zlib.reset()` over `zlib.close()` is dictated by the\n // fact that in Node.js versions prior to 13.10.0, the callback for\n // `zlib.flush()` is not called if `zlib.close()` is used. Utilizing\n // `zlib.reset()` ensures that either the callback is invoked or an error is\n // emitted.\n //\n this.reset();\n}\n\n/**\n * The listener of the `zlib.InflateRaw` stream `'error'` event.\n *\n * @param {Error} err The emitted error\n * @private\n */\nfunction inflateOnError(err) {\n //\n // There is no need to call `Zlib#close()` as the handle is automatically\n // closed when an error is emitted.\n //\n this[kPerMessageDeflate]._inflate = null;\n\n if (this[kError]) {\n this[kCallback](this[kError]);\n return;\n }\n\n err[kStatusCode] = 1007;\n this[kCallback](err);\n}\n","'use strict';\n\nconst { isUtf8 } = require('buffer');\n\nconst { hasBlob } = require('./constants');\n\n//\n// Allowed token characters:\n//\n// '!', '#', '$', '%', '&', ''', '*', '+', '-',\n// '.', 0-9, A-Z, '^', '_', '`', a-z, '|', '~'\n//\n// tokenChars[32] === 0 // ' '\n// tokenChars[33] === 1 // '!'\n// tokenChars[34] === 0 // '\"'\n// ...\n//\n// prettier-ignore\nconst tokenChars = [\n 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 0 - 15\n 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 16 - 31\n 0, 1, 0, 1, 1, 1, 1, 1, 0, 0, 1, 1, 0, 1, 1, 0, // 32 - 47\n 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, // 48 - 63\n 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 64 - 79\n 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 1, 1, // 80 - 95\n 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 96 - 111\n 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 0, 1, 0 // 112 - 127\n];\n\n/**\n * Checks if a status code is allowed in a close frame.\n *\n * @param {Number} code The status code\n * @return {Boolean} `true` if the status code is valid, else `false`\n * @public\n */\nfunction isValidStatusCode(code) {\n return (\n (code >= 1000 &&\n code <= 1014 &&\n code !== 1004 &&\n code !== 1005 &&\n code !== 1006) ||\n (code >= 3000 && code <= 4999)\n );\n}\n\n/**\n * Checks if a given buffer contains only correct UTF-8.\n * Ported from https://www.cl.cam.ac.uk/%7Emgk25/ucs/utf8_check.c by\n * Markus Kuhn.\n *\n * @param {Buffer} buf The buffer to check\n * @return {Boolean} `true` if `buf` contains only correct UTF-8, else `false`\n * @public\n */\nfunction _isValidUTF8(buf) {\n const len = buf.length;\n let i = 0;\n\n while (i < len) {\n if ((buf[i] & 0x80) === 0) {\n // 0xxxxxxx\n i++;\n } else if ((buf[i] & 0xe0) === 0xc0) {\n // 110xxxxx 10xxxxxx\n if (\n i + 1 === len ||\n (buf[i + 1] & 0xc0) !== 0x80 ||\n (buf[i] & 0xfe) === 0xc0 // Overlong\n ) {\n return false;\n }\n\n i += 2;\n } else if ((buf[i] & 0xf0) === 0xe0) {\n // 1110xxxx 10xxxxxx 10xxxxxx\n if (\n i + 2 >= len ||\n (buf[i + 1] & 0xc0) !== 0x80 ||\n (buf[i + 2] & 0xc0) !== 0x80 ||\n (buf[i] === 0xe0 && (buf[i + 1] & 0xe0) === 0x80) || // Overlong\n (buf[i] === 0xed && (buf[i + 1] & 0xe0) === 0xa0) // Surrogate (U+D800 - U+DFFF)\n ) {\n return false;\n }\n\n i += 3;\n } else if ((buf[i] & 0xf8) === 0xf0) {\n // 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx\n if (\n i + 3 >= len ||\n (buf[i + 1] & 0xc0) !== 0x80 ||\n (buf[i + 2] & 0xc0) !== 0x80 ||\n (buf[i + 3] & 0xc0) !== 0x80 ||\n (buf[i] === 0xf0 && (buf[i + 1] & 0xf0) === 0x80) || // Overlong\n (buf[i] === 0xf4 && buf[i + 1] > 0x8f) ||\n buf[i] > 0xf4 // > U+10FFFF\n ) {\n return false;\n }\n\n i += 4;\n } else {\n return false;\n }\n }\n\n return true;\n}\n\n/**\n * Determines whether a value is a `Blob`.\n *\n * @param {*} value The value to be tested\n * @return {Boolean} `true` if `value` is a `Blob`, else `false`\n * @private\n */\nfunction isBlob(value) {\n return (\n hasBlob &&\n typeof value === 'object' &&\n typeof value.arrayBuffer === 'function' &&\n typeof value.type === 'string' &&\n typeof value.stream === 'function' &&\n (value[Symbol.toStringTag] === 'Blob' ||\n value[Symbol.toStringTag] === 'File')\n );\n}\n\nmodule.exports = {\n isBlob,\n isValidStatusCode,\n isValidUTF8: _isValidUTF8,\n tokenChars\n};\n\nif (isUtf8) {\n module.exports.isValidUTF8 = function (buf) {\n return buf.length < 24 ? _isValidUTF8(buf) : isUtf8(buf);\n };\n} /* istanbul ignore else */ else if (!process.env.WS_NO_UTF_8_VALIDATE) {\n try {\n const isValidUTF8 = require('utf-8-validate');\n\n module.exports.isValidUTF8 = function (buf) {\n return buf.length < 32 ? _isValidUTF8(buf) : isValidUTF8(buf);\n };\n } catch (e) {\n // Continue regardless of the error.\n }\n}\n","'use strict';\n\nconst { Writable } = require('stream');\n\nconst PerMessageDeflate = require('./permessage-deflate');\nconst {\n BINARY_TYPES,\n EMPTY_BUFFER,\n kStatusCode,\n kWebSocket\n} = require('./constants');\nconst { concat, toArrayBuffer, unmask } = require('./buffer-util');\nconst { isValidStatusCode, isValidUTF8 } = require('./validation');\n\nconst FastBuffer = Buffer[Symbol.species];\n\nconst GET_INFO = 0;\nconst GET_PAYLOAD_LENGTH_16 = 1;\nconst GET_PAYLOAD_LENGTH_64 = 2;\nconst GET_MASK = 3;\nconst GET_DATA = 4;\nconst INFLATING = 5;\nconst DEFER_EVENT = 6;\n\n/**\n * HyBi Receiver implementation.\n *\n * @extends Writable\n */\nclass Receiver extends Writable {\n /**\n * Creates a Receiver instance.\n *\n * @param {Object} [options] Options object\n * @param {Boolean} [options.allowSynchronousEvents=true] Specifies whether\n * any of the `'message'`, `'ping'`, and `'pong'` events can be emitted\n * multiple times in the same tick\n * @param {String} [options.binaryType=nodebuffer] The type for binary data\n * @param {Object} [options.extensions] An object containing the negotiated\n * extensions\n * @param {Boolean} [options.isServer=false] Specifies whether to operate in\n * client or server mode\n * @param {Number} [options.maxPayload=0] The maximum allowed message length\n * @param {Boolean} [options.skipUTF8Validation=false] Specifies whether or\n * not to skip UTF-8 validation for text and close messages\n */\n constructor(options = {}) {\n super();\n\n this._allowSynchronousEvents =\n options.allowSynchronousEvents !== undefined\n ? options.allowSynchronousEvents\n : true;\n this._binaryType = options.binaryType || BINARY_TYPES[0];\n this._extensions = options.extensions || {};\n this._isServer = !!options.isServer;\n this._maxPayload = options.maxPayload | 0;\n this._skipUTF8Validation = !!options.skipUTF8Validation;\n this[kWebSocket] = undefined;\n\n this._bufferedBytes = 0;\n this._buffers = [];\n\n this._compressed = false;\n this._payloadLength = 0;\n this._mask = undefined;\n this._fragmented = 0;\n this._masked = false;\n this._fin = false;\n this._opcode = 0;\n\n this._totalPayloadLength = 0;\n this._messageLength = 0;\n this._fragments = [];\n\n this._errored = false;\n this._loop = false;\n this._state = GET_INFO;\n }\n\n /**\n * Implements `Writable.prototype._write()`.\n *\n * @param {Buffer} chunk The chunk of data to write\n * @param {String} encoding The character encoding of `chunk`\n * @param {Function} cb Callback\n * @private\n */\n _write(chunk, encoding, cb) {\n if (this._opcode === 0x08 && this._state == GET_INFO) return cb();\n\n this._bufferedBytes += chunk.length;\n this._buffers.push(chunk);\n this.startLoop(cb);\n }\n\n /**\n * Consumes `n` bytes from the buffered data.\n *\n * @param {Number} n The number of bytes to consume\n * @return {Buffer} The consumed bytes\n * @private\n */\n consume(n) {\n this._bufferedBytes -= n;\n\n if (n === this._buffers[0].length) return this._buffers.shift();\n\n if (n < this._buffers[0].length) {\n const buf = this._buffers[0];\n this._buffers[0] = new FastBuffer(\n buf.buffer,\n buf.byteOffset + n,\n buf.length - n\n );\n\n return new FastBuffer(buf.buffer, buf.byteOffset, n);\n }\n\n const dst = Buffer.allocUnsafe(n);\n\n do {\n const buf = this._buffers[0];\n const offset = dst.length - n;\n\n if (n >= buf.length) {\n dst.set(this._buffers.shift(), offset);\n } else {\n dst.set(new Uint8Array(buf.buffer, buf.byteOffset, n), offset);\n this._buffers[0] = new FastBuffer(\n buf.buffer,\n buf.byteOffset + n,\n buf.length - n\n );\n }\n\n n -= buf.length;\n } while (n > 0);\n\n return dst;\n }\n\n /**\n * Starts the parsing loop.\n *\n * @param {Function} cb Callback\n * @private\n */\n startLoop(cb) {\n this._loop = true;\n\n do {\n switch (this._state) {\n case GET_INFO:\n this.getInfo(cb);\n break;\n case GET_PAYLOAD_LENGTH_16:\n this.getPayloadLength16(cb);\n break;\n case GET_PAYLOAD_LENGTH_64:\n this.getPayloadLength64(cb);\n break;\n case GET_MASK:\n this.getMask();\n break;\n case GET_DATA:\n this.getData(cb);\n break;\n case INFLATING:\n case DEFER_EVENT:\n this._loop = false;\n return;\n }\n } while (this._loop);\n\n if (!this._errored) cb();\n }\n\n /**\n * Reads the first two bytes of a frame.\n *\n * @param {Function} cb Callback\n * @private\n */\n getInfo(cb) {\n if (this._bufferedBytes < 2) {\n this._loop = false;\n return;\n }\n\n const buf = this.consume(2);\n\n if ((buf[0] & 0x30) !== 0x00) {\n const error = this.createError(\n RangeError,\n 'RSV2 and RSV3 must be clear',\n true,\n 1002,\n 'WS_ERR_UNEXPECTED_RSV_2_3'\n );\n\n cb(error);\n return;\n }\n\n const compressed = (buf[0] & 0x40) === 0x40;\n\n if (compressed && !this._extensions[PerMessageDeflate.extensionName]) {\n const error = this.createError(\n RangeError,\n 'RSV1 must be clear',\n true,\n 1002,\n 'WS_ERR_UNEXPECTED_RSV_1'\n );\n\n cb(error);\n return;\n }\n\n this._fin = (buf[0] & 0x80) === 0x80;\n this._opcode = buf[0] & 0x0f;\n this._payloadLength = buf[1] & 0x7f;\n\n if (this._opcode === 0x00) {\n if (compressed) {\n const error = this.createError(\n RangeError,\n 'RSV1 must be clear',\n true,\n 1002,\n 'WS_ERR_UNEXPECTED_RSV_1'\n );\n\n cb(error);\n return;\n }\n\n if (!this._fragmented) {\n const error = this.createError(\n RangeError,\n 'invalid opcode 0',\n true,\n 1002,\n 'WS_ERR_INVALID_OPCODE'\n );\n\n cb(error);\n return;\n }\n\n this._opcode = this._fragmented;\n } else if (this._opcode === 0x01 || this._opcode === 0x02) {\n if (this._fragmented) {\n const error = this.createError(\n RangeError,\n `invalid opcode ${this._opcode}`,\n true,\n 1002,\n 'WS_ERR_INVALID_OPCODE'\n );\n\n cb(error);\n return;\n }\n\n this._compressed = compressed;\n } else if (this._opcode > 0x07 && this._opcode < 0x0b) {\n if (!this._fin) {\n const error = this.createError(\n RangeError,\n 'FIN must be set',\n true,\n 1002,\n 'WS_ERR_EXPECTED_FIN'\n );\n\n cb(error);\n return;\n }\n\n if (compressed) {\n const error = this.createError(\n RangeError,\n 'RSV1 must be clear',\n true,\n 1002,\n 'WS_ERR_UNEXPECTED_RSV_1'\n );\n\n cb(error);\n return;\n }\n\n if (\n this._payloadLength > 0x7d ||\n (this._opcode === 0x08 && this._payloadLength === 1)\n ) {\n const error = this.createError(\n RangeError,\n `invalid payload length ${this._payloadLength}`,\n true,\n 1002,\n 'WS_ERR_INVALID_CONTROL_PAYLOAD_LENGTH'\n );\n\n cb(error);\n return;\n }\n } else {\n const error = this.createError(\n RangeError,\n `invalid opcode ${this._opcode}`,\n true,\n 1002,\n 'WS_ERR_INVALID_OPCODE'\n );\n\n cb(error);\n return;\n }\n\n if (!this._fin && !this._fragmented) this._fragmented = this._opcode;\n this._masked = (buf[1] & 0x80) === 0x80;\n\n if (this._isServer) {\n if (!this._masked) {\n const error = this.createError(\n RangeError,\n 'MASK must be set',\n true,\n 1002,\n 'WS_ERR_EXPECTED_MASK'\n );\n\n cb(error);\n return;\n }\n } else if (this._masked) {\n const error = this.createError(\n RangeError,\n 'MASK must be clear',\n true,\n 1002,\n 'WS_ERR_UNEXPECTED_MASK'\n );\n\n cb(error);\n return;\n }\n\n if (this._payloadLength === 126) this._state = GET_PAYLOAD_LENGTH_16;\n else if (this._payloadLength === 127) this._state = GET_PAYLOAD_LENGTH_64;\n else this.haveLength(cb);\n }\n\n /**\n * Gets extended payload length (7+16).\n *\n * @param {Function} cb Callback\n * @private\n */\n getPayloadLength16(cb) {\n if (this._bufferedBytes < 2) {\n this._loop = false;\n return;\n }\n\n this._payloadLength = this.consume(2).readUInt16BE(0);\n this.haveLength(cb);\n }\n\n /**\n * Gets extended payload length (7+64).\n *\n * @param {Function} cb Callback\n * @private\n */\n getPayloadLength64(cb) {\n if (this._bufferedBytes < 8) {\n this._loop = false;\n return;\n }\n\n const buf = this.consume(8);\n const num = buf.readUInt32BE(0);\n\n //\n // The maximum safe integer in JavaScript is 2^53 - 1. An error is returned\n // if payload length is greater than this number.\n //\n if (num > Math.pow(2, 53 - 32) - 1) {\n const error = this.createError(\n RangeError,\n 'Unsupported WebSocket frame: payload length > 2^53 - 1',\n false,\n 1009,\n 'WS_ERR_UNSUPPORTED_DATA_PAYLOAD_LENGTH'\n );\n\n cb(error);\n return;\n }\n\n this._payloadLength = num * Math.pow(2, 32) + buf.readUInt32BE(4);\n this.haveLength(cb);\n }\n\n /**\n * Payload length has been read.\n *\n * @param {Function} cb Callback\n * @private\n */\n haveLength(cb) {\n if (this._payloadLength && this._opcode < 0x08) {\n this._totalPayloadLength += this._payloadLength;\n if (this._totalPayloadLength > this._maxPayload && this._maxPayload > 0) {\n const error = this.createError(\n RangeError,\n 'Max payload size exceeded',\n false,\n 1009,\n 'WS_ERR_UNSUPPORTED_MESSAGE_LENGTH'\n );\n\n cb(error);\n return;\n }\n }\n\n if (this._masked) this._state = GET_MASK;\n else this._state = GET_DATA;\n }\n\n /**\n * Reads mask bytes.\n *\n * @private\n */\n getMask() {\n if (this._bufferedBytes < 4) {\n this._loop = false;\n return;\n }\n\n this._mask = this.consume(4);\n this._state = GET_DATA;\n }\n\n /**\n * Reads data bytes.\n *\n * @param {Function} cb Callback\n * @private\n */\n getData(cb) {\n let data = EMPTY_BUFFER;\n\n if (this._payloadLength) {\n if (this._bufferedBytes < this._payloadLength) {\n this._loop = false;\n return;\n }\n\n data = this.consume(this._payloadLength);\n\n if (\n this._masked &&\n (this._mask[0] | this._mask[1] | this._mask[2] | this._mask[3]) !== 0\n ) {\n unmask(data, this._mask);\n }\n }\n\n if (this._opcode > 0x07) {\n this.controlMessage(data, cb);\n return;\n }\n\n if (this._compressed) {\n this._state = INFLATING;\n this.decompress(data, cb);\n return;\n }\n\n if (data.length) {\n //\n // This message is not compressed so its length is the sum of the payload\n // length of all fragments.\n //\n this._messageLength = this._totalPayloadLength;\n this._fragments.push(data);\n }\n\n this.dataMessage(cb);\n }\n\n /**\n * Decompresses data.\n *\n * @param {Buffer} data Compressed data\n * @param {Function} cb Callback\n * @private\n */\n decompress(data, cb) {\n const perMessageDeflate = this._extensions[PerMessageDeflate.extensionName];\n\n perMessageDeflate.decompress(data, this._fin, (err, buf) => {\n if (err) return cb(err);\n\n if (buf.length) {\n this._messageLength += buf.length;\n if (this._messageLength > this._maxPayload && this._maxPayload > 0) {\n const error = this.createError(\n RangeError,\n 'Max payload size exceeded',\n false,\n 1009,\n 'WS_ERR_UNSUPPORTED_MESSAGE_LENGTH'\n );\n\n cb(error);\n return;\n }\n\n this._fragments.push(buf);\n }\n\n this.dataMessage(cb);\n if (this._state === GET_INFO) this.startLoop(cb);\n });\n }\n\n /**\n * Handles a data message.\n *\n * @param {Function} cb Callback\n * @private\n */\n dataMessage(cb) {\n if (!this._fin) {\n this._state = GET_INFO;\n return;\n }\n\n const messageLength = this._messageLength;\n const fragments = this._fragments;\n\n this._totalPayloadLength = 0;\n this._messageLength = 0;\n this._fragmented = 0;\n this._fragments = [];\n\n if (this._opcode === 2) {\n let data;\n\n if (this._binaryType === 'nodebuffer') {\n data = concat(fragments, messageLength);\n } else if (this._binaryType === 'arraybuffer') {\n data = toArrayBuffer(concat(fragments, messageLength));\n } else if (this._binaryType === 'blob') {\n data = new Blob(fragments);\n } else {\n data = fragments;\n }\n\n if (this._allowSynchronousEvents) {\n this.emit('message', data, true);\n this._state = GET_INFO;\n } else {\n this._state = DEFER_EVENT;\n setImmediate(() => {\n this.emit('message', data, true);\n this._state = GET_INFO;\n this.startLoop(cb);\n });\n }\n } else {\n const buf = concat(fragments, messageLength);\n\n if (!this._skipUTF8Validation && !isValidUTF8(buf)) {\n const error = this.createError(\n Error,\n 'invalid UTF-8 sequence',\n true,\n 1007,\n 'WS_ERR_INVALID_UTF8'\n );\n\n cb(error);\n return;\n }\n\n if (this._state === INFLATING || this._allowSynchronousEvents) {\n this.emit('message', buf, false);\n this._state = GET_INFO;\n } else {\n this._state = DEFER_EVENT;\n setImmediate(() => {\n this.emit('message', buf, false);\n this._state = GET_INFO;\n this.startLoop(cb);\n });\n }\n }\n }\n\n /**\n * Handles a control message.\n *\n * @param {Buffer} data Data to handle\n * @return {(Error|RangeError|undefined)} A possible error\n * @private\n */\n controlMessage(data, cb) {\n if (this._opcode === 0x08) {\n if (data.length === 0) {\n this._loop = false;\n this.emit('conclude', 1005, EMPTY_BUFFER);\n this.end();\n } else {\n const code = data.readUInt16BE(0);\n\n if (!isValidStatusCode(code)) {\n const error = this.createError(\n RangeError,\n `invalid status code ${code}`,\n true,\n 1002,\n 'WS_ERR_INVALID_CLOSE_CODE'\n );\n\n cb(error);\n return;\n }\n\n const buf = new FastBuffer(\n data.buffer,\n data.byteOffset + 2,\n data.length - 2\n );\n\n if (!this._skipUTF8Validation && !isValidUTF8(buf)) {\n const error = this.createError(\n Error,\n 'invalid UTF-8 sequence',\n true,\n 1007,\n 'WS_ERR_INVALID_UTF8'\n );\n\n cb(error);\n return;\n }\n\n this._loop = false;\n this.emit('conclude', code, buf);\n this.end();\n }\n\n this._state = GET_INFO;\n return;\n }\n\n if (this._allowSynchronousEvents) {\n this.emit(this._opcode === 0x09 ? 'ping' : 'pong', data);\n this._state = GET_INFO;\n } else {\n this._state = DEFER_EVENT;\n setImmediate(() => {\n this.emit(this._opcode === 0x09 ? 'ping' : 'pong', data);\n this._state = GET_INFO;\n this.startLoop(cb);\n });\n }\n }\n\n /**\n * Builds an error object.\n *\n * @param {function(new:Error|RangeError)} ErrorCtor The error constructor\n * @param {String} message The error message\n * @param {Boolean} prefix Specifies whether or not to add a default prefix to\n * `message`\n * @param {Number} statusCode The status code\n * @param {String} errorCode The exposed error code\n * @return {(Error|RangeError)} The error\n * @private\n */\n createError(ErrorCtor, message, prefix, statusCode, errorCode) {\n this._loop = false;\n this._errored = true;\n\n const err = new ErrorCtor(\n prefix ? `Invalid WebSocket frame: ${message}` : message\n );\n\n Error.captureStackTrace(err, this.createError);\n err.code = errorCode;\n err[kStatusCode] = statusCode;\n return err;\n }\n}\n\nmodule.exports = Receiver;\n","/* eslint no-unused-vars: [\"error\", { \"varsIgnorePattern\": \"^Duplex\" }] */\n\n'use strict';\n\nconst { Duplex } = require('stream');\nconst { randomFillSync } = require('crypto');\n\nconst PerMessageDeflate = require('./permessage-deflate');\nconst { EMPTY_BUFFER, kWebSocket, NOOP } = require('./constants');\nconst { isBlob, isValidStatusCode } = require('./validation');\nconst { mask: applyMask, toBuffer } = require('./buffer-util');\n\nconst kByteLength = Symbol('kByteLength');\nconst maskBuffer = Buffer.alloc(4);\nconst RANDOM_POOL_SIZE = 8 * 1024;\nlet randomPool;\nlet randomPoolPointer = RANDOM_POOL_SIZE;\n\nconst DEFAULT = 0;\nconst DEFLATING = 1;\nconst GET_BLOB_DATA = 2;\n\n/**\n * HyBi Sender implementation.\n */\nclass Sender {\n /**\n * Creates a Sender instance.\n *\n * @param {Duplex} socket The connection socket\n * @param {Object} [extensions] An object containing the negotiated extensions\n * @param {Function} [generateMask] The function used to generate the masking\n * key\n */\n constructor(socket, extensions, generateMask) {\n this._extensions = extensions || {};\n\n if (generateMask) {\n this._generateMask = generateMask;\n this._maskBuffer = Buffer.alloc(4);\n }\n\n this._socket = socket;\n\n this._firstFragment = true;\n this._compress = false;\n\n this._bufferedBytes = 0;\n this._queue = [];\n this._state = DEFAULT;\n this.onerror = NOOP;\n this[kWebSocket] = undefined;\n }\n\n /**\n * Frames a piece of data according to the HyBi WebSocket protocol.\n *\n * @param {(Buffer|String)} data The data to frame\n * @param {Object} options Options object\n * @param {Boolean} [options.fin=false] Specifies whether or not to set the\n * FIN bit\n * @param {Function} [options.generateMask] The function used to generate the\n * masking key\n * @param {Boolean} [options.mask=false] Specifies whether or not to mask\n * `data`\n * @param {Buffer} [options.maskBuffer] The buffer used to store the masking\n * key\n * @param {Number} options.opcode The opcode\n * @param {Boolean} [options.readOnly=false] Specifies whether `data` can be\n * modified\n * @param {Boolean} [options.rsv1=false] Specifies whether or not to set the\n * RSV1 bit\n * @return {(Buffer|String)[]} The framed data\n * @public\n */\n static frame(data, options) {\n let mask;\n let merge = false;\n let offset = 2;\n let skipMasking = false;\n\n if (options.mask) {\n mask = options.maskBuffer || maskBuffer;\n\n if (options.generateMask) {\n options.generateMask(mask);\n } else {\n if (randomPoolPointer === RANDOM_POOL_SIZE) {\n /* istanbul ignore else */\n if (randomPool === undefined) {\n //\n // This is lazily initialized because server-sent frames must not\n // be masked so it may never be used.\n //\n randomPool = Buffer.alloc(RANDOM_POOL_SIZE);\n }\n\n randomFillSync(randomPool, 0, RANDOM_POOL_SIZE);\n randomPoolPointer = 0;\n }\n\n mask[0] = randomPool[randomPoolPointer++];\n mask[1] = randomPool[randomPoolPointer++];\n mask[2] = randomPool[randomPoolPointer++];\n mask[3] = randomPool[randomPoolPointer++];\n }\n\n skipMasking = (mask[0] | mask[1] | mask[2] | mask[3]) === 0;\n offset = 6;\n }\n\n let dataLength;\n\n if (typeof data === 'string') {\n if (\n (!options.mask || skipMasking) &&\n options[kByteLength] !== undefined\n ) {\n dataLength = options[kByteLength];\n } else {\n data = Buffer.from(data);\n dataLength = data.length;\n }\n } else {\n dataLength = data.length;\n merge = options.mask && options.readOnly && !skipMasking;\n }\n\n let payloadLength = dataLength;\n\n if (dataLength >= 65536) {\n offset += 8;\n payloadLength = 127;\n } else if (dataLength > 125) {\n offset += 2;\n payloadLength = 126;\n }\n\n const target = Buffer.allocUnsafe(merge ? dataLength + offset : offset);\n\n target[0] = options.fin ? options.opcode | 0x80 : options.opcode;\n if (options.rsv1) target[0] |= 0x40;\n\n target[1] = payloadLength;\n\n if (payloadLength === 126) {\n target.writeUInt16BE(dataLength, 2);\n } else if (payloadLength === 127) {\n target[2] = target[3] = 0;\n target.writeUIntBE(dataLength, 4, 6);\n }\n\n if (!options.mask) return [target, data];\n\n target[1] |= 0x80;\n target[offset - 4] = mask[0];\n target[offset - 3] = mask[1];\n target[offset - 2] = mask[2];\n target[offset - 1] = mask[3];\n\n if (skipMasking) return [target, data];\n\n if (merge) {\n applyMask(data, mask, target, offset, dataLength);\n return [target];\n }\n\n applyMask(data, mask, data, 0, dataLength);\n return [target, data];\n }\n\n /**\n * Sends a close message to the other peer.\n *\n * @param {Number} [code] The status code component of the body\n * @param {(String|Buffer)} [data] The message component of the body\n * @param {Boolean} [mask=false] Specifies whether or not to mask the message\n * @param {Function} [cb] Callback\n * @public\n */\n close(code, data, mask, cb) {\n let buf;\n\n if (code === undefined) {\n buf = EMPTY_BUFFER;\n } else if (typeof code !== 'number' || !isValidStatusCode(code)) {\n throw new TypeError('First argument must be a valid error code number');\n } else if (data === undefined || !data.length) {\n buf = Buffer.allocUnsafe(2);\n buf.writeUInt16BE(code, 0);\n } else {\n const length = Buffer.byteLength(data);\n\n if (length > 123) {\n throw new RangeError('The message must not be greater than 123 bytes');\n }\n\n buf = Buffer.allocUnsafe(2 + length);\n buf.writeUInt16BE(code, 0);\n\n if (typeof data === 'string') {\n buf.write(data, 2);\n } else {\n buf.set(data, 2);\n }\n }\n\n const options = {\n [kByteLength]: buf.length,\n fin: true,\n generateMask: this._generateMask,\n mask,\n maskBuffer: this._maskBuffer,\n opcode: 0x08,\n readOnly: false,\n rsv1: false\n };\n\n if (this._state !== DEFAULT) {\n this.enqueue([this.dispatch, buf, false, options, cb]);\n } else {\n this.sendFrame(Sender.frame(buf, options), cb);\n }\n }\n\n /**\n * Sends a ping message to the other peer.\n *\n * @param {*} data The message to send\n * @param {Boolean} [mask=false] Specifies whether or not to mask `data`\n * @param {Function} [cb] Callback\n * @public\n */\n ping(data, mask, cb) {\n let byteLength;\n let readOnly;\n\n if (typeof data === 'string') {\n byteLength = Buffer.byteLength(data);\n readOnly = false;\n } else if (isBlob(data)) {\n byteLength = data.size;\n readOnly = false;\n } else {\n data = toBuffer(data);\n byteLength = data.length;\n readOnly = toBuffer.readOnly;\n }\n\n if (byteLength > 125) {\n throw new RangeError('The data size must not be greater than 125 bytes');\n }\n\n const options = {\n [kByteLength]: byteLength,\n fin: true,\n generateMask: this._generateMask,\n mask,\n maskBuffer: this._maskBuffer,\n opcode: 0x09,\n readOnly,\n rsv1: false\n };\n\n if (isBlob(data)) {\n if (this._state !== DEFAULT) {\n this.enqueue([this.getBlobData, data, false, options, cb]);\n } else {\n this.getBlobData(data, false, options, cb);\n }\n } else if (this._state !== DEFAULT) {\n this.enqueue([this.dispatch, data, false, options, cb]);\n } else {\n this.sendFrame(Sender.frame(data, options), cb);\n }\n }\n\n /**\n * Sends a pong message to the other peer.\n *\n * @param {*} data The message to send\n * @param {Boolean} [mask=false] Specifies whether or not to mask `data`\n * @param {Function} [cb] Callback\n * @public\n */\n pong(data, mask, cb) {\n let byteLength;\n let readOnly;\n\n if (typeof data === 'string') {\n byteLength = Buffer.byteLength(data);\n readOnly = false;\n } else if (isBlob(data)) {\n byteLength = data.size;\n readOnly = false;\n } else {\n data = toBuffer(data);\n byteLength = data.length;\n readOnly = toBuffer.readOnly;\n }\n\n if (byteLength > 125) {\n throw new RangeError('The data size must not be greater than 125 bytes');\n }\n\n const options = {\n [kByteLength]: byteLength,\n fin: true,\n generateMask: this._generateMask,\n mask,\n maskBuffer: this._maskBuffer,\n opcode: 0x0a,\n readOnly,\n rsv1: false\n };\n\n if (isBlob(data)) {\n if (this._state !== DEFAULT) {\n this.enqueue([this.getBlobData, data, false, options, cb]);\n } else {\n this.getBlobData(data, false, options, cb);\n }\n } else if (this._state !== DEFAULT) {\n this.enqueue([this.dispatch, data, false, options, cb]);\n } else {\n this.sendFrame(Sender.frame(data, options), cb);\n }\n }\n\n /**\n * Sends a data message to the other peer.\n *\n * @param {*} data The message to send\n * @param {Object} options Options object\n * @param {Boolean} [options.binary=false] Specifies whether `data` is binary\n * or text\n * @param {Boolean} [options.compress=false] Specifies whether or not to\n * compress `data`\n * @param {Boolean} [options.fin=false] Specifies whether the fragment is the\n * last one\n * @param {Boolean} [options.mask=false] Specifies whether or not to mask\n * `data`\n * @param {Function} [cb] Callback\n * @public\n */\n send(data, options, cb) {\n const perMessageDeflate = this._extensions[PerMessageDeflate.extensionName];\n let opcode = options.binary ? 2 : 1;\n let rsv1 = options.compress;\n\n let byteLength;\n let readOnly;\n\n if (typeof data === 'string') {\n byteLength = Buffer.byteLength(data);\n readOnly = false;\n } else if (isBlob(data)) {\n byteLength = data.size;\n readOnly = false;\n } else {\n data = toBuffer(data);\n byteLength = data.length;\n readOnly = toBuffer.readOnly;\n }\n\n if (this._firstFragment) {\n this._firstFragment = false;\n if (\n rsv1 &&\n perMessageDeflate &&\n perMessageDeflate.params[\n perMessageDeflate._isServer\n ? 'server_no_context_takeover'\n : 'client_no_context_takeover'\n ]\n ) {\n rsv1 = byteLength >= perMessageDeflate._threshold;\n }\n this._compress = rsv1;\n } else {\n rsv1 = false;\n opcode = 0;\n }\n\n if (options.fin) this._firstFragment = true;\n\n const opts = {\n [kByteLength]: byteLength,\n fin: options.fin,\n generateMask: this._generateMask,\n mask: options.mask,\n maskBuffer: this._maskBuffer,\n opcode,\n readOnly,\n rsv1\n };\n\n if (isBlob(data)) {\n if (this._state !== DEFAULT) {\n this.enqueue([this.getBlobData, data, this._compress, opts, cb]);\n } else {\n this.getBlobData(data, this._compress, opts, cb);\n }\n } else if (this._state !== DEFAULT) {\n this.enqueue([this.dispatch, data, this._compress, opts, cb]);\n } else {\n this.dispatch(data, this._compress, opts, cb);\n }\n }\n\n /**\n * Gets the contents of a blob as binary data.\n *\n * @param {Blob} blob The blob\n * @param {Boolean} [compress=false] Specifies whether or not to compress\n * the data\n * @param {Object} options Options object\n * @param {Boolean} [options.fin=false] Specifies whether or not to set the\n * FIN bit\n * @param {Function} [options.generateMask] The function used to generate the\n * masking key\n * @param {Boolean} [options.mask=false] Specifies whether or not to mask\n * `data`\n * @param {Buffer} [options.maskBuffer] The buffer used to store the masking\n * key\n * @param {Number} options.opcode The opcode\n * @param {Boolean} [options.readOnly=false] Specifies whether `data` can be\n * modified\n * @param {Boolean} [options.rsv1=false] Specifies whether or not to set the\n * RSV1 bit\n * @param {Function} [cb] Callback\n * @private\n */\n getBlobData(blob, compress, options, cb) {\n this._bufferedBytes += options[kByteLength];\n this._state = GET_BLOB_DATA;\n\n blob\n .arrayBuffer()\n .then((arrayBuffer) => {\n if (this._socket.destroyed) {\n const err = new Error(\n 'The socket was closed while the blob was being read'\n );\n\n //\n // `callCallbacks` is called in the next tick to ensure that errors\n // that might be thrown in the callbacks behave like errors thrown\n // outside the promise chain.\n //\n process.nextTick(callCallbacks, this, err, cb);\n return;\n }\n\n this._bufferedBytes -= options[kByteLength];\n const data = toBuffer(arrayBuffer);\n\n if (!compress) {\n this._state = DEFAULT;\n this.sendFrame(Sender.frame(data, options), cb);\n this.dequeue();\n } else {\n this.dispatch(data, compress, options, cb);\n }\n })\n .catch((err) => {\n //\n // `onError` is called in the next tick for the same reason that\n // `callCallbacks` above is.\n //\n process.nextTick(onError, this, err, cb);\n });\n }\n\n /**\n * Dispatches a message.\n *\n * @param {(Buffer|String)} data The message to send\n * @param {Boolean} [compress=false] Specifies whether or not to compress\n * `data`\n * @param {Object} options Options object\n * @param {Boolean} [options.fin=false] Specifies whether or not to set the\n * FIN bit\n * @param {Function} [options.generateMask] The function used to generate the\n * masking key\n * @param {Boolean} [options.mask=false] Specifies whether or not to mask\n * `data`\n * @param {Buffer} [options.maskBuffer] The buffer used to store the masking\n * key\n * @param {Number} options.opcode The opcode\n * @param {Boolean} [options.readOnly=false] Specifies whether `data` can be\n * modified\n * @param {Boolean} [options.rsv1=false] Specifies whether or not to set the\n * RSV1 bit\n * @param {Function} [cb] Callback\n * @private\n */\n dispatch(data, compress, options, cb) {\n if (!compress) {\n this.sendFrame(Sender.frame(data, options), cb);\n return;\n }\n\n const perMessageDeflate = this._extensions[PerMessageDeflate.extensionName];\n\n this._bufferedBytes += options[kByteLength];\n this._state = DEFLATING;\n perMessageDeflate.compress(data, options.fin, (_, buf) => {\n if (this._socket.destroyed) {\n const err = new Error(\n 'The socket was closed while data was being compressed'\n );\n\n callCallbacks(this, err, cb);\n return;\n }\n\n this._bufferedBytes -= options[kByteLength];\n this._state = DEFAULT;\n options.readOnly = false;\n this.sendFrame(Sender.frame(buf, options), cb);\n this.dequeue();\n });\n }\n\n /**\n * Executes queued send operations.\n *\n * @private\n */\n dequeue() {\n while (this._state === DEFAULT && this._queue.length) {\n const params = this._queue.shift();\n\n this._bufferedBytes -= params[3][kByteLength];\n Reflect.apply(params[0], this, params.slice(1));\n }\n }\n\n /**\n * Enqueues a send operation.\n *\n * @param {Array} params Send operation parameters.\n * @private\n */\n enqueue(params) {\n this._bufferedBytes += params[3][kByteLength];\n this._queue.push(params);\n }\n\n /**\n * Sends a frame.\n *\n * @param {(Buffer | String)[]} list The frame to send\n * @param {Function} [cb] Callback\n * @private\n */\n sendFrame(list, cb) {\n if (list.length === 2) {\n this._socket.cork();\n this._socket.write(list[0]);\n this._socket.write(list[1], cb);\n this._socket.uncork();\n } else {\n this._socket.write(list[0], cb);\n }\n }\n}\n\nmodule.exports = Sender;\n\n/**\n * Calls queued callbacks with an error.\n *\n * @param {Sender} sender The `Sender` instance\n * @param {Error} err The error to call the callbacks with\n * @param {Function} [cb] The first callback\n * @private\n */\nfunction callCallbacks(sender, err, cb) {\n if (typeof cb === 'function') cb(err);\n\n for (let i = 0; i < sender._queue.length; i++) {\n const params = sender._queue[i];\n const callback = params[params.length - 1];\n\n if (typeof callback === 'function') callback(err);\n }\n}\n\n/**\n * Handles a `Sender` error.\n *\n * @param {Sender} sender The `Sender` instance\n * @param {Error} err The error\n * @param {Function} [cb] The first pending callback\n * @private\n */\nfunction onError(sender, err, cb) {\n callCallbacks(sender, err, cb);\n sender.onerror(err);\n}\n","'use strict';\n\nconst { kForOnEventAttribute, kListener } = require('./constants');\n\nconst kCode = Symbol('kCode');\nconst kData = Symbol('kData');\nconst kError = Symbol('kError');\nconst kMessage = Symbol('kMessage');\nconst kReason = Symbol('kReason');\nconst kTarget = Symbol('kTarget');\nconst kType = Symbol('kType');\nconst kWasClean = Symbol('kWasClean');\n\n/**\n * Class representing an event.\n */\nclass Event {\n /**\n * Create a new `Event`.\n *\n * @param {String} type The name of the event\n * @throws {TypeError} If the `type` argument is not specified\n */\n constructor(type) {\n this[kTarget] = null;\n this[kType] = type;\n }\n\n /**\n * @type {*}\n */\n get target() {\n return this[kTarget];\n }\n\n /**\n * @type {String}\n */\n get type() {\n return this[kType];\n }\n}\n\nObject.defineProperty(Event.prototype, 'target', { enumerable: true });\nObject.defineProperty(Event.prototype, 'type', { enumerable: true });\n\n/**\n * Class representing a close event.\n *\n * @extends Event\n */\nclass CloseEvent extends Event {\n /**\n * Create a new `CloseEvent`.\n *\n * @param {String} type The name of the event\n * @param {Object} [options] A dictionary object that allows for setting\n * attributes via object members of the same name\n * @param {Number} [options.code=0] The status code explaining why the\n * connection was closed\n * @param {String} [options.reason=''] A human-readable string explaining why\n * the connection was closed\n * @param {Boolean} [options.wasClean=false] Indicates whether or not the\n * connection was cleanly closed\n */\n constructor(type, options = {}) {\n super(type);\n\n this[kCode] = options.code === undefined ? 0 : options.code;\n this[kReason] = options.reason === undefined ? '' : options.reason;\n this[kWasClean] = options.wasClean === undefined ? false : options.wasClean;\n }\n\n /**\n * @type {Number}\n */\n get code() {\n return this[kCode];\n }\n\n /**\n * @type {String}\n */\n get reason() {\n return this[kReason];\n }\n\n /**\n * @type {Boolean}\n */\n get wasClean() {\n return this[kWasClean];\n }\n}\n\nObject.defineProperty(CloseEvent.prototype, 'code', { enumerable: true });\nObject.defineProperty(CloseEvent.prototype, 'reason', { enumerable: true });\nObject.defineProperty(CloseEvent.prototype, 'wasClean', { enumerable: true });\n\n/**\n * Class representing an error event.\n *\n * @extends Event\n */\nclass ErrorEvent extends Event {\n /**\n * Create a new `ErrorEvent`.\n *\n * @param {String} type The name of the event\n * @param {Object} [options] A dictionary object that allows for setting\n * attributes via object members of the same name\n * @param {*} [options.error=null] The error that generated this event\n * @param {String} [options.message=''] The error message\n */\n constructor(type, options = {}) {\n super(type);\n\n this[kError] = options.error === undefined ? null : options.error;\n this[kMessage] = options.message === undefined ? '' : options.message;\n }\n\n /**\n * @type {*}\n */\n get error() {\n return this[kError];\n }\n\n /**\n * @type {String}\n */\n get message() {\n return this[kMessage];\n }\n}\n\nObject.defineProperty(ErrorEvent.prototype, 'error', { enumerable: true });\nObject.defineProperty(ErrorEvent.prototype, 'message', { enumerable: true });\n\n/**\n * Class representing a message event.\n *\n * @extends Event\n */\nclass MessageEvent extends Event {\n /**\n * Create a new `MessageEvent`.\n *\n * @param {String} type The name of the event\n * @param {Object} [options] A dictionary object that allows for setting\n * attributes via object members of the same name\n * @param {*} [options.data=null] The message content\n */\n constructor(type, options = {}) {\n super(type);\n\n this[kData] = options.data === undefined ? null : options.data;\n }\n\n /**\n * @type {*}\n */\n get data() {\n return this[kData];\n }\n}\n\nObject.defineProperty(MessageEvent.prototype, 'data', { enumerable: true });\n\n/**\n * This provides methods for emulating the `EventTarget` interface. It's not\n * meant to be used directly.\n *\n * @mixin\n */\nconst EventTarget = {\n /**\n * Register an event listener.\n *\n * @param {String} type A string representing the event type to listen for\n * @param {(Function|Object)} handler The listener to add\n * @param {Object} [options] An options object specifies characteristics about\n * the event listener\n * @param {Boolean} [options.once=false] A `Boolean` indicating that the\n * listener should be invoked at most once after being added. If `true`,\n * the listener would be automatically removed when invoked.\n * @public\n */\n addEventListener(type, handler, options = {}) {\n for (const listener of this.listeners(type)) {\n if (\n !options[kForOnEventAttribute] &&\n listener[kListener] === handler &&\n !listener[kForOnEventAttribute]\n ) {\n return;\n }\n }\n\n let wrapper;\n\n if (type === 'message') {\n wrapper = function onMessage(data, isBinary) {\n const event = new MessageEvent('message', {\n data: isBinary ? data : data.toString()\n });\n\n event[kTarget] = this;\n callListener(handler, this, event);\n };\n } else if (type === 'close') {\n wrapper = function onClose(code, message) {\n const event = new CloseEvent('close', {\n code,\n reason: message.toString(),\n wasClean: this._closeFrameReceived && this._closeFrameSent\n });\n\n event[kTarget] = this;\n callListener(handler, this, event);\n };\n } else if (type === 'error') {\n wrapper = function onError(error) {\n const event = new ErrorEvent('error', {\n error,\n message: error.message\n });\n\n event[kTarget] = this;\n callListener(handler, this, event);\n };\n } else if (type === 'open') {\n wrapper = function onOpen() {\n const event = new Event('open');\n\n event[kTarget] = this;\n callListener(handler, this, event);\n };\n } else {\n return;\n }\n\n wrapper[kForOnEventAttribute] = !!options[kForOnEventAttribute];\n wrapper[kListener] = handler;\n\n if (options.once) {\n this.once(type, wrapper);\n } else {\n this.on(type, wrapper);\n }\n },\n\n /**\n * Remove an event listener.\n *\n * @param {String} type A string representing the event type to remove\n * @param {(Function|Object)} handler The listener to remove\n * @public\n */\n removeEventListener(type, handler) {\n for (const listener of this.listeners(type)) {\n if (listener[kListener] === handler && !listener[kForOnEventAttribute]) {\n this.removeListener(type, listener);\n break;\n }\n }\n }\n};\n\nmodule.exports = {\n CloseEvent,\n ErrorEvent,\n Event,\n EventTarget,\n MessageEvent\n};\n\n/**\n * Call an event listener\n *\n * @param {(Function|Object)} listener The listener to call\n * @param {*} thisArg The value to use as `this`` when calling the listener\n * @param {Event} event The event to pass to the listener\n * @private\n */\nfunction callListener(listener, thisArg, event) {\n if (typeof listener === 'object' && listener.handleEvent) {\n listener.handleEvent.call(listener, event);\n } else {\n listener.call(thisArg, event);\n }\n}\n","'use strict';\n\nconst { tokenChars } = require('./validation');\n\n/**\n * Adds an offer to the map of extension offers or a parameter to the map of\n * parameters.\n *\n * @param {Object} dest The map of extension offers or parameters\n * @param {String} name The extension or parameter name\n * @param {(Object|Boolean|String)} elem The extension parameters or the\n * parameter value\n * @private\n */\nfunction push(dest, name, elem) {\n if (dest[name] === undefined) dest[name] = [elem];\n else dest[name].push(elem);\n}\n\n/**\n * Parses the `Sec-WebSocket-Extensions` header into an object.\n *\n * @param {String} header The field value of the header\n * @return {Object} The parsed object\n * @public\n */\nfunction parse(header) {\n const offers = Object.create(null);\n let params = Object.create(null);\n let mustUnescape = false;\n let isEscaping = false;\n let inQuotes = false;\n let extensionName;\n let paramName;\n let start = -1;\n let code = -1;\n let end = -1;\n let i = 0;\n\n for (; i < header.length; i++) {\n code = header.charCodeAt(i);\n\n if (extensionName === undefined) {\n if (end === -1 && tokenChars[code] === 1) {\n if (start === -1) start = i;\n } else if (\n i !== 0 &&\n (code === 0x20 /* ' ' */ || code === 0x09) /* '\\t' */\n ) {\n if (end === -1 && start !== -1) end = i;\n } else if (code === 0x3b /* ';' */ || code === 0x2c /* ',' */) {\n if (start === -1) {\n throw new SyntaxError(`Unexpected character at index ${i}`);\n }\n\n if (end === -1) end = i;\n const name = header.slice(start, end);\n if (code === 0x2c) {\n push(offers, name, params);\n params = Object.create(null);\n } else {\n extensionName = name;\n }\n\n start = end = -1;\n } else {\n throw new SyntaxError(`Unexpected character at index ${i}`);\n }\n } else if (paramName === undefined) {\n if (end === -1 && tokenChars[code] === 1) {\n if (start === -1) start = i;\n } else if (code === 0x20 || code === 0x09) {\n if (end === -1 && start !== -1) end = i;\n } else if (code === 0x3b || code === 0x2c) {\n if (start === -1) {\n throw new SyntaxError(`Unexpected character at index ${i}`);\n }\n\n if (end === -1) end = i;\n push(params, header.slice(start, end), true);\n if (code === 0x2c) {\n push(offers, extensionName, params);\n params = Object.create(null);\n extensionName = undefined;\n }\n\n start = end = -1;\n } else if (code === 0x3d /* '=' */ && start !== -1 && end === -1) {\n paramName = header.slice(start, i);\n start = end = -1;\n } else {\n throw new SyntaxError(`Unexpected character at index ${i}`);\n }\n } else {\n //\n // The value of a quoted-string after unescaping must conform to the\n // token ABNF, so only token characters are valid.\n // Ref: https://tools.ietf.org/html/rfc6455#section-9.1\n //\n if (isEscaping) {\n if (tokenChars[code] !== 1) {\n throw new SyntaxError(`Unexpected character at index ${i}`);\n }\n if (start === -1) start = i;\n else if (!mustUnescape) mustUnescape = true;\n isEscaping = false;\n } else if (inQuotes) {\n if (tokenChars[code] === 1) {\n if (start === -1) start = i;\n } else if (code === 0x22 /* '\"' */ && start !== -1) {\n inQuotes = false;\n end = i;\n } else if (code === 0x5c /* '\\' */) {\n isEscaping = true;\n } else {\n throw new SyntaxError(`Unexpected character at index ${i}`);\n }\n } else if (code === 0x22 && header.charCodeAt(i - 1) === 0x3d) {\n inQuotes = true;\n } else if (end === -1 && tokenChars[code] === 1) {\n if (start === -1) start = i;\n } else if (start !== -1 && (code === 0x20 || code === 0x09)) {\n if (end === -1) end = i;\n } else if (code === 0x3b || code === 0x2c) {\n if (start === -1) {\n throw new SyntaxError(`Unexpected character at index ${i}`);\n }\n\n if (end === -1) end = i;\n let value = header.slice(start, end);\n if (mustUnescape) {\n value = value.replace(/\\\\/g, '');\n mustUnescape = false;\n }\n push(params, paramName, value);\n if (code === 0x2c) {\n push(offers, extensionName, params);\n params = Object.create(null);\n extensionName = undefined;\n }\n\n paramName = undefined;\n start = end = -1;\n } else {\n throw new SyntaxError(`Unexpected character at index ${i}`);\n }\n }\n }\n\n if (start === -1 || inQuotes || code === 0x20 || code === 0x09) {\n throw new SyntaxError('Unexpected end of input');\n }\n\n if (end === -1) end = i;\n const token = header.slice(start, end);\n if (extensionName === undefined) {\n push(offers, token, params);\n } else {\n if (paramName === undefined) {\n push(params, token, true);\n } else if (mustUnescape) {\n push(params, paramName, token.replace(/\\\\/g, ''));\n } else {\n push(params, paramName, token);\n }\n push(offers, extensionName, params);\n }\n\n return offers;\n}\n\n/**\n * Builds the `Sec-WebSocket-Extensions` header field value.\n *\n * @param {Object} extensions The map of extensions and parameters to format\n * @return {String} A string representing the given object\n * @public\n */\nfunction format(extensions) {\n return Object.keys(extensions)\n .map((extension) => {\n let configurations = extensions[extension];\n if (!Array.isArray(configurations)) configurations = [configurations];\n return configurations\n .map((params) => {\n return [extension]\n .concat(\n Object.keys(params).map((k) => {\n let values = params[k];\n if (!Array.isArray(values)) values = [values];\n return values\n .map((v) => (v === true ? k : `${k}=${v}`))\n .join('; ');\n })\n )\n .join('; ');\n })\n .join(', ');\n })\n .join(', ');\n}\n\nmodule.exports = { format, parse };\n","/* eslint no-unused-vars: [\"error\", { \"varsIgnorePattern\": \"^Duplex|Readable$\", \"caughtErrors\": \"none\" }] */\n\n'use strict';\n\nconst EventEmitter = require('events');\nconst https = require('https');\nconst http = require('http');\nconst net = require('net');\nconst tls = require('tls');\nconst { randomBytes, createHash } = require('crypto');\nconst { Duplex, Readable } = require('stream');\nconst { URL } = require('url');\n\nconst PerMessageDeflate = require('./permessage-deflate');\nconst Receiver = require('./receiver');\nconst Sender = require('./sender');\nconst { isBlob } = require('./validation');\n\nconst {\n BINARY_TYPES,\n CLOSE_TIMEOUT,\n EMPTY_BUFFER,\n GUID,\n kForOnEventAttribute,\n kListener,\n kStatusCode,\n kWebSocket,\n NOOP\n} = require('./constants');\nconst {\n EventTarget: { addEventListener, removeEventListener }\n} = require('./event-target');\nconst { format, parse } = require('./extension');\nconst { toBuffer } = require('./buffer-util');\n\nconst kAborted = Symbol('kAborted');\nconst protocolVersions = [8, 13];\nconst readyStates = ['CONNECTING', 'OPEN', 'CLOSING', 'CLOSED'];\nconst subprotocolRegex = /^[!#$%&'*+\\-.0-9A-Z^_`|a-z~]+$/;\n\n/**\n * Class representing a WebSocket.\n *\n * @extends EventEmitter\n */\nclass WebSocket extends EventEmitter {\n /**\n * Create a new `WebSocket`.\n *\n * @param {(String|URL)} address The URL to which to connect\n * @param {(String|String[])} [protocols] The subprotocols\n * @param {Object} [options] Connection options\n */\n constructor(address, protocols, options) {\n super();\n\n this._binaryType = BINARY_TYPES[0];\n this._closeCode = 1006;\n this._closeFrameReceived = false;\n this._closeFrameSent = false;\n this._closeMessage = EMPTY_BUFFER;\n this._closeTimer = null;\n this._errorEmitted = false;\n this._extensions = {};\n this._paused = false;\n this._protocol = '';\n this._readyState = WebSocket.CONNECTING;\n this._receiver = null;\n this._sender = null;\n this._socket = null;\n\n if (address !== null) {\n this._bufferedAmount = 0;\n this._isServer = false;\n this._redirects = 0;\n\n if (protocols === undefined) {\n protocols = [];\n } else if (!Array.isArray(protocols)) {\n if (typeof protocols === 'object' && protocols !== null) {\n options = protocols;\n protocols = [];\n } else {\n protocols = [protocols];\n }\n }\n\n initAsClient(this, address, protocols, options);\n } else {\n this._autoPong = options.autoPong;\n this._closeTimeout = options.closeTimeout;\n this._isServer = true;\n }\n }\n\n /**\n * For historical reasons, the custom \"nodebuffer\" type is used by the default\n * instead of \"blob\".\n *\n * @type {String}\n */\n get binaryType() {\n return this._binaryType;\n }\n\n set binaryType(type) {\n if (!BINARY_TYPES.includes(type)) return;\n\n this._binaryType = type;\n\n //\n // Allow to change `binaryType` on the fly.\n //\n if (this._receiver) this._receiver._binaryType = type;\n }\n\n /**\n * @type {Number}\n */\n get bufferedAmount() {\n if (!this._socket) return this._bufferedAmount;\n\n return this._socket._writableState.length + this._sender._bufferedBytes;\n }\n\n /**\n * @type {String}\n */\n get extensions() {\n return Object.keys(this._extensions).join();\n }\n\n /**\n * @type {Boolean}\n */\n get isPaused() {\n return this._paused;\n }\n\n /**\n * @type {Function}\n */\n /* istanbul ignore next */\n get onclose() {\n return null;\n }\n\n /**\n * @type {Function}\n */\n /* istanbul ignore next */\n get onerror() {\n return null;\n }\n\n /**\n * @type {Function}\n */\n /* istanbul ignore next */\n get onopen() {\n return null;\n }\n\n /**\n * @type {Function}\n */\n /* istanbul ignore next */\n get onmessage() {\n return null;\n }\n\n /**\n * @type {String}\n */\n get protocol() {\n return this._protocol;\n }\n\n /**\n * @type {Number}\n */\n get readyState() {\n return this._readyState;\n }\n\n /**\n * @type {String}\n */\n get url() {\n return this._url;\n }\n\n /**\n * Set up the socket and the internal resources.\n *\n * @param {Duplex} socket The network socket between the server and client\n * @param {Buffer} head The first packet of the upgraded stream\n * @param {Object} options Options object\n * @param {Boolean} [options.allowSynchronousEvents=false] Specifies whether\n * any of the `'message'`, `'ping'`, and `'pong'` events can be emitted\n * multiple times in the same tick\n * @param {Function} [options.generateMask] The function used to generate the\n * masking key\n * @param {Number} [options.maxPayload=0] The maximum allowed message size\n * @param {Boolean} [options.skipUTF8Validation=false] Specifies whether or\n * not to skip UTF-8 validation for text and close messages\n * @private\n */\n setSocket(socket, head, options) {\n const receiver = new Receiver({\n allowSynchronousEvents: options.allowSynchronousEvents,\n binaryType: this.binaryType,\n extensions: this._extensions,\n isServer: this._isServer,\n maxPayload: options.maxPayload,\n skipUTF8Validation: options.skipUTF8Validation\n });\n\n const sender = new Sender(socket, this._extensions, options.generateMask);\n\n this._receiver = receiver;\n this._sender = sender;\n this._socket = socket;\n\n receiver[kWebSocket] = this;\n sender[kWebSocket] = this;\n socket[kWebSocket] = this;\n\n receiver.on('conclude', receiverOnConclude);\n receiver.on('drain', receiverOnDrain);\n receiver.on('error', receiverOnError);\n receiver.on('message', receiverOnMessage);\n receiver.on('ping', receiverOnPing);\n receiver.on('pong', receiverOnPong);\n\n sender.onerror = senderOnError;\n\n //\n // These methods may not be available if `socket` is just a `Duplex`.\n //\n if (socket.setTimeout) socket.setTimeout(0);\n if (socket.setNoDelay) socket.setNoDelay();\n\n if (head.length > 0) socket.unshift(head);\n\n socket.on('close', socketOnClose);\n socket.on('data', socketOnData);\n socket.on('end', socketOnEnd);\n socket.on('error', socketOnError);\n\n this._readyState = WebSocket.OPEN;\n this.emit('open');\n }\n\n /**\n * Emit the `'close'` event.\n *\n * @private\n */\n emitClose() {\n if (!this._socket) {\n this._readyState = WebSocket.CLOSED;\n this.emit('close', this._closeCode, this._closeMessage);\n return;\n }\n\n if (this._extensions[PerMessageDeflate.extensionName]) {\n this._extensions[PerMessageDeflate.extensionName].cleanup();\n }\n\n this._receiver.removeAllListeners();\n this._readyState = WebSocket.CLOSED;\n this.emit('close', this._closeCode, this._closeMessage);\n }\n\n /**\n * Start a closing handshake.\n *\n * +----------+ +-----------+ +----------+\n * - - -|ws.close()|-->|close frame|-->|ws.close()|- - -\n * | +----------+ +-----------+ +----------+ |\n * +----------+ +-----------+ |\n * CLOSING |ws.close()|<--|close frame|<--+-----+ CLOSING\n * +----------+ +-----------+ |\n * | | | +---+ |\n * +------------------------+-->|fin| - - - -\n * | +---+ | +---+\n * - - - - -|fin|<---------------------+\n * +---+\n *\n * @param {Number} [code] Status code explaining why the connection is closing\n * @param {(String|Buffer)} [data] The reason why the connection is\n * closing\n * @public\n */\n close(code, data) {\n if (this.readyState === WebSocket.CLOSED) return;\n if (this.readyState === WebSocket.CONNECTING) {\n const msg = 'WebSocket was closed before the connection was established';\n abortHandshake(this, this._req, msg);\n return;\n }\n\n if (this.readyState === WebSocket.CLOSING) {\n if (\n this._closeFrameSent &&\n (this._closeFrameReceived || this._receiver._writableState.errorEmitted)\n ) {\n this._socket.end();\n }\n\n return;\n }\n\n this._readyState = WebSocket.CLOSING;\n this._sender.close(code, data, !this._isServer, (err) => {\n //\n // This error is handled by the `'error'` listener on the socket. We only\n // want to know if the close frame has been sent here.\n //\n if (err) return;\n\n this._closeFrameSent = true;\n\n if (\n this._closeFrameReceived ||\n this._receiver._writableState.errorEmitted\n ) {\n this._socket.end();\n }\n });\n\n setCloseTimer(this);\n }\n\n /**\n * Pause the socket.\n *\n * @public\n */\n pause() {\n if (\n this.readyState === WebSocket.CONNECTING ||\n this.readyState === WebSocket.CLOSED\n ) {\n return;\n }\n\n this._paused = true;\n this._socket.pause();\n }\n\n /**\n * Send a ping.\n *\n * @param {*} [data] The data to send\n * @param {Boolean} [mask] Indicates whether or not to mask `data`\n * @param {Function} [cb] Callback which is executed when the ping is sent\n * @public\n */\n ping(data, mask, cb) {\n if (this.readyState === WebSocket.CONNECTING) {\n throw new Error('WebSocket is not open: readyState 0 (CONNECTING)');\n }\n\n if (typeof data === 'function') {\n cb = data;\n data = mask = undefined;\n } else if (typeof mask === 'function') {\n cb = mask;\n mask = undefined;\n }\n\n if (typeof data === 'number') data = data.toString();\n\n if (this.readyState !== WebSocket.OPEN) {\n sendAfterClose(this, data, cb);\n return;\n }\n\n if (mask === undefined) mask = !this._isServer;\n this._sender.ping(data || EMPTY_BUFFER, mask, cb);\n }\n\n /**\n * Send a pong.\n *\n * @param {*} [data] The data to send\n * @param {Boolean} [mask] Indicates whether or not to mask `data`\n * @param {Function} [cb] Callback which is executed when the pong is sent\n * @public\n */\n pong(data, mask, cb) {\n if (this.readyState === WebSocket.CONNECTING) {\n throw new Error('WebSocket is not open: readyState 0 (CONNECTING)');\n }\n\n if (typeof data === 'function') {\n cb = data;\n data = mask = undefined;\n } else if (typeof mask === 'function') {\n cb = mask;\n mask = undefined;\n }\n\n if (typeof data === 'number') data = data.toString();\n\n if (this.readyState !== WebSocket.OPEN) {\n sendAfterClose(this, data, cb);\n return;\n }\n\n if (mask === undefined) mask = !this._isServer;\n this._sender.pong(data || EMPTY_BUFFER, mask, cb);\n }\n\n /**\n * Resume the socket.\n *\n * @public\n */\n resume() {\n if (\n this.readyState === WebSocket.CONNECTING ||\n this.readyState === WebSocket.CLOSED\n ) {\n return;\n }\n\n this._paused = false;\n if (!this._receiver._writableState.needDrain) this._socket.resume();\n }\n\n /**\n * Send a data message.\n *\n * @param {*} data The message to send\n * @param {Object} [options] Options object\n * @param {Boolean} [options.binary] Specifies whether `data` is binary or\n * text\n * @param {Boolean} [options.compress] Specifies whether or not to compress\n * `data`\n * @param {Boolean} [options.fin=true] Specifies whether the fragment is the\n * last one\n * @param {Boolean} [options.mask] Specifies whether or not to mask `data`\n * @param {Function} [cb] Callback which is executed when data is written out\n * @public\n */\n send(data, options, cb) {\n if (this.readyState === WebSocket.CONNECTING) {\n throw new Error('WebSocket is not open: readyState 0 (CONNECTING)');\n }\n\n if (typeof options === 'function') {\n cb = options;\n options = {};\n }\n\n if (typeof data === 'number') data = data.toString();\n\n if (this.readyState !== WebSocket.OPEN) {\n sendAfterClose(this, data, cb);\n return;\n }\n\n const opts = {\n binary: typeof data !== 'string',\n mask: !this._isServer,\n compress: true,\n fin: true,\n ...options\n };\n\n if (!this._extensions[PerMessageDeflate.extensionName]) {\n opts.compress = false;\n }\n\n this._sender.send(data || EMPTY_BUFFER, opts, cb);\n }\n\n /**\n * Forcibly close the connection.\n *\n * @public\n */\n terminate() {\n if (this.readyState === WebSocket.CLOSED) return;\n if (this.readyState === WebSocket.CONNECTING) {\n const msg = 'WebSocket was closed before the connection was established';\n abortHandshake(this, this._req, msg);\n return;\n }\n\n if (this._socket) {\n this._readyState = WebSocket.CLOSING;\n this._socket.destroy();\n }\n }\n}\n\n/**\n * @constant {Number} CONNECTING\n * @memberof WebSocket\n */\nObject.defineProperty(WebSocket, 'CONNECTING', {\n enumerable: true,\n value: readyStates.indexOf('CONNECTING')\n});\n\n/**\n * @constant {Number} CONNECTING\n * @memberof WebSocket.prototype\n */\nObject.defineProperty(WebSocket.prototype, 'CONNECTING', {\n enumerable: true,\n value: readyStates.indexOf('CONNECTING')\n});\n\n/**\n * @constant {Number} OPEN\n * @memberof WebSocket\n */\nObject.defineProperty(WebSocket, 'OPEN', {\n enumerable: true,\n value: readyStates.indexOf('OPEN')\n});\n\n/**\n * @constant {Number} OPEN\n * @memberof WebSocket.prototype\n */\nObject.defineProperty(WebSocket.prototype, 'OPEN', {\n enumerable: true,\n value: readyStates.indexOf('OPEN')\n});\n\n/**\n * @constant {Number} CLOSING\n * @memberof WebSocket\n */\nObject.defineProperty(WebSocket, 'CLOSING', {\n enumerable: true,\n value: readyStates.indexOf('CLOSING')\n});\n\n/**\n * @constant {Number} CLOSING\n * @memberof WebSocket.prototype\n */\nObject.defineProperty(WebSocket.prototype, 'CLOSING', {\n enumerable: true,\n value: readyStates.indexOf('CLOSING')\n});\n\n/**\n * @constant {Number} CLOSED\n * @memberof WebSocket\n */\nObject.defineProperty(WebSocket, 'CLOSED', {\n enumerable: true,\n value: readyStates.indexOf('CLOSED')\n});\n\n/**\n * @constant {Number} CLOSED\n * @memberof WebSocket.prototype\n */\nObject.defineProperty(WebSocket.prototype, 'CLOSED', {\n enumerable: true,\n value: readyStates.indexOf('CLOSED')\n});\n\n[\n 'binaryType',\n 'bufferedAmount',\n 'extensions',\n 'isPaused',\n 'protocol',\n 'readyState',\n 'url'\n].forEach((property) => {\n Object.defineProperty(WebSocket.prototype, property, { enumerable: true });\n});\n\n//\n// Add the `onopen`, `onerror`, `onclose`, and `onmessage` attributes.\n// See https://html.spec.whatwg.org/multipage/comms.html#the-websocket-interface\n//\n['open', 'error', 'close', 'message'].forEach((method) => {\n Object.defineProperty(WebSocket.prototype, `on${method}`, {\n enumerable: true,\n get() {\n for (const listener of this.listeners(method)) {\n if (listener[kForOnEventAttribute]) return listener[kListener];\n }\n\n return null;\n },\n set(handler) {\n for (const listener of this.listeners(method)) {\n if (listener[kForOnEventAttribute]) {\n this.removeListener(method, listener);\n break;\n }\n }\n\n if (typeof handler !== 'function') return;\n\n this.addEventListener(method, handler, {\n [kForOnEventAttribute]: true\n });\n }\n });\n});\n\nWebSocket.prototype.addEventListener = addEventListener;\nWebSocket.prototype.removeEventListener = removeEventListener;\n\nmodule.exports = WebSocket;\n\n/**\n * Initialize a WebSocket client.\n *\n * @param {WebSocket} websocket The client to initialize\n * @param {(String|URL)} address The URL to which to connect\n * @param {Array} protocols The subprotocols\n * @param {Object} [options] Connection options\n * @param {Boolean} [options.allowSynchronousEvents=true] Specifies whether any\n * of the `'message'`, `'ping'`, and `'pong'` events can be emitted multiple\n * times in the same tick\n * @param {Boolean} [options.autoPong=true] Specifies whether or not to\n * automatically send a pong in response to a ping\n * @param {Number} [options.closeTimeout=30000] Duration in milliseconds to wait\n * for the closing handshake to finish after `websocket.close()` is called\n * @param {Function} [options.finishRequest] A function which can be used to\n * customize the headers of each http request before it is sent\n * @param {Boolean} [options.followRedirects=false] Whether or not to follow\n * redirects\n * @param {Function} [options.generateMask] The function used to generate the\n * masking key\n * @param {Number} [options.handshakeTimeout] Timeout in milliseconds for the\n * handshake request\n * @param {Number} [options.maxPayload=104857600] The maximum allowed message\n * size\n * @param {Number} [options.maxRedirects=10] The maximum number of redirects\n * allowed\n * @param {String} [options.origin] Value of the `Origin` or\n * `Sec-WebSocket-Origin` header\n * @param {(Boolean|Object)} [options.perMessageDeflate=true] Enable/disable\n * permessage-deflate\n * @param {Number} [options.protocolVersion=13] Value of the\n * `Sec-WebSocket-Version` header\n * @param {Boolean} [options.skipUTF8Validation=false] Specifies whether or\n * not to skip UTF-8 validation for text and close messages\n * @private\n */\nfunction initAsClient(websocket, address, protocols, options) {\n const opts = {\n allowSynchronousEvents: true,\n autoPong: true,\n closeTimeout: CLOSE_TIMEOUT,\n protocolVersion: protocolVersions[1],\n maxPayload: 100 * 1024 * 1024,\n skipUTF8Validation: false,\n perMessageDeflate: true,\n followRedirects: false,\n maxRedirects: 10,\n ...options,\n socketPath: undefined,\n hostname: undefined,\n protocol: undefined,\n timeout: undefined,\n method: 'GET',\n host: undefined,\n path: undefined,\n port: undefined\n };\n\n websocket._autoPong = opts.autoPong;\n websocket._closeTimeout = opts.closeTimeout;\n\n if (!protocolVersions.includes(opts.protocolVersion)) {\n throw new RangeError(\n `Unsupported protocol version: ${opts.protocolVersion} ` +\n `(supported versions: ${protocolVersions.join(', ')})`\n );\n }\n\n let parsedUrl;\n\n if (address instanceof URL) {\n parsedUrl = address;\n } else {\n try {\n parsedUrl = new URL(address);\n } catch (e) {\n throw new SyntaxError(`Invalid URL: ${address}`);\n }\n }\n\n if (parsedUrl.protocol === 'http:') {\n parsedUrl.protocol = 'ws:';\n } else if (parsedUrl.protocol === 'https:') {\n parsedUrl.protocol = 'wss:';\n }\n\n websocket._url = parsedUrl.href;\n\n const isSecure = parsedUrl.protocol === 'wss:';\n const isIpcUrl = parsedUrl.protocol === 'ws+unix:';\n let invalidUrlMessage;\n\n if (parsedUrl.protocol !== 'ws:' && !isSecure && !isIpcUrl) {\n invalidUrlMessage =\n 'The URL\\'s protocol must be one of \"ws:\", \"wss:\", ' +\n '\"http:\", \"https:\", or \"ws+unix:\"';\n } else if (isIpcUrl && !parsedUrl.pathname) {\n invalidUrlMessage = \"The URL's pathname is empty\";\n } else if (parsedUrl.hash) {\n invalidUrlMessage = 'The URL contains a fragment identifier';\n }\n\n if (invalidUrlMessage) {\n const err = new SyntaxError(invalidUrlMessage);\n\n if (websocket._redirects === 0) {\n throw err;\n } else {\n emitErrorAndClose(websocket, err);\n return;\n }\n }\n\n const defaultPort = isSecure ? 443 : 80;\n const key = randomBytes(16).toString('base64');\n const request = isSecure ? https.request : http.request;\n const protocolSet = new Set();\n let perMessageDeflate;\n\n opts.createConnection =\n opts.createConnection || (isSecure ? tlsConnect : netConnect);\n opts.defaultPort = opts.defaultPort || defaultPort;\n opts.port = parsedUrl.port || defaultPort;\n opts.host = parsedUrl.hostname.startsWith('[')\n ? parsedUrl.hostname.slice(1, -1)\n : parsedUrl.hostname;\n opts.headers = {\n ...opts.headers,\n 'Sec-WebSocket-Version': opts.protocolVersion,\n 'Sec-WebSocket-Key': key,\n Connection: 'Upgrade',\n Upgrade: 'websocket'\n };\n opts.path = parsedUrl.pathname + parsedUrl.search;\n opts.timeout = opts.handshakeTimeout;\n\n if (opts.perMessageDeflate) {\n perMessageDeflate = new PerMessageDeflate(\n opts.perMessageDeflate !== true ? opts.perMessageDeflate : {},\n false,\n opts.maxPayload\n );\n opts.headers['Sec-WebSocket-Extensions'] = format({\n [PerMessageDeflate.extensionName]: perMessageDeflate.offer()\n });\n }\n if (protocols.length) {\n for (const protocol of protocols) {\n if (\n typeof protocol !== 'string' ||\n !subprotocolRegex.test(protocol) ||\n protocolSet.has(protocol)\n ) {\n throw new SyntaxError(\n 'An invalid or duplicated subprotocol was specified'\n );\n }\n\n protocolSet.add(protocol);\n }\n\n opts.headers['Sec-WebSocket-Protocol'] = protocols.join(',');\n }\n if (opts.origin) {\n if (opts.protocolVersion < 13) {\n opts.headers['Sec-WebSocket-Origin'] = opts.origin;\n } else {\n opts.headers.Origin = opts.origin;\n }\n }\n if (parsedUrl.username || parsedUrl.password) {\n opts.auth = `${parsedUrl.username}:${parsedUrl.password}`;\n }\n\n if (isIpcUrl) {\n const parts = opts.path.split(':');\n\n opts.socketPath = parts[0];\n opts.path = parts[1];\n }\n\n let req;\n\n if (opts.followRedirects) {\n if (websocket._redirects === 0) {\n websocket._originalIpc = isIpcUrl;\n websocket._originalSecure = isSecure;\n websocket._originalHostOrSocketPath = isIpcUrl\n ? opts.socketPath\n : parsedUrl.host;\n\n const headers = options && options.headers;\n\n //\n // Shallow copy the user provided options so that headers can be changed\n // without mutating the original object.\n //\n options = { ...options, headers: {} };\n\n if (headers) {\n for (const [key, value] of Object.entries(headers)) {\n options.headers[key.toLowerCase()] = value;\n }\n }\n } else if (websocket.listenerCount('redirect') === 0) {\n const isSameHost = isIpcUrl\n ? websocket._originalIpc\n ? opts.socketPath === websocket._originalHostOrSocketPath\n : false\n : websocket._originalIpc\n ? false\n : parsedUrl.host === websocket._originalHostOrSocketPath;\n\n if (!isSameHost || (websocket._originalSecure && !isSecure)) {\n //\n // Match curl 7.77.0 behavior and drop the following headers. These\n // headers are also dropped when following a redirect to a subdomain.\n //\n delete opts.headers.authorization;\n delete opts.headers.cookie;\n\n if (!isSameHost) delete opts.headers.host;\n\n opts.auth = undefined;\n }\n }\n\n //\n // Match curl 7.77.0 behavior and make the first `Authorization` header win.\n // If the `Authorization` header is set, then there is nothing to do as it\n // will take precedence.\n //\n if (opts.auth && !options.headers.authorization) {\n options.headers.authorization =\n 'Basic ' + Buffer.from(opts.auth).toString('base64');\n }\n\n req = websocket._req = request(opts);\n\n if (websocket._redirects) {\n //\n // Unlike what is done for the `'upgrade'` event, no early exit is\n // triggered here if the user calls `websocket.close()` or\n // `websocket.terminate()` from a listener of the `'redirect'` event. This\n // is because the user can also call `request.destroy()` with an error\n // before calling `websocket.close()` or `websocket.terminate()` and this\n // would result in an error being emitted on the `request` object with no\n // `'error'` event listeners attached.\n //\n websocket.emit('redirect', websocket.url, req);\n }\n } else {\n req = websocket._req = request(opts);\n }\n\n if (opts.timeout) {\n req.on('timeout', () => {\n abortHandshake(websocket, req, 'Opening handshake has timed out');\n });\n }\n\n req.on('error', (err) => {\n if (req === null || req[kAborted]) return;\n\n req = websocket._req = null;\n emitErrorAndClose(websocket, err);\n });\n\n req.on('response', (res) => {\n const location = res.headers.location;\n const statusCode = res.statusCode;\n\n if (\n location &&\n opts.followRedirects &&\n statusCode >= 300 &&\n statusCode < 400\n ) {\n if (++websocket._redirects > opts.maxRedirects) {\n abortHandshake(websocket, req, 'Maximum redirects exceeded');\n return;\n }\n\n req.abort();\n\n let addr;\n\n try {\n addr = new URL(location, address);\n } catch (e) {\n const err = new SyntaxError(`Invalid URL: ${location}`);\n emitErrorAndClose(websocket, err);\n return;\n }\n\n initAsClient(websocket, addr, protocols, options);\n } else if (!websocket.emit('unexpected-response', req, res)) {\n abortHandshake(\n websocket,\n req,\n `Unexpected server response: ${res.statusCode}`\n );\n }\n });\n\n req.on('upgrade', (res, socket, head) => {\n websocket.emit('upgrade', res);\n\n //\n // The user may have closed the connection from a listener of the\n // `'upgrade'` event.\n //\n if (websocket.readyState !== WebSocket.CONNECTING) return;\n\n req = websocket._req = null;\n\n const upgrade = res.headers.upgrade;\n\n if (upgrade === undefined || upgrade.toLowerCase() !== 'websocket') {\n abortHandshake(websocket, socket, 'Invalid Upgrade header');\n return;\n }\n\n const digest = createHash('sha1')\n .update(key + GUID)\n .digest('base64');\n\n if (res.headers['sec-websocket-accept'] !== digest) {\n abortHandshake(websocket, socket, 'Invalid Sec-WebSocket-Accept header');\n return;\n }\n\n const serverProt = res.headers['sec-websocket-protocol'];\n let protError;\n\n if (serverProt !== undefined) {\n if (!protocolSet.size) {\n protError = 'Server sent a subprotocol but none was requested';\n } else if (!protocolSet.has(serverProt)) {\n protError = 'Server sent an invalid subprotocol';\n }\n } else if (protocolSet.size) {\n protError = 'Server sent no subprotocol';\n }\n\n if (protError) {\n abortHandshake(websocket, socket, protError);\n return;\n }\n\n if (serverProt) websocket._protocol = serverProt;\n\n const secWebSocketExtensions = res.headers['sec-websocket-extensions'];\n\n if (secWebSocketExtensions !== undefined) {\n if (!perMessageDeflate) {\n const message =\n 'Server sent a Sec-WebSocket-Extensions header but no extension ' +\n 'was requested';\n abortHandshake(websocket, socket, message);\n return;\n }\n\n let extensions;\n\n try {\n extensions = parse(secWebSocketExtensions);\n } catch (err) {\n const message = 'Invalid Sec-WebSocket-Extensions header';\n abortHandshake(websocket, socket, message);\n return;\n }\n\n const extensionNames = Object.keys(extensions);\n\n if (\n extensionNames.length !== 1 ||\n extensionNames[0] !== PerMessageDeflate.extensionName\n ) {\n const message = 'Server indicated an extension that was not requested';\n abortHandshake(websocket, socket, message);\n return;\n }\n\n try {\n perMessageDeflate.accept(extensions[PerMessageDeflate.extensionName]);\n } catch (err) {\n const message = 'Invalid Sec-WebSocket-Extensions header';\n abortHandshake(websocket, socket, message);\n return;\n }\n\n websocket._extensions[PerMessageDeflate.extensionName] =\n perMessageDeflate;\n }\n\n websocket.setSocket(socket, head, {\n allowSynchronousEvents: opts.allowSynchronousEvents,\n generateMask: opts.generateMask,\n maxPayload: opts.maxPayload,\n skipUTF8Validation: opts.skipUTF8Validation\n });\n });\n\n if (opts.finishRequest) {\n opts.finishRequest(req, websocket);\n } else {\n req.end();\n }\n}\n\n/**\n * Emit the `'error'` and `'close'` events.\n *\n * @param {WebSocket} websocket The WebSocket instance\n * @param {Error} The error to emit\n * @private\n */\nfunction emitErrorAndClose(websocket, err) {\n websocket._readyState = WebSocket.CLOSING;\n //\n // The following assignment is practically useless and is done only for\n // consistency.\n //\n websocket._errorEmitted = true;\n websocket.emit('error', err);\n websocket.emitClose();\n}\n\n/**\n * Create a `net.Socket` and initiate a connection.\n *\n * @param {Object} options Connection options\n * @return {net.Socket} The newly created socket used to start the connection\n * @private\n */\nfunction netConnect(options) {\n options.path = options.socketPath;\n return net.connect(options);\n}\n\n/**\n * Create a `tls.TLSSocket` and initiate a connection.\n *\n * @param {Object} options Connection options\n * @return {tls.TLSSocket} The newly created socket used to start the connection\n * @private\n */\nfunction tlsConnect(options) {\n options.path = undefined;\n\n if (!options.servername && options.servername !== '') {\n options.servername = net.isIP(options.host) ? '' : options.host;\n }\n\n return tls.connect(options);\n}\n\n/**\n * Abort the handshake and emit an error.\n *\n * @param {WebSocket} websocket The WebSocket instance\n * @param {(http.ClientRequest|net.Socket|tls.Socket)} stream The request to\n * abort or the socket to destroy\n * @param {String} message The error message\n * @private\n */\nfunction abortHandshake(websocket, stream, message) {\n websocket._readyState = WebSocket.CLOSING;\n\n const err = new Error(message);\n Error.captureStackTrace(err, abortHandshake);\n\n if (stream.setHeader) {\n stream[kAborted] = true;\n stream.abort();\n\n if (stream.socket && !stream.socket.destroyed) {\n //\n // On Node.js >= 14.3.0 `request.abort()` does not destroy the socket if\n // called after the request completed. See\n // https://github.com/websockets/ws/issues/1869.\n //\n stream.socket.destroy();\n }\n\n process.nextTick(emitErrorAndClose, websocket, err);\n } else {\n stream.destroy(err);\n stream.once('error', websocket.emit.bind(websocket, 'error'));\n stream.once('close', websocket.emitClose.bind(websocket));\n }\n}\n\n/**\n * Handle cases where the `ping()`, `pong()`, or `send()` methods are called\n * when the `readyState` attribute is `CLOSING` or `CLOSED`.\n *\n * @param {WebSocket} websocket The WebSocket instance\n * @param {*} [data] The data to send\n * @param {Function} [cb] Callback\n * @private\n */\nfunction sendAfterClose(websocket, data, cb) {\n if (data) {\n const length = isBlob(data) ? data.size : toBuffer(data).length;\n\n //\n // The `_bufferedAmount` property is used only when the peer is a client and\n // the opening handshake fails. Under these circumstances, in fact, the\n // `setSocket()` method is not called, so the `_socket` and `_sender`\n // properties are set to `null`.\n //\n if (websocket._socket) websocket._sender._bufferedBytes += length;\n else websocket._bufferedAmount += length;\n }\n\n if (cb) {\n const err = new Error(\n `WebSocket is not open: readyState ${websocket.readyState} ` +\n `(${readyStates[websocket.readyState]})`\n );\n process.nextTick(cb, err);\n }\n}\n\n/**\n * The listener of the `Receiver` `'conclude'` event.\n *\n * @param {Number} code The status code\n * @param {Buffer} reason The reason for closing\n * @private\n */\nfunction receiverOnConclude(code, reason) {\n const websocket = this[kWebSocket];\n\n websocket._closeFrameReceived = true;\n websocket._closeMessage = reason;\n websocket._closeCode = code;\n\n if (websocket._socket[kWebSocket] === undefined) return;\n\n websocket._socket.removeListener('data', socketOnData);\n process.nextTick(resume, websocket._socket);\n\n if (code === 1005) websocket.close();\n else websocket.close(code, reason);\n}\n\n/**\n * The listener of the `Receiver` `'drain'` event.\n *\n * @private\n */\nfunction receiverOnDrain() {\n const websocket = this[kWebSocket];\n\n if (!websocket.isPaused) websocket._socket.resume();\n}\n\n/**\n * The listener of the `Receiver` `'error'` event.\n *\n * @param {(RangeError|Error)} err The emitted error\n * @private\n */\nfunction receiverOnError(err) {\n const websocket = this[kWebSocket];\n\n if (websocket._socket[kWebSocket] !== undefined) {\n websocket._socket.removeListener('data', socketOnData);\n\n //\n // On Node.js < 14.0.0 the `'error'` event is emitted synchronously. See\n // https://github.com/websockets/ws/issues/1940.\n //\n process.nextTick(resume, websocket._socket);\n\n websocket.close(err[kStatusCode]);\n }\n\n if (!websocket._errorEmitted) {\n websocket._errorEmitted = true;\n websocket.emit('error', err);\n }\n}\n\n/**\n * The listener of the `Receiver` `'finish'` event.\n *\n * @private\n */\nfunction receiverOnFinish() {\n this[kWebSocket].emitClose();\n}\n\n/**\n * The listener of the `Receiver` `'message'` event.\n *\n * @param {Buffer|ArrayBuffer|Buffer[])} data The message\n * @param {Boolean} isBinary Specifies whether the message is binary or not\n * @private\n */\nfunction receiverOnMessage(data, isBinary) {\n this[kWebSocket].emit('message', data, isBinary);\n}\n\n/**\n * The listener of the `Receiver` `'ping'` event.\n *\n * @param {Buffer} data The data included in the ping frame\n * @private\n */\nfunction receiverOnPing(data) {\n const websocket = this[kWebSocket];\n\n if (websocket._autoPong) websocket.pong(data, !this._isServer, NOOP);\n websocket.emit('ping', data);\n}\n\n/**\n * The listener of the `Receiver` `'pong'` event.\n *\n * @param {Buffer} data The data included in the pong frame\n * @private\n */\nfunction receiverOnPong(data) {\n this[kWebSocket].emit('pong', data);\n}\n\n/**\n * Resume a readable stream\n *\n * @param {Readable} stream The readable stream\n * @private\n */\nfunction resume(stream) {\n stream.resume();\n}\n\n/**\n * The `Sender` error event handler.\n *\n * @param {Error} The error\n * @private\n */\nfunction senderOnError(err) {\n const websocket = this[kWebSocket];\n\n if (websocket.readyState === WebSocket.CLOSED) return;\n if (websocket.readyState === WebSocket.OPEN) {\n websocket._readyState = WebSocket.CLOSING;\n setCloseTimer(websocket);\n }\n\n //\n // `socket.end()` is used instead of `socket.destroy()` to allow the other\n // peer to finish sending queued data. There is no need to set a timer here\n // because `CLOSING` means that it is already set or not needed.\n //\n this._socket.end();\n\n if (!websocket._errorEmitted) {\n websocket._errorEmitted = true;\n websocket.emit('error', err);\n }\n}\n\n/**\n * Set a timer to destroy the underlying raw socket of a WebSocket.\n *\n * @param {WebSocket} websocket The WebSocket instance\n * @private\n */\nfunction setCloseTimer(websocket) {\n websocket._closeTimer = setTimeout(\n websocket._socket.destroy.bind(websocket._socket),\n websocket._closeTimeout\n );\n}\n\n/**\n * The listener of the socket `'close'` event.\n *\n * @private\n */\nfunction socketOnClose() {\n const websocket = this[kWebSocket];\n\n this.removeListener('close', socketOnClose);\n this.removeListener('data', socketOnData);\n this.removeListener('end', socketOnEnd);\n\n websocket._readyState = WebSocket.CLOSING;\n\n //\n // The close frame might not have been received or the `'end'` event emitted,\n // for example, if the socket was destroyed due to an error. Ensure that the\n // `receiver` stream is closed after writing any remaining buffered data to\n // it. If the readable side of the socket is in flowing mode then there is no\n // buffered data as everything has been already written. If instead, the\n // socket is paused, any possible buffered data will be read as a single\n // chunk.\n //\n if (\n !this._readableState.endEmitted &&\n !websocket._closeFrameReceived &&\n !websocket._receiver._writableState.errorEmitted &&\n this._readableState.length !== 0\n ) {\n const chunk = this.read(this._readableState.length);\n\n websocket._receiver.write(chunk);\n }\n\n websocket._receiver.end();\n\n this[kWebSocket] = undefined;\n\n clearTimeout(websocket._closeTimer);\n\n if (\n websocket._receiver._writableState.finished ||\n websocket._receiver._writableState.errorEmitted\n ) {\n websocket.emitClose();\n } else {\n websocket._receiver.on('error', receiverOnFinish);\n websocket._receiver.on('finish', receiverOnFinish);\n }\n}\n\n/**\n * The listener of the socket `'data'` event.\n *\n * @param {Buffer} chunk A chunk of data\n * @private\n */\nfunction socketOnData(chunk) {\n if (!this[kWebSocket]._receiver.write(chunk)) {\n this.pause();\n }\n}\n\n/**\n * The listener of the socket `'end'` event.\n *\n * @private\n */\nfunction socketOnEnd() {\n const websocket = this[kWebSocket];\n\n websocket._readyState = WebSocket.CLOSING;\n websocket._receiver.end();\n this.end();\n}\n\n/**\n * The listener of the socket `'error'` event.\n *\n * @private\n */\nfunction socketOnError() {\n const websocket = this[kWebSocket];\n\n this.removeListener('error', socketOnError);\n this.on('error', NOOP);\n\n if (websocket) {\n websocket._readyState = WebSocket.CLOSING;\n this.destroy();\n }\n}\n","/* eslint no-unused-vars: [\"error\", { \"varsIgnorePattern\": \"^WebSocket$\" }] */\n'use strict';\n\nconst WebSocket = require('./websocket');\nconst { Duplex } = require('stream');\n\n/**\n * Emits the `'close'` event on a stream.\n *\n * @param {Duplex} stream The stream.\n * @private\n */\nfunction emitClose(stream) {\n stream.emit('close');\n}\n\n/**\n * The listener of the `'end'` event.\n *\n * @private\n */\nfunction duplexOnEnd() {\n if (!this.destroyed && this._writableState.finished) {\n this.destroy();\n }\n}\n\n/**\n * The listener of the `'error'` event.\n *\n * @param {Error} err The error\n * @private\n */\nfunction duplexOnError(err) {\n this.removeListener('error', duplexOnError);\n this.destroy();\n if (this.listenerCount('error') === 0) {\n // Do not suppress the throwing behavior.\n this.emit('error', err);\n }\n}\n\n/**\n * Wraps a `WebSocket` in a duplex stream.\n *\n * @param {WebSocket} ws The `WebSocket` to wrap\n * @param {Object} [options] The options for the `Duplex` constructor\n * @return {Duplex} The duplex stream\n * @public\n */\nfunction createWebSocketStream(ws, options) {\n let terminateOnDestroy = true;\n\n const duplex = new Duplex({\n ...options,\n autoDestroy: false,\n emitClose: false,\n objectMode: false,\n writableObjectMode: false\n });\n\n ws.on('message', function message(msg, isBinary) {\n const data =\n !isBinary && duplex._readableState.objectMode ? msg.toString() : msg;\n\n if (!duplex.push(data)) ws.pause();\n });\n\n ws.once('error', function error(err) {\n if (duplex.destroyed) return;\n\n // Prevent `ws.terminate()` from being called by `duplex._destroy()`.\n //\n // - If the `'error'` event is emitted before the `'open'` event, then\n // `ws.terminate()` is a noop as no socket is assigned.\n // - Otherwise, the error is re-emitted by the listener of the `'error'`\n // event of the `Receiver` object. The listener already closes the\n // connection by calling `ws.close()`. This allows a close frame to be\n // sent to the other peer. If `ws.terminate()` is called right after this,\n // then the close frame might not be sent.\n terminateOnDestroy = false;\n duplex.destroy(err);\n });\n\n ws.once('close', function close() {\n if (duplex.destroyed) return;\n\n duplex.push(null);\n });\n\n duplex._destroy = function (err, callback) {\n if (ws.readyState === ws.CLOSED) {\n callback(err);\n process.nextTick(emitClose, duplex);\n return;\n }\n\n let called = false;\n\n ws.once('error', function error(err) {\n called = true;\n callback(err);\n });\n\n ws.once('close', function close() {\n if (!called) callback(err);\n process.nextTick(emitClose, duplex);\n });\n\n if (terminateOnDestroy) ws.terminate();\n };\n\n duplex._final = function (callback) {\n if (ws.readyState === ws.CONNECTING) {\n ws.once('open', function open() {\n duplex._final(callback);\n });\n return;\n }\n\n // If the value of the `_socket` property is `null` it means that `ws` is a\n // client websocket and the handshake failed. In fact, when this happens, a\n // socket is never assigned to the websocket. Wait for the `'error'` event\n // that will be emitted by the websocket.\n if (ws._socket === null) return;\n\n if (ws._socket._writableState.finished) {\n callback();\n if (duplex._readableState.endEmitted) duplex.destroy();\n } else {\n ws._socket.once('finish', function finish() {\n // `duplex` is not destroyed here because the `'end'` event will be\n // emitted on `duplex` after this `'finish'` event. The EOF signaling\n // `null` chunk is, in fact, pushed when the websocket emits `'close'`.\n callback();\n });\n ws.close();\n }\n };\n\n duplex._read = function () {\n if (ws.isPaused) ws.resume();\n };\n\n duplex._write = function (chunk, encoding, callback) {\n if (ws.readyState === ws.CONNECTING) {\n ws.once('open', function open() {\n duplex._write(chunk, encoding, callback);\n });\n return;\n }\n\n ws.send(chunk, callback);\n };\n\n duplex.on('end', duplexOnEnd);\n duplex.on('error', duplexOnError);\n return duplex;\n}\n\nmodule.exports = createWebSocketStream;\n","'use strict';\n\nconst { tokenChars } = require('./validation');\n\n/**\n * Parses the `Sec-WebSocket-Protocol` header into a set of subprotocol names.\n *\n * @param {String} header The field value of the header\n * @return {Set} The subprotocol names\n * @public\n */\nfunction parse(header) {\n const protocols = new Set();\n let start = -1;\n let end = -1;\n let i = 0;\n\n for (i; i < header.length; i++) {\n const code = header.charCodeAt(i);\n\n if (end === -1 && tokenChars[code] === 1) {\n if (start === -1) start = i;\n } else if (\n i !== 0 &&\n (code === 0x20 /* ' ' */ || code === 0x09) /* '\\t' */\n ) {\n if (end === -1 && start !== -1) end = i;\n } else if (code === 0x2c /* ',' */) {\n if (start === -1) {\n throw new SyntaxError(`Unexpected character at index ${i}`);\n }\n\n if (end === -1) end = i;\n\n const protocol = header.slice(start, end);\n\n if (protocols.has(protocol)) {\n throw new SyntaxError(`The \"${protocol}\" subprotocol is duplicated`);\n }\n\n protocols.add(protocol);\n start = end = -1;\n } else {\n throw new SyntaxError(`Unexpected character at index ${i}`);\n }\n }\n\n if (start === -1 || end !== -1) {\n throw new SyntaxError('Unexpected end of input');\n }\n\n const protocol = header.slice(start, i);\n\n if (protocols.has(protocol)) {\n throw new SyntaxError(`The \"${protocol}\" subprotocol is duplicated`);\n }\n\n protocols.add(protocol);\n return protocols;\n}\n\nmodule.exports = { parse };\n","/* eslint no-unused-vars: [\"error\", { \"varsIgnorePattern\": \"^Duplex$\", \"caughtErrors\": \"none\" }] */\n\n'use strict';\n\nconst EventEmitter = require('events');\nconst http = require('http');\nconst { Duplex } = require('stream');\nconst { createHash } = require('crypto');\n\nconst extension = require('./extension');\nconst PerMessageDeflate = require('./permessage-deflate');\nconst subprotocol = require('./subprotocol');\nconst WebSocket = require('./websocket');\nconst { CLOSE_TIMEOUT, GUID, kWebSocket } = require('./constants');\n\nconst keyRegex = /^[+/0-9A-Za-z]{22}==$/;\n\nconst RUNNING = 0;\nconst CLOSING = 1;\nconst CLOSED = 2;\n\n/**\n * Class representing a WebSocket server.\n *\n * @extends EventEmitter\n */\nclass WebSocketServer extends EventEmitter {\n /**\n * Create a `WebSocketServer` instance.\n *\n * @param {Object} options Configuration options\n * @param {Boolean} [options.allowSynchronousEvents=true] Specifies whether\n * any of the `'message'`, `'ping'`, and `'pong'` events can be emitted\n * multiple times in the same tick\n * @param {Boolean} [options.autoPong=true] Specifies whether or not to\n * automatically send a pong in response to a ping\n * @param {Number} [options.backlog=511] The maximum length of the queue of\n * pending connections\n * @param {Boolean} [options.clientTracking=true] Specifies whether or not to\n * track clients\n * @param {Number} [options.closeTimeout=30000] Duration in milliseconds to\n * wait for the closing handshake to finish after `websocket.close()` is\n * called\n * @param {Function} [options.handleProtocols] A hook to handle protocols\n * @param {String} [options.host] The hostname where to bind the server\n * @param {Number} [options.maxPayload=104857600] The maximum allowed message\n * size\n * @param {Boolean} [options.noServer=false] Enable no server mode\n * @param {String} [options.path] Accept only connections matching this path\n * @param {(Boolean|Object)} [options.perMessageDeflate=false] Enable/disable\n * permessage-deflate\n * @param {Number} [options.port] The port where to bind the server\n * @param {(http.Server|https.Server)} [options.server] A pre-created HTTP/S\n * server to use\n * @param {Boolean} [options.skipUTF8Validation=false] Specifies whether or\n * not to skip UTF-8 validation for text and close messages\n * @param {Function} [options.verifyClient] A hook to reject connections\n * @param {Function} [options.WebSocket=WebSocket] Specifies the `WebSocket`\n * class to use. It must be the `WebSocket` class or class that extends it\n * @param {Function} [callback] A listener for the `listening` event\n */\n constructor(options, callback) {\n super();\n\n options = {\n allowSynchronousEvents: true,\n autoPong: true,\n maxPayload: 100 * 1024 * 1024,\n skipUTF8Validation: false,\n perMessageDeflate: false,\n handleProtocols: null,\n clientTracking: true,\n closeTimeout: CLOSE_TIMEOUT,\n verifyClient: null,\n noServer: false,\n backlog: null, // use default (511 as implemented in net.js)\n server: null,\n host: null,\n path: null,\n port: null,\n WebSocket,\n ...options\n };\n\n if (\n (options.port == null && !options.server && !options.noServer) ||\n (options.port != null && (options.server || options.noServer)) ||\n (options.server && options.noServer)\n ) {\n throw new TypeError(\n 'One and only one of the \"port\", \"server\", or \"noServer\" options ' +\n 'must be specified'\n );\n }\n\n if (options.port != null) {\n this._server = http.createServer((req, res) => {\n const body = http.STATUS_CODES[426];\n\n res.writeHead(426, {\n 'Content-Length': body.length,\n 'Content-Type': 'text/plain'\n });\n res.end(body);\n });\n this._server.listen(\n options.port,\n options.host,\n options.backlog,\n callback\n );\n } else if (options.server) {\n this._server = options.server;\n }\n\n if (this._server) {\n const emitConnection = this.emit.bind(this, 'connection');\n\n this._removeListeners = addListeners(this._server, {\n listening: this.emit.bind(this, 'listening'),\n error: this.emit.bind(this, 'error'),\n upgrade: (req, socket, head) => {\n this.handleUpgrade(req, socket, head, emitConnection);\n }\n });\n }\n\n if (options.perMessageDeflate === true) options.perMessageDeflate = {};\n if (options.clientTracking) {\n this.clients = new Set();\n this._shouldEmitClose = false;\n }\n\n this.options = options;\n this._state = RUNNING;\n }\n\n /**\n * Returns the bound address, the address family name, and port of the server\n * as reported by the operating system if listening on an IP socket.\n * If the server is listening on a pipe or UNIX domain socket, the name is\n * returned as a string.\n *\n * @return {(Object|String|null)} The address of the server\n * @public\n */\n address() {\n if (this.options.noServer) {\n throw new Error('The server is operating in \"noServer\" mode');\n }\n\n if (!this._server) return null;\n return this._server.address();\n }\n\n /**\n * Stop the server from accepting new connections and emit the `'close'` event\n * when all existing connections are closed.\n *\n * @param {Function} [cb] A one-time listener for the `'close'` event\n * @public\n */\n close(cb) {\n if (this._state === CLOSED) {\n if (cb) {\n this.once('close', () => {\n cb(new Error('The server is not running'));\n });\n }\n\n process.nextTick(emitClose, this);\n return;\n }\n\n if (cb) this.once('close', cb);\n\n if (this._state === CLOSING) return;\n this._state = CLOSING;\n\n if (this.options.noServer || this.options.server) {\n if (this._server) {\n this._removeListeners();\n this._removeListeners = this._server = null;\n }\n\n if (this.clients) {\n if (!this.clients.size) {\n process.nextTick(emitClose, this);\n } else {\n this._shouldEmitClose = true;\n }\n } else {\n process.nextTick(emitClose, this);\n }\n } else {\n const server = this._server;\n\n this._removeListeners();\n this._removeListeners = this._server = null;\n\n //\n // The HTTP/S server was created internally. Close it, and rely on its\n // `'close'` event.\n //\n server.close(() => {\n emitClose(this);\n });\n }\n }\n\n /**\n * See if a given request should be handled by this server instance.\n *\n * @param {http.IncomingMessage} req Request object to inspect\n * @return {Boolean} `true` if the request is valid, else `false`\n * @public\n */\n shouldHandle(req) {\n if (this.options.path) {\n const index = req.url.indexOf('?');\n const pathname = index !== -1 ? req.url.slice(0, index) : req.url;\n\n if (pathname !== this.options.path) return false;\n }\n\n return true;\n }\n\n /**\n * Handle a HTTP Upgrade request.\n *\n * @param {http.IncomingMessage} req The request object\n * @param {Duplex} socket The network socket between the server and client\n * @param {Buffer} head The first packet of the upgraded stream\n * @param {Function} cb Callback\n * @public\n */\n handleUpgrade(req, socket, head, cb) {\n socket.on('error', socketOnError);\n\n const key = req.headers['sec-websocket-key'];\n const upgrade = req.headers.upgrade;\n const version = +req.headers['sec-websocket-version'];\n\n if (req.method !== 'GET') {\n const message = 'Invalid HTTP method';\n abortHandshakeOrEmitwsClientError(this, req, socket, 405, message);\n return;\n }\n\n if (upgrade === undefined || upgrade.toLowerCase() !== 'websocket') {\n const message = 'Invalid Upgrade header';\n abortHandshakeOrEmitwsClientError(this, req, socket, 400, message);\n return;\n }\n\n if (key === undefined || !keyRegex.test(key)) {\n const message = 'Missing or invalid Sec-WebSocket-Key header';\n abortHandshakeOrEmitwsClientError(this, req, socket, 400, message);\n return;\n }\n\n if (version !== 13 && version !== 8) {\n const message = 'Missing or invalid Sec-WebSocket-Version header';\n abortHandshakeOrEmitwsClientError(this, req, socket, 400, message, {\n 'Sec-WebSocket-Version': '13, 8'\n });\n return;\n }\n\n if (!this.shouldHandle(req)) {\n abortHandshake(socket, 400);\n return;\n }\n\n const secWebSocketProtocol = req.headers['sec-websocket-protocol'];\n let protocols = new Set();\n\n if (secWebSocketProtocol !== undefined) {\n try {\n protocols = subprotocol.parse(secWebSocketProtocol);\n } catch (err) {\n const message = 'Invalid Sec-WebSocket-Protocol header';\n abortHandshakeOrEmitwsClientError(this, req, socket, 400, message);\n return;\n }\n }\n\n const secWebSocketExtensions = req.headers['sec-websocket-extensions'];\n const extensions = {};\n\n if (\n this.options.perMessageDeflate &&\n secWebSocketExtensions !== undefined\n ) {\n const perMessageDeflate = new PerMessageDeflate(\n this.options.perMessageDeflate,\n true,\n this.options.maxPayload\n );\n\n try {\n const offers = extension.parse(secWebSocketExtensions);\n\n if (offers[PerMessageDeflate.extensionName]) {\n perMessageDeflate.accept(offers[PerMessageDeflate.extensionName]);\n extensions[PerMessageDeflate.extensionName] = perMessageDeflate;\n }\n } catch (err) {\n const message =\n 'Invalid or unacceptable Sec-WebSocket-Extensions header';\n abortHandshakeOrEmitwsClientError(this, req, socket, 400, message);\n return;\n }\n }\n\n //\n // Optionally call external client verification handler.\n //\n if (this.options.verifyClient) {\n const info = {\n origin:\n req.headers[`${version === 8 ? 'sec-websocket-origin' : 'origin'}`],\n secure: !!(req.socket.authorized || req.socket.encrypted),\n req\n };\n\n if (this.options.verifyClient.length === 2) {\n this.options.verifyClient(info, (verified, code, message, headers) => {\n if (!verified) {\n return abortHandshake(socket, code || 401, message, headers);\n }\n\n this.completeUpgrade(\n extensions,\n key,\n protocols,\n req,\n socket,\n head,\n cb\n );\n });\n return;\n }\n\n if (!this.options.verifyClient(info)) return abortHandshake(socket, 401);\n }\n\n this.completeUpgrade(extensions, key, protocols, req, socket, head, cb);\n }\n\n /**\n * Upgrade the connection to WebSocket.\n *\n * @param {Object} extensions The accepted extensions\n * @param {String} key The value of the `Sec-WebSocket-Key` header\n * @param {Set} protocols The subprotocols\n * @param {http.IncomingMessage} req The request object\n * @param {Duplex} socket The network socket between the server and client\n * @param {Buffer} head The first packet of the upgraded stream\n * @param {Function} cb Callback\n * @throws {Error} If called more than once with the same socket\n * @private\n */\n completeUpgrade(extensions, key, protocols, req, socket, head, cb) {\n //\n // Destroy the socket if the client has already sent a FIN packet.\n //\n if (!socket.readable || !socket.writable) return socket.destroy();\n\n if (socket[kWebSocket]) {\n throw new Error(\n 'server.handleUpgrade() was called more than once with the same ' +\n 'socket, possibly due to a misconfiguration'\n );\n }\n\n if (this._state > RUNNING) return abortHandshake(socket, 503);\n\n const digest = createHash('sha1')\n .update(key + GUID)\n .digest('base64');\n\n const headers = [\n 'HTTP/1.1 101 Switching Protocols',\n 'Upgrade: websocket',\n 'Connection: Upgrade',\n `Sec-WebSocket-Accept: ${digest}`\n ];\n\n const ws = new this.options.WebSocket(null, undefined, this.options);\n\n if (protocols.size) {\n //\n // Optionally call external protocol selection handler.\n //\n const protocol = this.options.handleProtocols\n ? this.options.handleProtocols(protocols, req)\n : protocols.values().next().value;\n\n if (protocol) {\n headers.push(`Sec-WebSocket-Protocol: ${protocol}`);\n ws._protocol = protocol;\n }\n }\n\n if (extensions[PerMessageDeflate.extensionName]) {\n const params = extensions[PerMessageDeflate.extensionName].params;\n const value = extension.format({\n [PerMessageDeflate.extensionName]: [params]\n });\n headers.push(`Sec-WebSocket-Extensions: ${value}`);\n ws._extensions = extensions;\n }\n\n //\n // Allow external modification/inspection of handshake headers.\n //\n this.emit('headers', headers, req);\n\n socket.write(headers.concat('\\r\\n').join('\\r\\n'));\n socket.removeListener('error', socketOnError);\n\n ws.setSocket(socket, head, {\n allowSynchronousEvents: this.options.allowSynchronousEvents,\n maxPayload: this.options.maxPayload,\n skipUTF8Validation: this.options.skipUTF8Validation\n });\n\n if (this.clients) {\n this.clients.add(ws);\n ws.on('close', () => {\n this.clients.delete(ws);\n\n if (this._shouldEmitClose && !this.clients.size) {\n process.nextTick(emitClose, this);\n }\n });\n }\n\n cb(ws, req);\n }\n}\n\nmodule.exports = WebSocketServer;\n\n/**\n * Add event listeners on an `EventEmitter` using a map of <event, listener>\n * pairs.\n *\n * @param {EventEmitter} server The event emitter\n * @param {Object.<String, Function>} map The listeners to add\n * @return {Function} A function that will remove the added listeners when\n * called\n * @private\n */\nfunction addListeners(server, map) {\n for (const event of Object.keys(map)) server.on(event, map[event]);\n\n return function removeListeners() {\n for (const event of Object.keys(map)) {\n server.removeListener(event, map[event]);\n }\n };\n}\n\n/**\n * Emit a `'close'` event on an `EventEmitter`.\n *\n * @param {EventEmitter} server The event emitter\n * @private\n */\nfunction emitClose(server) {\n server._state = CLOSED;\n server.emit('close');\n}\n\n/**\n * Handle socket errors.\n *\n * @private\n */\nfunction socketOnError() {\n this.destroy();\n}\n\n/**\n * Close the connection when preconditions are not fulfilled.\n *\n * @param {Duplex} socket The socket of the upgrade request\n * @param {Number} code The HTTP response status code\n * @param {String} [message] The HTTP response body\n * @param {Object} [headers] Additional HTTP response headers\n * @private\n */\nfunction abortHandshake(socket, code, message, headers) {\n //\n // The socket is writable unless the user destroyed or ended it before calling\n // `server.handleUpgrade()` or in the `verifyClient` function, which is a user\n // error. Handling this does not make much sense as the worst that can happen\n // is that some of the data written by the user might be discarded due to the\n // call to `socket.end()` below, which triggers an `'error'` event that in\n // turn causes the socket to be destroyed.\n //\n message = message || http.STATUS_CODES[code];\n headers = {\n Connection: 'close',\n 'Content-Type': 'text/html',\n 'Content-Length': Buffer.byteLength(message),\n ...headers\n };\n\n socket.once('finish', socket.destroy);\n\n socket.end(\n `HTTP/1.1 ${code} ${http.STATUS_CODES[code]}\\r\\n` +\n Object.keys(headers)\n .map((h) => `${h}: ${headers[h]}`)\n .join('\\r\\n') +\n '\\r\\n\\r\\n' +\n message\n );\n}\n\n/**\n * Emit a `'wsClientError'` event on a `WebSocketServer` if there is at least\n * one listener for it, otherwise call `abortHandshake()`.\n *\n * @param {WebSocketServer} server The WebSocket server\n * @param {http.IncomingMessage} req The request object\n * @param {Duplex} socket The socket of the upgrade request\n * @param {Number} code The HTTP response status code\n * @param {String} message The HTTP response body\n * @param {Object} [headers] The HTTP response headers\n * @private\n */\nfunction abortHandshakeOrEmitwsClientError(\n server,\n req,\n socket,\n code,\n message,\n headers\n) {\n if (server.listenerCount('wsClientError')) {\n const err = new Error(message);\n Error.captureStackTrace(err, abortHandshakeOrEmitwsClientError);\n\n server.emit('wsClientError', err, socket, req);\n } else {\n abortHandshake(socket, code, message, headers);\n }\n}\n","import createWebSocketStream from './lib/stream.js';\nimport Receiver from './lib/receiver.js';\nimport Sender from './lib/sender.js';\nimport WebSocket from './lib/websocket.js';\nimport WebSocketServer from './lib/websocket-server.js';\n\nexport { createWebSocketStream, Receiver, Sender, WebSocket, WebSocketServer };\nexport default WebSocket;\n","/**\n * WebSocket-to-TCP Proxy for Cap'n Proto\n *\n * Allows browsers to connect to native Cap'n Proto services (C++, etc.)\n * via WebSocket. Handles the protocol bridging between WebSocket (browser)\n * and raw TCP (Cap'n Proto services).\n */\n\nimport { EventEmitter } from 'node:events';\nimport { type Socket, createConnection } from 'node:net';\nimport { WebSocket, WebSocketServer } from 'ws';\n\nexport interface ProxyOptions {\n /** WebSocket server port */\n wsPort: number;\n /** Target TCP host */\n targetHost: string;\n /** Target TCP port */\n targetPort: number;\n /** Maximum message size in bytes (default: 16MB) */\n maxMessageSize?: number;\n /** Connection timeout in ms (default: 30000) */\n connectionTimeout?: number;\n /** Enable debug logging */\n debug?: boolean;\n}\n\ninterface ConnectionStats {\n wsMessagesIn: number;\n wsMessagesOut: number;\n tcpBytesIn: number;\n tcpBytesOut: number;\n connectedAt: Date;\n}\n\nclass ProxyConnection extends EventEmitter {\n private ws: WebSocket;\n private tcpSocket: Socket | null = null;\n private stats: ConnectionStats;\n private options: ProxyOptions;\n private closed = false;\n\n constructor(ws: WebSocket, options: ProxyOptions) {\n super();\n this.ws = ws;\n this.options = options;\n this.stats = {\n wsMessagesIn: 0,\n wsMessagesOut: 0,\n tcpBytesIn: 0,\n tcpBytesOut: 0,\n connectedAt: new Date(),\n };\n\n this.setupWebSocket();\n }\n\n private setupWebSocket(): void {\n this.ws.on('message', (data: WebSocket.RawData) => {\n this.handleWebSocketMessage(data);\n });\n\n this.ws.on('close', () => {\n this.log('WebSocket closed');\n this.close();\n });\n\n this.ws.on('error', (err) => {\n this.log('WebSocket error:', err);\n this.emit('error', err);\n this.close();\n });\n\n // Connect to target TCP service\n this.connectToTarget();\n }\n\n private connectToTarget(): void {\n const timeout = this.options.connectionTimeout ?? 30000;\n\n this.tcpSocket = createConnection({\n host: this.options.targetHost,\n port: this.options.targetPort,\n timeout,\n });\n\n this.tcpSocket.on('connect', () => {\n this.log('Connected to target TCP service');\n this.emit('connected');\n });\n\n this.tcpSocket.on('data', (data) => {\n this.handleTcpData(data);\n });\n\n this.tcpSocket.on('close', () => {\n this.log('TCP connection closed');\n this.close();\n });\n\n this.tcpSocket.on('error', (err) => {\n this.log('TCP error:', err);\n this.emit('error', err);\n this.close();\n });\n\n this.tcpSocket.on('timeout', () => {\n this.log('TCP connection timeout');\n this.close();\n });\n }\n\n private handleWebSocketMessage(data: WebSocket.RawData): void {\n if (this.closed) return;\n\n // Handle both single Buffer and Buffer[]\n const buffer = Buffer.isBuffer(data)\n ? data\n : Array.isArray(data)\n ? Buffer.concat(data)\n : Buffer.from(data);\n\n const maxSize = this.options.maxMessageSize ?? 16 * 1024 * 1024;\n if (buffer.length > maxSize) {\n this.log(`Message too large: ${buffer.length} bytes`);\n this.close();\n return;\n }\n\n this.stats.wsMessagesIn++;\n this.stats.tcpBytesOut += buffer.length;\n\n // Forward to TCP service\n if (this.tcpSocket?.writable) {\n this.tcpSocket.write(buffer);\n this.log(`Forwarded ${buffer.length} bytes to TCP`);\n }\n }\n\n private handleTcpData(data: Buffer): void {\n if (this.closed) return;\n\n this.stats.tcpBytesIn += data.length;\n this.stats.wsMessagesOut++;\n\n // Forward to WebSocket client\n if (this.ws.readyState === WebSocket.OPEN) {\n this.ws.send(data);\n this.log(`Forwarded ${data.length} bytes to WebSocket`);\n }\n }\n\n private log(...args: unknown[]): void {\n if (this.options.debug) {\n console.log('[ProxyConnection]', ...args);\n }\n }\n\n getStats(): ConnectionStats {\n return { ...this.stats };\n }\n\n close(): void {\n if (this.closed) return;\n this.closed = true;\n\n this.ws.close();\n this.tcpSocket?.destroy();\n\n this.emit('closed');\n }\n}\n\nexport class CapnpWebSocketProxy extends EventEmitter {\n private wss: WebSocketServer;\n private connections = new Map<WebSocket, ProxyConnection>();\n private options: ProxyOptions;\n\n constructor(options: ProxyOptions) {\n super();\n this.options = options;\n this.wss = new WebSocketServer({ port: options.wsPort });\n this.setupServer();\n }\n\n private setupServer(): void {\n this.wss.on('connection', (ws, req) => {\n const clientIp = req.socket.remoteAddress;\n this.log(`New WebSocket connection from ${clientIp}`);\n\n const connection = new ProxyConnection(ws, this.options);\n this.connections.set(ws, connection);\n\n connection.on('connected', () => {\n this.emit('connection', connection);\n });\n\n connection.on('error', (err) => {\n this.emit('error', err, connection);\n });\n\n connection.on('closed', () => {\n this.connections.delete(ws);\n this.emit('disconnection', connection);\n });\n });\n\n this.wss.on('error', (err) => {\n this.emit('error', err);\n });\n }\n\n private log(...args: unknown[]): void {\n if (this.options.debug) {\n console.log('[CapnpWebSocketProxy]', ...args);\n }\n }\n\n getConnectionCount(): number {\n return this.connections.size;\n }\n\n getAllStats(): ConnectionStats[] {\n return Array.from(this.connections.values()).map((c) => c.getStats());\n }\n\n close(): Promise<void> {\n return new Promise((resolve) => {\n // Close all connections\n for (const connection of this.connections.values()) {\n connection.close();\n }\n this.connections.clear();\n\n // Close server\n this.wss.close(() => {\n this.emit('closed');\n resolve();\n });\n });\n }\n}\n\n// CLI entry point\nif (import.meta.url === `file://${process.argv[1]}`) {\n const args = process.argv.slice(2);\n\n // Parse CLI arguments\n let wsPort = 8080;\n let targetHost = 'localhost';\n let targetPort = 8081;\n let debug = false;\n\n for (let i = 0; i < args.length; i++) {\n switch (args[i]) {\n case '--ws-port':\n case '-p':\n wsPort = Number.parseInt(args[++i], 10);\n break;\n case '--target':\n case '-t': {\n const [host, port] = args[++i].split(':');\n targetHost = host;\n targetPort = Number.parseInt(port, 10);\n break;\n }\n case '--debug':\n case '-d':\n debug = true;\n break;\n case '--help':\n case '-h':\n console.log(`\nCap'n Proto WebSocket-to-TCP Proxy\n\nUsage: npx @naeemo/capnp proxy [options]\n\nOptions:\n -p, --ws-port <port> WebSocket server port (default: 8080)\n -t, --target <host:port> Target TCP service (default: localhost:8081)\n -d, --debug Enable debug logging\n -h, --help Show this help\n\nExample:\n npx @naeemo/capnp proxy -p 9000 -t 192.168.1.100:7000\n`);\n process.exit(0);\n }\n }\n\n const proxy = new CapnpWebSocketProxy({\n wsPort,\n targetHost,\n targetPort,\n debug,\n });\n\n console.log('WebSocket-to-TCP Proxy started');\n console.log(` WebSocket: ws://localhost:${wsPort}`);\n console.log(` Target: ${targetHost}:${targetPort}`);\n\n proxy.on('connection', () => {\n console.log(`Active connections: ${proxy.getConnectionCount()}`);\n });\n\n proxy.on('disconnection', () => {\n console.log(`Active connections: ${proxy.getConnectionCount()}`);\n });\n\n process.on('SIGINT', async () => {\n console.log('\\nShutting down...');\n await proxy.close();\n process.exit(0);\n });\n}\n","#!/usr/bin/env node\n/**\n * Cap'n Proto Benchmark CLI\n *\n * Usage: capnp bench [options]\n */\n\nimport { writeFileSync } from 'node:fs';\nimport * as os from 'node:os';\nimport { MessageBuilder, MessageReader } from './index.js';\n\nconst CLI_VERSION = '0.9.0';\n\n/**\n * 基准测试结果\n */\nexport interface BenchmarkResult {\n name: string;\n serializeTime: number; // μs\n deserializeTime: number; // μs\n dataSize: number; // bytes\n opsPerSecond: number;\n}\n\n/**\n * 测试环境信息\n */\nexport interface EnvironmentInfo {\n nodeVersion: string;\n platform: string;\n arch: string;\n cpu: string;\n cpus: number;\n totalMemory: string;\n freeMemory: string;\n}\n\n/**\n * 基准测试报告\n */\nexport interface BenchmarkReport {\n environment: EnvironmentInfo;\n results: BenchmarkResult[];\n sizes: string[];\n comparisons: string[];\n generatedAt: string;\n}\n\n// 高精度计时(微秒)\nfunction nowUs(): number {\n return Number(process.hrtime.bigint()) / 1000;\n}\n\n/**\n * 获取测试环境信息\n */\nfunction getEnvironmentInfo(): EnvironmentInfo {\n const cpus = os.cpus();\n const cpuModel = cpus.length > 0 ? cpus[0].model : 'Unknown';\n\n return {\n nodeVersion: process.version,\n platform: process.platform,\n arch: process.arch,\n cpu: cpuModel,\n cpus: cpus.length,\n totalMemory: formatBytes(os.totalmem()),\n freeMemory: formatBytes(os.freemem()),\n };\n}\n\n/**\n * 格式化字节大小\n */\nfunction formatBytes(bytes: number): string {\n if (bytes === 0) return '0 B';\n const k = 1024;\n const sizes = ['B', 'KB', 'MB', 'GB', 'TB'];\n const i = Math.floor(Math.log(bytes) / Math.log(k));\n return `${Number.parseFloat((bytes / k ** i).toFixed(2))} ${sizes[i]}`;\n}\n\n/**\n * 解析大小字符串(如 \"64B\", \"1KB\", \"1MB\")为字节数\n */\nfunction parseSize(sizeStr: string): number {\n const match = sizeStr.trim().match(/^(\\d+(?:\\.\\d+)?)\\s*(B|KB|MB|GB)?$/i);\n if (!match) {\n throw new Error(`Invalid size format: ${sizeStr}`);\n }\n const value = Number.parseFloat(match[1]);\n const unit = (match[2] || 'B').toUpperCase();\n const multipliers: Record<string, number> = {\n B: 1,\n KB: 1024,\n MB: 1024 * 1024,\n GB: 1024 * 1024 * 1024,\n };\n return Math.floor(value * (multipliers[unit] || 1));\n}\n\n/**\n * 生成测试数据\n */\nfunction generateTestData(sizeBytes: number): {\n id: number;\n name: string;\n data: number[];\n metadata: Record<string, unknown>;\n} {\n const dataArraySize = Math.max(1, Math.floor((sizeBytes - 200) / 4)); // 预留空间给其他字段\n return {\n id: Date.now(),\n name: `Test User ${Math.random().toString(36).substring(7)}`,\n data: Array.from({ length: Math.min(dataArraySize, 100000) }, (_, i) => i),\n metadata: {\n timestamp: new Date().toISOString(),\n version: '1.0.0',\n tags: ['benchmark', 'test', 'performance'],\n nested: {\n level1: {\n level2: {\n value: Math.random(),\n },\n },\n },\n },\n };\n}\n\n/**\n * 运行单次基准测试\n */\nfunction runBenchmark<T>(\n name: string,\n serialize: () => T,\n deserialize: (data: T) => unknown,\n getSize: (data: T) => number,\n iterations = 10000\n): BenchmarkResult {\n // 预热\n for (let i = 0; i < 100; i++) {\n const data = serialize();\n deserialize(data);\n }\n\n // 测试序列化\n const serializeStart = nowUs();\n const samples: T[] = [];\n for (let i = 0; i < iterations; i++) {\n samples.push(serialize());\n }\n const serializeTime = (nowUs() - serializeStart) / iterations;\n\n // 测试反序列化\n const deserializeStart = nowUs();\n for (const sample of samples) {\n deserialize(sample);\n }\n const deserializeTime = (nowUs() - deserializeStart) / iterations;\n\n // 数据大小\n const dataSize = getSize(samples[0]);\n\n // 总 ops/sec\n const totalTime = serializeTime + deserializeTime;\n const opsPerSecond = totalTime > 0 ? 1000000 / totalTime : 0;\n\n return {\n name,\n serializeTime,\n deserializeTime,\n dataSize,\n opsPerSecond,\n };\n}\n\n// ========== Cap'n Proto 序列化/反序列化 ==========\n\nfunction capnpSerialize(data: ReturnType<typeof generateTestData>): ArrayBuffer {\n const builder = new MessageBuilder();\n const root = builder.initRoot(2, 3);\n\n root.setInt32(0, data.id);\n\n // 文本字段\n root.setText(0, data.name);\n\n // 列表\n const dataList = root.initList(1, 4, data.data.length);\n for (let i = 0; i < data.data.length; i++) {\n dataList.setPrimitive(i, data.data[i]);\n }\n\n // 嵌套结构(简化处理)\n const metadata = root.initStruct(2, 1, 1);\n metadata.setText(0, data.metadata.timestamp as string);\n\n return builder.toArrayBuffer();\n}\n\nfunction capnpDeserialize(buffer: ArrayBuffer): boolean {\n const reader = new MessageReader(buffer);\n const root = reader.getRoot(2, 3);\n\n root.getInt32(0);\n root.getText(0);\n\n const list = root.getList(1, 4);\n if (list) {\n for (let i = 0; i < Math.min(list.length, 100); i++) {\n list.getPrimitive(i);\n }\n }\n\n const metadata = root.getStruct(2, 1, 1);\n if (metadata) {\n metadata.getText(0);\n }\n\n return true;\n}\n\n// ========== JSON 序列化/反序列化 ==========\n\nfunction jsonSerialize(data: ReturnType<typeof generateTestData>): string {\n return JSON.stringify(data);\n}\n\nfunction jsonDeserialize(jsonStr: string): unknown {\n return JSON.parse(jsonStr);\n}\n\n// ========== Protobuf 模拟(使用 Buffer) ==========\n\n// 简化的 Protobuf 模拟,实际项目中可以使用 protobufjs\nfunction protobufSerialize(data: ReturnType<typeof generateTestData>): Buffer {\n // 使用 BSON-like 的简单编码\n const header = Buffer.alloc(8);\n header.writeUInt32LE((data.id & 0xffffffff) >>> 0, 0);\n header.writeUInt32LE(data.data.length, 4);\n\n const nameBuf = Buffer.from(data.name, 'utf-8');\n const nameLen = Buffer.alloc(4);\n nameLen.writeUInt32LE(nameBuf.length, 0);\n\n const dataBuf = Buffer.from(new Int32Array(data.data).buffer);\n\n return Buffer.concat([header, nameLen, nameBuf, dataBuf]);\n}\n\nfunction protobufDeserialize(buffer: Buffer): boolean {\n let offset = 0;\n const _id = buffer.readUInt32LE(offset);\n offset += 4;\n const dataLen = buffer.readUInt32LE(offset);\n offset += 4;\n\n const nameLen = buffer.readUInt32LE(offset);\n offset += 4;\n offset += nameLen; // 跳过 name\n\n // 读取数据\n for (let i = 0; i < Math.min(dataLen, 100); i++) {\n buffer.readInt32LE(offset + i * 4);\n }\n\n return true;\n}\n\n/**\n * 运行所有基准测试\n */\nfunction runAllBenchmarks(\n sizes: string[],\n includeJson: boolean,\n includeProtobuf: boolean\n): BenchmarkResult[] {\n const results: BenchmarkResult[] = [];\n\n for (const sizeStr of sizes) {\n const sizeBytes = parseSize(sizeStr);\n const testData = generateTestData(sizeBytes);\n const iterations = Math.max(1000, Math.min(50000, Math.floor(10000000 / sizeBytes)));\n\n // Cap'n Proto\n const capnpResult = runBenchmark(\n `Cap'n Proto (${sizeStr})`,\n () => capnpSerialize(testData),\n capnpDeserialize,\n (data) => (data as ArrayBuffer).byteLength,\n iterations\n );\n results.push(capnpResult);\n\n // JSON\n if (includeJson) {\n const jsonResult = runBenchmark(\n `JSON (${sizeStr})`,\n () => jsonSerialize(testData),\n jsonDeserialize,\n (data) => new TextEncoder().encode(data as string).length,\n iterations\n );\n results.push(jsonResult);\n }\n\n // Protobuf\n if (includeProtobuf) {\n const protobufResult = runBenchmark(\n `Protobuf (${sizeStr})`,\n () => protobufSerialize(testData),\n protobufDeserialize,\n (data) => (data as Buffer).length,\n iterations\n );\n results.push(protobufResult);\n }\n }\n\n return results;\n}\n\n/**\n * 格式化基准测试报告为 Markdown\n */\nfunction formatMarkdownReport(report: BenchmarkReport): string {\n const lines: string[] = [];\n\n lines.push(\"# Cap'n Proto Benchmark Report\");\n lines.push('');\n lines.push(`Generated: ${report.generatedAt}`);\n lines.push('');\n\n // 环境信息\n lines.push('## Environment');\n lines.push('');\n lines.push(`- **Node.js**: ${report.environment.nodeVersion}`);\n lines.push(`- **Platform**: ${report.environment.platform} (${report.environment.arch})`);\n lines.push(`- **CPU**: ${report.environment.cpu}`);\n lines.push(`- **CPU Cores**: ${report.environment.cpus}`);\n lines.push(`- **Total Memory**: ${report.environment.totalMemory}`);\n lines.push(`- **Free Memory**: ${report.environment.freeMemory}`);\n lines.push('');\n\n // 结果表格\n lines.push('## Results');\n lines.push('');\n lines.push('| Format | Payload Size | Serialize (μs) | Deserialize (μs) | Data Size | Ops/Sec |');\n lines.push('|--------|--------------|----------------|------------------|-----------|---------|');\n\n for (const result of report.results) {\n lines.push(\n `| ${result.name.padEnd(20)} | ${formatBytes(result.dataSize).padStart(10)} | ` +\n `${result.serializeTime.toFixed(3).padStart(14)} | ${result.deserializeTime.toFixed(3).padStart(16)} | ` +\n `${result.dataSize.toString().padStart(9)} | ${result.opsPerSecond.toFixed(0).padStart(7)} |`\n );\n }\n\n lines.push('');\n\n // 对比分析\n if (report.comparisons.length > 1) {\n lines.push('## Comparison Summary');\n lines.push('');\n\n const groupedBySize = new Map<string, BenchmarkResult[]>();\n for (const result of report.results) {\n const sizeMatch = result.name.match(/\\(([^)]+)\\)/);\n const size = sizeMatch ? sizeMatch[1] : 'unknown';\n if (!groupedBySize.has(size)) {\n groupedBySize.set(size, []);\n }\n groupedBySize.get(size)!.push(result);\n }\n\n for (const [size, results] of groupedBySize) {\n if (results.length > 1) {\n lines.push(`### ${size} Payload`);\n lines.push('');\n\n // 找出最快的\n const fastest = results.reduce((a, b) => (a.opsPerSecond > b.opsPerSecond ? a : b));\n lines.push(\n `- **Fastest**: ${fastest.name.split(' ')[0]} (${fastest.opsPerSecond.toFixed(0)} ops/sec)`\n );\n\n // 找出最小的\n const smallest = results.reduce((a, b) => (a.dataSize < b.dataSize ? a : b));\n lines.push(\n `- **Smallest**: ${smallest.name.split(' ')[0]} (${formatBytes(smallest.dataSize)})`\n );\n\n // 计算相对性能\n const capnpResult = results.find((r) => r.name.includes(\"Cap'n Proto\"));\n if (capnpResult) {\n for (const result of results) {\n if (result !== capnpResult) {\n const format = result.name.split(' ')[0];\n const speedup = capnpResult.opsPerSecond / result.opsPerSecond;\n const sizeDiff = ((capnpResult.dataSize - result.dataSize) / result.dataSize) * 100;\n lines.push(\n `- vs ${format}: ${speedup > 1 ? `**${speedup.toFixed(2)}x faster**` : `${(1 / speedup).toFixed(2)}x slower`}, data size ${sizeDiff > 0 ? '+' : ''}${sizeDiff.toFixed(1)}%`\n );\n }\n }\n }\n lines.push('');\n }\n }\n }\n\n lines.push('## Conclusion');\n lines.push('');\n\n const capnpResults = report.results.filter((r) => r.name.includes(\"Cap'n Proto\"));\n if (capnpResults.length > 0) {\n const avgOps = capnpResults.reduce((sum, r) => sum + r.opsPerSecond, 0) / capnpResults.length;\n lines.push(\n `Cap'n Proto achieved an average throughput of **${avgOps.toFixed(0)} ops/sec** across all tested payload sizes.`\n );\n lines.push('');\n\n if (report.comparisons.length > 1) {\n lines.push('Based on the benchmark results:');\n lines.push('');\n lines.push(\"- Cap'n Proto offers competitive serialization performance\");\n lines.push('- Zero-copy architecture provides advantages for large payloads');\n lines.push('- Compact binary format reduces network overhead');\n }\n }\n\n lines.push('');\n lines.push('---');\n lines.push(`*Generated by Cap'n Proto Benchmark CLI v${CLI_VERSION}*`);\n\n return lines.join('\\n');\n}\n\n/**\n * 格式化为 CSV\n */\nfunction formatCsvReport(report: BenchmarkReport): string {\n const lines: string[] = [];\n lines.push('Format,Payload Size,Serialize (μs),Deserialize (μs),Data Size (bytes),Ops/Sec');\n\n for (const result of report.results) {\n lines.push(\n `${result.name},${result.dataSize},${result.serializeTime},${result.deserializeTime},${result.dataSize},${result.opsPerSecond}`\n );\n }\n\n return lines.join('\\n');\n}\n\n/**\n * 格式化报告为终端输出(类似 audit 命令风格)\n */\nfunction formatTerminalReport(report: BenchmarkReport): string {\n const lines: string[] = [];\n\n lines.push('='.repeat(70));\n lines.push(\"CAP'N PROTO BENCHMARK REPORT\");\n lines.push('='.repeat(70));\n lines.push('');\n\n // 环境信息\n lines.push('Environment:');\n lines.push(` Node.js: ${report.environment.nodeVersion}`);\n lines.push(` Platform: ${report.environment.platform} (${report.environment.arch})`);\n lines.push(` CPU: ${report.environment.cpu}`);\n lines.push(` CPU Cores: ${report.environment.cpus}`);\n lines.push(` Memory: ${report.environment.freeMemory} / ${report.environment.totalMemory}`);\n lines.push('');\n\n // 结果表格\n lines.push('Benchmark Results:');\n lines.push('-'.repeat(70));\n lines.push(\n `${'Format'.padEnd(22)} ${'Size'.padStart(10)} ${'Serialize'.padStart(12)} ${'Deserialize'.padStart(12)} ${'Ops/Sec'.padStart(12)}`\n );\n lines.push('-'.repeat(70));\n\n for (const result of report.results) {\n const format = result.name.padEnd(22);\n const size = formatBytes(result.dataSize).padStart(10);\n const ser = `${result.serializeTime.toFixed(2)} μs`.padStart(12);\n const deser = `${result.deserializeTime.toFixed(2)} μs`.padStart(12);\n const ops = result.opsPerSecond.toFixed(0).padStart(12);\n lines.push(`${format} ${size} ${ser} ${deser} ${ops}`);\n }\n\n lines.push('-'.repeat(70));\n lines.push('');\n\n // 对比总结\n if (report.comparisons.length > 1) {\n lines.push('Comparison Summary:');\n lines.push('');\n\n const groupedBySize = new Map<string, BenchmarkResult[]>();\n for (const result of report.results) {\n const sizeMatch = result.name.match(/\\(([^)]+)\\)/);\n const size = sizeMatch ? sizeMatch[1] : 'unknown';\n if (!groupedBySize.has(size)) {\n groupedBySize.set(size, []);\n }\n groupedBySize.get(size)!.push(result);\n }\n\n for (const [size, results] of groupedBySize) {\n if (results.length > 1) {\n lines.push(` ${size} Payload:`);\n\n const capnpResult = results.find((r) => r.name.includes(\"Cap'n Proto\"));\n if (capnpResult) {\n for (const result of results) {\n if (result !== capnpResult) {\n const format = result.name.split(' ')[0];\n const speedup = capnpResult.opsPerSecond / result.opsPerSecond;\n const icon = speedup > 1 ? '✅' : '⚠️';\n lines.push(\n ` ${icon} vs ${format}: ${speedup.toFixed(2)}x ${speedup > 1 ? 'faster' : 'slower'}`\n );\n }\n }\n }\n lines.push('');\n }\n }\n }\n\n lines.push('Conclusion:');\n const capnpResults = report.results.filter((r) => r.name.includes(\"Cap'n Proto\"));\n if (capnpResults.length > 0) {\n const avgOps = capnpResults.reduce((sum, r) => sum + r.opsPerSecond, 0) / capnpResults.length;\n lines.push(` Cap'n Proto average: ${avgOps.toFixed(0)} ops/sec`);\n }\n lines.push('');\n lines.push('='.repeat(70));\n\n return lines.join('\\n');\n}\n\nfunction printUsage() {\n console.log(`\nCap'n Proto Benchmark CLI v${CLI_VERSION}\n\nUsage: capnp bench [options]\n\nOptions:\n --sizes <list> Comma-separated payload sizes (default: 1KB,10KB,100KB)\n Examples: 64B,1KB,1MB or 100B,500B,1KB,10KB\n --vs-json Include JSON comparison\n --vs-protobuf Include Protobuf comparison\n -o, --output <file> Write report to file\n --format <format> Output format: markdown, json, csv (default: terminal)\n -h, --help Show this help\n\nExamples:\n capnp bench\n capnp bench --sizes 64B,1KB,1MB --vs-json\n capnp bench --sizes 100KB --vs-json --vs-protobuf --output report.md --format markdown\n capnp bench --sizes 10KB,100KB --vs-json --format json -o results.json\n`);\n}\n\nfunction parseArgs(args: string[]) {\n const options: {\n sizes?: string[];\n vsJson?: boolean;\n vsProtobuf?: boolean;\n output?: string;\n format?: 'markdown' | 'json' | 'csv' | 'terminal';\n } = {};\n\n for (let i = 0; i < args.length; i++) {\n const arg = args[i];\n\n if (arg === '-h' || arg === '--help') {\n printUsage();\n process.exit(0);\n }\n\n if (arg === '--sizes') {\n const sizesStr = args[++i];\n if (!sizesStr) {\n console.error('Error: --sizes requires a value');\n process.exit(1);\n }\n options.sizes = sizesStr.split(',').map((s) => s.trim());\n } else if (arg === '--vs-json') {\n options.vsJson = true;\n } else if (arg === '--vs-protobuf') {\n options.vsProtobuf = true;\n } else if (arg === '-o' || arg === '--output') {\n options.output = args[++i];\n } else if (arg === '--format') {\n const format = args[++i];\n if (format !== 'markdown' && format !== 'json' && format !== 'csv' && format !== 'terminal') {\n console.error('Error: --format must be one of: markdown, json, csv, terminal');\n process.exit(1);\n }\n options.format = format;\n }\n }\n\n // 默认值\n if (!options.sizes) {\n options.sizes = ['1KB', '10KB', '100KB'];\n }\n if (!options.format) {\n options.format = options.output ? 'markdown' : 'terminal';\n }\n\n return options;\n}\n\nexport async function run(args: string[]): Promise<void> {\n const options = parseArgs(args);\n\n console.log(\"Running Cap'n Proto Benchmark...\");\n console.log('');\n\n const env = getEnvironmentInfo();\n console.log(`Environment: Node.js ${env.nodeVersion} on ${env.platform} (${env.arch})`);\n console.log(`CPU: ${env.cpu} (${env.cpus} cores)`);\n console.log('');\n\n // 运行基准测试\n const comparisons: string[] = [\"Cap'n Proto\"];\n if (options.vsJson) comparisons.push('JSON');\n if (options.vsProtobuf) comparisons.push('Protobuf');\n\n console.log(`Testing payload sizes: ${options.sizes?.join(', ')}`);\n console.log(`Comparisons: ${comparisons.join(', ')}`);\n console.log('');\n\n const results = runAllBenchmarks(\n options.sizes || ['1KB', '10KB', '100KB'],\n options.vsJson || false,\n options.vsProtobuf || false\n );\n\n const report: BenchmarkReport = {\n environment: env,\n results,\n sizes: options.sizes || ['1KB', '10KB', '100KB'],\n comparisons,\n generatedAt: new Date().toISOString(),\n };\n\n // 格式化输出\n let output: string;\n switch (options.format) {\n case 'json':\n output = JSON.stringify(report, null, 2);\n break;\n case 'csv':\n output = formatCsvReport(report);\n break;\n case 'markdown':\n output = formatMarkdownReport(report);\n break;\n default:\n output = formatTerminalReport(report);\n break;\n }\n\n if (options.output) {\n writeFileSync(options.output, output, 'utf-8');\n console.log(`Report written to: ${options.output}`);\n } else {\n console.log(output);\n }\n\n process.exit(0);\n}\n"],"x_google_ignoreList":[5,6,7,8,9,10,11,12,13,14,15,16,17,18],"mappings":"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"}