/** * @license * Copyright 2018 Google Inc. * SPDX-License-Identifier: Apache-2.0 */ import type {ConnectionTransport} from '../common/ConnectionTransport.js'; import {EventEmitter} from '../common/EventEmitter.js'; import {debugError} from '../common/util.js'; import {assert} from '../util/assert.js'; import {DisposableStack} from '../util/disposable.js'; /** * @internal */ export class PipeTransport implements ConnectionTransport { #pipeWrite: NodeJS.WritableStream; #subscriptions = new DisposableStack(); #isClosed = false; #pendingMessage = ''; onclose?: () => void; onmessage?: (value: string) => void; constructor( pipeWrite: NodeJS.WritableStream, pipeRead: NodeJS.ReadableStream ) { this.#pipeWrite = pipeWrite; const pipeReadEmitter = this.#subscriptions.use( // NodeJS event emitters don't support `*` so we need to typecast // As long as we don't use it we should be OK. new EventEmitter(pipeRead as unknown as EventEmitter<Record<string, any>>) ); pipeReadEmitter.on('data', (buffer: Buffer) => { return this.#dispatch(buffer); }); pipeReadEmitter.on('close', () => { if (this.onclose) { this.onclose.call(null); } }); pipeReadEmitter.on('error', debugError); const pipeWriteEmitter = this.#subscriptions.use( // NodeJS event emitters don't support `*` so we need to typecast // As long as we don't use it we should be OK. new EventEmitter(pipeRead as unknown as EventEmitter<Record<string, any>>) ); pipeWriteEmitter.on('error', debugError); } send(message: string): void { assert(!this.#isClosed, '`PipeTransport` is closed.'); this.#pipeWrite.write(message); this.#pipeWrite.write('\0'); } #dispatch(buffer: Buffer): void { assert(!this.#isClosed, '`PipeTransport` is closed.'); let end = buffer.indexOf('\0'); if (end === -1) { this.#pendingMessage += buffer.toString(); return; } const message = this.#pendingMessage + buffer.toString(undefined, 0, end); if (this.onmessage) { this.onmessage.call(null, message); } let start = end + 1; end = buffer.indexOf('\0', start); while (end !== -1) { if (this.onmessage) { this.onmessage.call(null, buffer.toString(undefined, start, end)); } start = end + 1; end = buffer.indexOf('\0', start); } this.#pendingMessage = buffer.toString(undefined, start); } close(): void { this.#isClosed = true; this.#subscriptions.dispose(); } }