mirror of
https://github.com/microsoft/playwright.git
synced 2025-06-26 21:40:17 +00:00
95 lines
2.9 KiB
TypeScript
95 lines
2.9 KiB
TypeScript
/**
|
|
* Copyright 2018 Google Inc. All rights reserved.
|
|
* Modifications copyright (c) Microsoft Corporation.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
import { makeWaitForNextTask } from '../utils';
|
|
import { debugLogger } from './utils/debugLogger';
|
|
|
|
import type { ConnectionTransport, ProtocolRequest, ProtocolResponse } from './transport';
|
|
|
|
export class PipeTransport implements ConnectionTransport {
|
|
private _pipeRead: NodeJS.ReadableStream;
|
|
private _pipeWrite: NodeJS.WritableStream;
|
|
private _pendingBuffers: Buffer[] = [];
|
|
private _waitForNextTask = makeWaitForNextTask();
|
|
private _closed = false;
|
|
private _onclose?: (reason?: string) => void;
|
|
|
|
onmessage?: (message: ProtocolResponse) => void;
|
|
|
|
constructor(pipeWrite: NodeJS.WritableStream, pipeRead: NodeJS.ReadableStream) {
|
|
this._pipeRead = pipeRead;
|
|
this._pipeWrite = pipeWrite;
|
|
pipeRead.on('data', buffer => this._dispatch(buffer));
|
|
pipeRead.on('close', () => {
|
|
this._closed = true;
|
|
if (this._onclose)
|
|
this._onclose.call(null);
|
|
});
|
|
pipeRead.on('error', e => debugLogger.log('error', e));
|
|
pipeWrite.on('error', e => debugLogger.log('error', e));
|
|
this.onmessage = undefined;
|
|
}
|
|
|
|
get onclose() {
|
|
return this._onclose;
|
|
}
|
|
|
|
set onclose(onclose: undefined | ((reason?: string) => void)) {
|
|
this._onclose = onclose;
|
|
if (onclose && !this._pipeRead.readable)
|
|
onclose();
|
|
}
|
|
|
|
send(message: ProtocolRequest) {
|
|
if (this._closed)
|
|
throw new Error('Pipe has been closed');
|
|
this._pipeWrite.write(JSON.stringify(message));
|
|
this._pipeWrite.write('\0');
|
|
}
|
|
|
|
close() {
|
|
throw new Error('unimplemented');
|
|
}
|
|
|
|
_dispatch(buffer: Buffer) {
|
|
let end = buffer.indexOf('\0');
|
|
if (end === -1) {
|
|
this._pendingBuffers.push(buffer);
|
|
return;
|
|
}
|
|
this._pendingBuffers.push(buffer.slice(0, end));
|
|
const message = Buffer.concat(this._pendingBuffers).toString();
|
|
this._waitForNextTask(() => {
|
|
if (this.onmessage)
|
|
this.onmessage.call(null, JSON.parse(message));
|
|
});
|
|
|
|
let start = end + 1;
|
|
end = buffer.indexOf('\0', start);
|
|
while (end !== -1) {
|
|
const message = buffer.toString(undefined, start, end);
|
|
this._waitForNextTask(() => {
|
|
if (this.onmessage)
|
|
this.onmessage.call(null, JSON.parse(message));
|
|
});
|
|
start = end + 1;
|
|
end = buffer.indexOf('\0', start);
|
|
}
|
|
this._pendingBuffers = [buffer.slice(start)];
|
|
}
|
|
}
|