2021-12-10 02:21:17 +01:00
|
|
|
/**
|
|
|
|
|
* 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 { ChannelOwner } from './channelOwner';
|
2025-02-11 00:04:33 +01:00
|
|
|
import { Connection } from './connection';
|
2025-02-13 04:27:24 +01:00
|
|
|
import * as localUtils from '../common/localUtils';
|
2025-02-07 22:54:01 +01:00
|
|
|
|
2025-02-11 17:28:28 +01:00
|
|
|
import type { HeadersArray, Size } from './types';
|
2025-02-13 04:27:24 +01:00
|
|
|
import type { HarBackend } from '../common/harBackend';
|
|
|
|
|
import type { Platform } from '../common/platform';
|
2025-02-07 22:54:01 +01:00
|
|
|
import type * as channels from '@protocol/channels';
|
2023-10-05 01:48:54 +02:00
|
|
|
|
|
|
|
|
type DeviceDescriptor = {
|
|
|
|
|
userAgent: string,
|
|
|
|
|
viewport: Size,
|
|
|
|
|
deviceScaleFactor: number,
|
|
|
|
|
isMobile: boolean,
|
|
|
|
|
hasTouch: boolean,
|
|
|
|
|
defaultBrowserType: 'chromium' | 'firefox' | 'webkit'
|
|
|
|
|
};
|
|
|
|
|
type Devices = { [name: string]: DeviceDescriptor };
|
2021-12-10 02:21:17 +01:00
|
|
|
|
|
|
|
|
export class LocalUtils extends ChannelOwner<channels.LocalUtilsChannel> {
|
2023-10-05 01:48:54 +02:00
|
|
|
readonly devices: Devices;
|
2025-02-10 23:19:58 +01:00
|
|
|
private _harBackends = new Map<string, HarBackend>();
|
|
|
|
|
private _stackSessions = new Map<string, localUtils.StackSession>();
|
2023-10-05 01:48:54 +02:00
|
|
|
|
2021-12-10 02:21:17 +01:00
|
|
|
constructor(parent: ChannelOwner, type: string, guid: string, initializer: channels.LocalUtilsInitializer) {
|
|
|
|
|
super(parent, type, guid, initializer);
|
2024-09-21 19:17:59 +02:00
|
|
|
this.markAsInternalType();
|
2023-10-05 01:48:54 +02:00
|
|
|
this.devices = {};
|
|
|
|
|
for (const { name, descriptor } of initializer.deviceDescriptors)
|
|
|
|
|
this.devices[name] = descriptor;
|
2021-12-10 02:21:17 +01:00
|
|
|
}
|
2025-02-10 23:19:58 +01:00
|
|
|
|
|
|
|
|
async zip(params: channels.LocalUtilsZipParams): Promise<void> {
|
|
|
|
|
return await localUtils.zip(this._platform, this._stackSessions, params);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async harOpen(params: channels.LocalUtilsHarOpenParams): Promise<channels.LocalUtilsHarOpenResult> {
|
2025-02-12 23:43:52 +01:00
|
|
|
return await localUtils.harOpen(this._platform, this._harBackends, params);
|
2025-02-10 23:19:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async harLookup(params: channels.LocalUtilsHarLookupParams): Promise<channels.LocalUtilsHarLookupResult> {
|
|
|
|
|
return await localUtils.harLookup(this._harBackends, params);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async harClose(params: channels.LocalUtilsHarCloseParams): Promise<void> {
|
|
|
|
|
return await localUtils.harClose(this._harBackends, params);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async harUnzip(params: channels.LocalUtilsHarUnzipParams): Promise<void> {
|
|
|
|
|
return await localUtils.harUnzip(params);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async tracingStarted(params: channels.LocalUtilsTracingStartedParams): Promise<channels.LocalUtilsTracingStartedResult> {
|
|
|
|
|
return await localUtils.tracingStarted(this._stackSessions, params);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async traceDiscarded(params: channels.LocalUtilsTraceDiscardedParams): Promise<void> {
|
|
|
|
|
return await localUtils.traceDiscarded(this._platform, this._stackSessions, params);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async addStackToTracingNoReply(params: channels.LocalUtilsAddStackToTracingNoReplyParams): Promise<void> {
|
|
|
|
|
return await localUtils.addStackToTracingNoReply(this._stackSessions, params);
|
|
|
|
|
}
|
|
|
|
|
|
2025-02-11 00:04:33 +01:00
|
|
|
async connect(params: channels.LocalUtilsConnectParams): Promise<Connection> {
|
2025-02-11 17:28:28 +01:00
|
|
|
const transport = this._platform.ws ? new WebSocketTransport(this._platform) : new JsonPipeTransport(this);
|
|
|
|
|
const connectHeaders = await transport.connect(params);
|
2025-02-11 00:04:33 +01:00
|
|
|
const connection = new Connection(this, this._platform, this._instrumentation, connectHeaders);
|
|
|
|
|
connection.markAsRemote();
|
2025-02-11 17:28:28 +01:00
|
|
|
connection.on('close', () => transport.close());
|
2025-02-11 00:04:33 +01:00
|
|
|
|
|
|
|
|
let closeError: string | undefined;
|
2025-02-11 17:28:28 +01:00
|
|
|
const onTransportClosed = (reason?: string) => {
|
2025-02-11 00:04:33 +01:00
|
|
|
connection.close(reason || closeError);
|
|
|
|
|
};
|
2025-02-11 17:28:28 +01:00
|
|
|
transport.onClose(reason => onTransportClosed(reason));
|
|
|
|
|
connection.onmessage = message => transport.send(message).catch(() => onTransportClosed());
|
|
|
|
|
transport.onMessage(message => {
|
2025-02-11 00:04:33 +01:00
|
|
|
try {
|
|
|
|
|
connection!.dispatch(message);
|
|
|
|
|
} catch (e) {
|
|
|
|
|
closeError = String(e);
|
2025-02-11 17:28:28 +01:00
|
|
|
transport.close();
|
2025-02-11 00:04:33 +01:00
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
return connection;
|
2025-02-10 23:19:58 +01:00
|
|
|
}
|
2021-12-10 02:21:17 +01:00
|
|
|
}
|
2025-02-11 17:28:28 +01:00
|
|
|
interface Transport {
|
|
|
|
|
connect(params: channels.LocalUtilsConnectParams): Promise<HeadersArray>;
|
|
|
|
|
send(message: any): Promise<void>;
|
|
|
|
|
onMessage(callback: (message: object) => void): void;
|
|
|
|
|
onClose(callback: (reason?: string) => void): void;
|
|
|
|
|
close(): Promise<void>;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class JsonPipeTransport implements Transport {
|
|
|
|
|
private _pipe: channels.JsonPipeChannel | undefined;
|
|
|
|
|
private _owner: ChannelOwner<channels.LocalUtilsChannel>;
|
|
|
|
|
|
|
|
|
|
constructor(owner: ChannelOwner<channels.LocalUtilsChannel>) {
|
|
|
|
|
this._owner = owner;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async connect(params: channels.LocalUtilsConnectParams) {
|
|
|
|
|
const { pipe, headers: connectHeaders } = await this._owner._wrapApiCall(async () => {
|
|
|
|
|
return await this._owner._channel.connect(params);
|
|
|
|
|
}, /* isInternal */ true);
|
|
|
|
|
this._pipe = pipe;
|
|
|
|
|
return connectHeaders;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async send(message: object) {
|
|
|
|
|
this._owner._wrapApiCall(async () => {
|
|
|
|
|
await this._pipe!.send({ message });
|
|
|
|
|
}, /* isInternal */ true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onMessage(callback: (message: object) => void) {
|
|
|
|
|
this._pipe!.on('message', ({ message }) => callback(message));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onClose(callback: (reason?: string) => void) {
|
|
|
|
|
this._pipe!.on('closed', ({ reason }) => callback(reason));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async close() {
|
|
|
|
|
await this._owner._wrapApiCall(async () => {
|
|
|
|
|
await this._pipe!.close().catch(() => {});
|
|
|
|
|
}, /* isInternal */ true);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class WebSocketTransport implements Transport {
|
|
|
|
|
private _platform: Platform;
|
|
|
|
|
private _ws: WebSocket | undefined;
|
|
|
|
|
|
|
|
|
|
constructor(platform: Platform) {
|
|
|
|
|
this._platform = platform;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async connect(params: channels.LocalUtilsConnectParams) {
|
|
|
|
|
this._ws = this._platform.ws!(params.wsEndpoint);
|
|
|
|
|
return [];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async send(message: object) {
|
|
|
|
|
this._ws!.send(JSON.stringify(message));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onMessage(callback: (message: object) => void) {
|
|
|
|
|
this._ws!.addEventListener('message', event => callback(JSON.parse(event.data)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onClose(callback: (reason?: string) => void) {
|
|
|
|
|
this._ws!.addEventListener('close', () => callback());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async close() {
|
|
|
|
|
this._ws!.close();
|
|
|
|
|
}
|
|
|
|
|
}
|