2023-02-15 10:29:18 +00:00
|
|
|
/**
|
|
|
|
* Copyright 2023 Google Inc. All rights reserved.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2023-02-15 23:09:31 +00:00
|
|
|
import * as Bidi from 'chromium-bidi/lib/cjs/protocol/protocol.js';
|
|
|
|
|
2023-02-15 10:29:18 +00:00
|
|
|
import {ElementHandle} from '../../api/ElementHandle.js';
|
2023-02-15 23:09:31 +00:00
|
|
|
import {JSHandle as BaseJSHandle} from '../../api/JSHandle.js';
|
2023-02-15 10:29:18 +00:00
|
|
|
import {EvaluateFuncWith, HandleFor, HandleOr} from '../../common/types.js';
|
2023-05-10 08:23:29 +00:00
|
|
|
import {withSourcePuppeteerURLIfNone} from '../util.js';
|
2023-02-15 23:09:31 +00:00
|
|
|
|
|
|
|
import {Connection} from './Connection.js';
|
2023-02-28 10:10:14 +00:00
|
|
|
import {Context} from './Context.js';
|
2023-02-15 10:29:18 +00:00
|
|
|
import {BidiSerializer} from './Serializer.js';
|
2023-02-15 23:09:31 +00:00
|
|
|
import {releaseReference} from './utils.js';
|
2023-02-15 10:29:18 +00:00
|
|
|
|
|
|
|
export class JSHandle<T = unknown> extends BaseJSHandle<T> {
|
|
|
|
#disposed = false;
|
|
|
|
#context;
|
|
|
|
#remoteValue;
|
|
|
|
|
2023-02-28 10:10:14 +00:00
|
|
|
constructor(context: Context, remoteValue: Bidi.CommonDataTypes.RemoteValue) {
|
2023-02-15 10:29:18 +00:00
|
|
|
super();
|
|
|
|
this.#context = context;
|
|
|
|
this.#remoteValue = remoteValue;
|
|
|
|
}
|
|
|
|
|
2023-02-28 10:10:14 +00:00
|
|
|
context(): Context {
|
2023-02-15 10:29:18 +00:00
|
|
|
return this.#context;
|
|
|
|
}
|
|
|
|
|
2023-02-28 10:10:14 +00:00
|
|
|
get connection(): Connection {
|
2023-02-15 10:29:18 +00:00
|
|
|
return this.#context.connection;
|
|
|
|
}
|
|
|
|
|
|
|
|
override get disposed(): boolean {
|
|
|
|
return this.#disposed;
|
|
|
|
}
|
|
|
|
|
|
|
|
override async evaluate<
|
|
|
|
Params extends unknown[],
|
|
|
|
Func extends EvaluateFuncWith<T, Params> = EvaluateFuncWith<T, Params>
|
|
|
|
>(
|
|
|
|
pageFunction: Func | string,
|
|
|
|
...args: Params
|
|
|
|
): Promise<Awaited<ReturnType<Func>>> {
|
2023-05-10 08:23:29 +00:00
|
|
|
pageFunction = withSourcePuppeteerURLIfNone(
|
|
|
|
this.evaluate.name,
|
|
|
|
pageFunction
|
|
|
|
);
|
2023-02-15 10:29:18 +00:00
|
|
|
return await this.context().evaluate(pageFunction, this, ...args);
|
|
|
|
}
|
|
|
|
|
|
|
|
override async evaluateHandle<
|
|
|
|
Params extends unknown[],
|
|
|
|
Func extends EvaluateFuncWith<T, Params> = EvaluateFuncWith<T, Params>
|
|
|
|
>(
|
|
|
|
pageFunction: Func | string,
|
|
|
|
...args: Params
|
|
|
|
): Promise<HandleFor<Awaited<ReturnType<Func>>>> {
|
2023-05-10 08:23:29 +00:00
|
|
|
pageFunction = withSourcePuppeteerURLIfNone(
|
|
|
|
this.evaluateHandle.name,
|
|
|
|
pageFunction
|
|
|
|
);
|
2023-02-15 10:29:18 +00:00
|
|
|
return await this.context().evaluateHandle(pageFunction, this, ...args);
|
|
|
|
}
|
|
|
|
|
|
|
|
override async getProperty<K extends keyof T>(
|
|
|
|
propertyName: HandleOr<K>
|
|
|
|
): Promise<HandleFor<T[K]>>;
|
|
|
|
override async getProperty(propertyName: string): Promise<HandleFor<unknown>>;
|
|
|
|
override async getProperty<K extends keyof T>(
|
|
|
|
propertyName: HandleOr<K>
|
|
|
|
): Promise<HandleFor<T[K]>> {
|
|
|
|
return await this.evaluateHandle((object, propertyName) => {
|
|
|
|
return object[propertyName as K];
|
|
|
|
}, propertyName);
|
|
|
|
}
|
|
|
|
|
|
|
|
override async getProperties(): Promise<Map<string, BaseJSHandle>> {
|
|
|
|
// TODO(lightning00blade): Either include return of depth Handles in RemoteValue
|
|
|
|
// or new BiDi command that returns array of remote value
|
|
|
|
const keys = await this.evaluate(object => {
|
|
|
|
return Object.getOwnPropertyNames(object);
|
|
|
|
});
|
|
|
|
const map: Map<string, BaseJSHandle> = new Map();
|
|
|
|
const results = await Promise.all(
|
|
|
|
keys.map(key => {
|
|
|
|
return this.getProperty(key);
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
for (const [key, value] of Object.entries(keys)) {
|
|
|
|
const handle = results[key as any];
|
|
|
|
if (handle) {
|
|
|
|
map.set(value, handle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
override async jsonValue(): Promise<T> {
|
2023-02-20 12:00:29 +00:00
|
|
|
const value = BidiSerializer.deserialize(this.#remoteValue);
|
|
|
|
|
|
|
|
if (this.#remoteValue.type !== 'undefined' && value === undefined) {
|
2023-02-15 10:29:18 +00:00
|
|
|
throw new Error('Could not serialize referenced object');
|
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
override asElement(): ElementHandle<Node> | null {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
override async dispose(): Promise<void> {
|
|
|
|
if (this.#disposed) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.#disposed = true;
|
|
|
|
if ('handle' in this.#remoteValue) {
|
2023-02-28 10:10:14 +00:00
|
|
|
await releaseReference(this.#context, this.#remoteValue);
|
2023-02-15 10:29:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-20 12:00:29 +00:00
|
|
|
get isPrimitiveValue(): boolean {
|
|
|
|
switch (this.#remoteValue.type) {
|
|
|
|
case 'string':
|
|
|
|
case 'number':
|
|
|
|
case 'bigint':
|
|
|
|
case 'boolean':
|
|
|
|
case 'undefined':
|
|
|
|
case 'null':
|
|
|
|
return true;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-15 10:29:18 +00:00
|
|
|
override toString(): string {
|
2023-02-20 12:00:29 +00:00
|
|
|
if (this.isPrimitiveValue) {
|
2023-02-15 10:29:18 +00:00
|
|
|
return 'JSHandle:' + BidiSerializer.deserialize(this.#remoteValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 'JSHandle@' + this.#remoteValue.type;
|
|
|
|
}
|
|
|
|
|
|
|
|
override get id(): string | undefined {
|
|
|
|
return 'handle' in this.#remoteValue ? this.#remoteValue.handle : undefined;
|
|
|
|
}
|
|
|
|
|
2023-02-28 10:10:14 +00:00
|
|
|
remoteValue(): Bidi.CommonDataTypes.RemoteValue {
|
2023-02-15 10:29:18 +00:00
|
|
|
return this.#remoteValue;
|
|
|
|
}
|
|
|
|
}
|