2020-05-13 13:57:21 +00:00
|
|
|
/**
|
|
|
|
* Copyright 2020 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 {Protocol} from 'devtools-protocol';
|
2021-11-23 07:19:14 +00:00
|
|
|
|
2023-03-10 15:59:02 +00:00
|
|
|
import {CDPSession} from './Connection.js';
|
2023-02-15 23:09:31 +00:00
|
|
|
import {ProtocolError} from './Errors.js';
|
2022-08-17 15:45:45 +00:00
|
|
|
import {Frame} from './Frame.js';
|
2022-06-22 13:25:44 +00:00
|
|
|
import {HTTPRequest} from './HTTPRequest.js';
|
|
|
|
import {SecurityDetails} from './SecurityDetails.js';
|
2020-05-13 13:57:21 +00:00
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @public
|
|
|
|
*/
|
|
|
|
export interface RemoteAddress {
|
2022-05-05 06:48:03 +00:00
|
|
|
ip?: string;
|
|
|
|
port?: number;
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* The HTTPResponse class represents responses which are received by the
|
|
|
|
* {@link Page} class.
|
|
|
|
*
|
|
|
|
* @public
|
|
|
|
*/
|
2020-05-29 10:49:30 +00:00
|
|
|
export class HTTPResponse {
|
2022-06-13 09:16:25 +00:00
|
|
|
#client: CDPSession;
|
|
|
|
#request: HTTPRequest;
|
|
|
|
#contentPromise: Promise<Buffer> | null = null;
|
|
|
|
#bodyLoadedPromise: Promise<Error | void>;
|
|
|
|
#bodyLoadedPromiseFulfill: (err: Error | void) => void = () => {};
|
|
|
|
#remoteAddress: RemoteAddress;
|
|
|
|
#status: number;
|
|
|
|
#statusText: string;
|
|
|
|
#url: string;
|
|
|
|
#fromDiskCache: boolean;
|
|
|
|
#fromServiceWorker: boolean;
|
|
|
|
#headers: Record<string, string> = {};
|
|
|
|
#securityDetails: SecurityDetails | null;
|
|
|
|
#timing: Protocol.Network.ResourceTiming | null;
|
2020-05-13 13:57:21 +00:00
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
constructor(
|
|
|
|
client: CDPSession,
|
2020-05-29 08:38:40 +00:00
|
|
|
request: HTTPRequest,
|
2021-11-23 07:19:14 +00:00
|
|
|
responsePayload: Protocol.Network.Response,
|
|
|
|
extraInfo: Protocol.Network.ResponseReceivedExtraInfoEvent | null
|
2020-05-13 13:57:21 +00:00
|
|
|
) {
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#client = client;
|
|
|
|
this.#request = request;
|
2020-05-13 13:57:21 +00:00
|
|
|
|
2022-06-22 13:25:44 +00:00
|
|
|
this.#bodyLoadedPromise = new Promise(fulfill => {
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#bodyLoadedPromiseFulfill = fulfill;
|
2020-05-13 13:57:21 +00:00
|
|
|
});
|
|
|
|
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#remoteAddress = {
|
2020-05-13 13:57:21 +00:00
|
|
|
ip: responsePayload.remoteIPAddress,
|
|
|
|
port: responsePayload.remotePort,
|
|
|
|
};
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#statusText =
|
|
|
|
this.#parseStatusTextFromExtrInfo(extraInfo) ||
|
2021-11-26 08:17:34 +00:00
|
|
|
responsePayload.statusText;
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#url = request.url();
|
|
|
|
this.#fromDiskCache = !!responsePayload.fromDiskCache;
|
|
|
|
this.#fromServiceWorker = !!responsePayload.fromServiceWorker;
|
2021-11-23 07:19:14 +00:00
|
|
|
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#status = extraInfo ? extraInfo.statusCode : responsePayload.status;
|
2021-11-23 07:19:14 +00:00
|
|
|
const headers = extraInfo ? extraInfo.headers : responsePayload.headers;
|
2022-05-31 14:34:16 +00:00
|
|
|
for (const [key, value] of Object.entries(headers)) {
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#headers[key.toLowerCase()] = value;
|
2022-05-31 14:34:16 +00:00
|
|
|
}
|
2021-11-23 07:19:14 +00:00
|
|
|
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#securityDetails = responsePayload.securityDetails
|
2020-05-13 13:57:21 +00:00
|
|
|
? new SecurityDetails(responsePayload.securityDetails)
|
|
|
|
: null;
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#timing = responsePayload.timing || null;
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2022-06-13 09:16:25 +00:00
|
|
|
#parseStatusTextFromExtrInfo(
|
2021-11-26 08:17:34 +00:00
|
|
|
extraInfo: Protocol.Network.ResponseReceivedExtraInfoEvent | null
|
|
|
|
): string | undefined {
|
2022-06-14 11:55:35 +00:00
|
|
|
if (!extraInfo || !extraInfo.headersText) {
|
|
|
|
return;
|
|
|
|
}
|
2021-11-26 08:17:34 +00:00
|
|
|
const firstLine = extraInfo.headersText.split('\r', 1)[0];
|
2022-06-14 11:55:35 +00:00
|
|
|
if (!firstLine) {
|
|
|
|
return;
|
|
|
|
}
|
2021-11-26 08:17:34 +00:00
|
|
|
const match = firstLine.match(/[^ ]* [^ ]* (.*)/);
|
2022-06-14 11:55:35 +00:00
|
|
|
if (!match) {
|
|
|
|
return;
|
|
|
|
}
|
2021-11-26 08:17:34 +00:00
|
|
|
const statusText = match[1];
|
2022-06-14 11:55:35 +00:00
|
|
|
if (!statusText) {
|
|
|
|
return;
|
|
|
|
}
|
2021-11-26 08:17:34 +00:00
|
|
|
return statusText;
|
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
_resolveBody(err: Error | null): void {
|
2022-05-05 06:48:03 +00:00
|
|
|
if (err) {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#bodyLoadedPromiseFulfill(err);
|
2022-05-05 06:48:03 +00:00
|
|
|
}
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#bodyLoadedPromiseFulfill();
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @returns The IP address and port number used to connect to the remote
|
|
|
|
* server.
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
remoteAddress(): RemoteAddress {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#remoteAddress;
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @returns The URL of the response.
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
url(): string {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#url;
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @returns True if the response was successful (status in the range 200-299).
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
ok(): boolean {
|
2020-06-25 09:26:36 +00:00
|
|
|
// TODO: document === 0 case?
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#status === 0 || (this.#status >= 200 && this.#status <= 299);
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @returns The status code of the response (e.g., 200 for a success).
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
status(): number {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#status;
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
2022-08-12 12:15:26 +00:00
|
|
|
* @returns The status text of the response (e.g. usually an "OK" for a
|
2020-06-25 09:26:36 +00:00
|
|
|
* success).
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
statusText(): string {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#statusText;
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @returns An object with HTTP headers associated with the response. All
|
|
|
|
* header names are lower-case.
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
headers(): Record<string, string> {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#headers;
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @returns {@link SecurityDetails} if the response was received over the
|
|
|
|
* secure connection, or `null` otherwise.
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
securityDetails(): SecurityDetails | null {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#securityDetails;
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2022-02-15 10:16:49 +00:00
|
|
|
/**
|
|
|
|
* @returns Timing information related to the response.
|
|
|
|
*/
|
|
|
|
timing(): Protocol.Network.ResourceTiming | null {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#timing;
|
2022-02-15 10:16:49 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @returns Promise which resolves to a buffer with response body.
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
buffer(): Promise<Buffer> {
|
2022-06-13 09:16:25 +00:00
|
|
|
if (!this.#contentPromise) {
|
2022-06-22 13:25:44 +00:00
|
|
|
this.#contentPromise = this.#bodyLoadedPromise.then(async error => {
|
2022-06-14 11:55:35 +00:00
|
|
|
if (error) {
|
|
|
|
throw error;
|
|
|
|
}
|
2021-10-29 13:44:40 +00:00
|
|
|
try {
|
2022-06-13 09:16:25 +00:00
|
|
|
const response = await this.#client.send('Network.getResponseBody', {
|
|
|
|
requestId: this.#request._requestId,
|
2021-10-29 13:44:40 +00:00
|
|
|
});
|
|
|
|
return Buffer.from(
|
|
|
|
response.body,
|
|
|
|
response.base64Encoded ? 'base64' : 'utf8'
|
|
|
|
);
|
|
|
|
} catch (error) {
|
|
|
|
if (
|
|
|
|
error instanceof ProtocolError &&
|
|
|
|
error.originalMessage === 'No resource with given identifier found'
|
|
|
|
) {
|
|
|
|
throw new ProtocolError(
|
|
|
|
'Could not load body for this request. This might happen if the request is a preflight request.'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
throw error;
|
|
|
|
}
|
2020-05-13 13:57:21 +00:00
|
|
|
});
|
|
|
|
}
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#contentPromise;
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @returns Promise which resolves to a text representation of response body.
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
async text(): Promise<string> {
|
|
|
|
const content = await this.buffer();
|
|
|
|
return content.toString('utf8');
|
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @returns Promise which resolves to a JSON representation of response body.
|
|
|
|
*
|
|
|
|
* @remarks
|
|
|
|
*
|
|
|
|
* This method will throw if the response body is not parsable via
|
|
|
|
* `JSON.parse`.
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
async json(): Promise<any> {
|
|
|
|
const content = await this.text();
|
|
|
|
return JSON.parse(content);
|
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @returns A matching {@link HTTPRequest} object.
|
|
|
|
*/
|
2020-05-29 08:38:40 +00:00
|
|
|
request(): HTTPRequest {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#request;
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @returns True if the response was served from either the browser's disk
|
|
|
|
* cache or memory cache.
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
fromCache(): boolean {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#fromDiskCache || this.#request._fromMemoryCache;
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @returns True if the response was served by a service worker.
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
fromServiceWorker(): boolean {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#fromServiceWorker;
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 09:26:36 +00:00
|
|
|
/**
|
|
|
|
* @returns A {@link Frame} that initiated this response, or `null` if
|
|
|
|
* navigating to error pages.
|
|
|
|
*/
|
2020-05-13 13:57:21 +00:00
|
|
|
frame(): Frame | null {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#request.frame();
|
2020-05-13 13:57:21 +00:00
|
|
|
}
|
|
|
|
}
|