puppeteer/packages/puppeteer-core/src/common/HTTPResponse.ts

193 lines
5.2 KiB
TypeScript
Raw Normal View History

/**
* 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.
*/
import {Protocol} from 'devtools-protocol';
import {
HTTPResponse as BaseHTTPResponse,
RemoteAddress,
} from '../api/HTTPResponse.js';
import {CDPSession} from './Connection.js';
import {ProtocolError} from './Errors.js';
import {Frame} from './Frame.js';
import {HTTPRequest} from './HTTPRequest.js';
import {SecurityDetails} from './SecurityDetails.js';
/**
* @internal
*/
export class HTTPResponse extends BaseHTTPResponse {
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;
constructor(
client: CDPSession,
request: HTTPRequest,
responsePayload: Protocol.Network.Response,
extraInfo: Protocol.Network.ResponseReceivedExtraInfoEvent | null
) {
super();
2022-06-13 09:16:25 +00:00
this.#client = client;
this.#request = request;
this.#bodyLoadedPromise = new Promise(fulfill => {
2022-06-13 09:16:25 +00:00
this.#bodyLoadedPromiseFulfill = fulfill;
});
2022-06-13 09:16:25 +00:00
this.#remoteAddress = {
ip: responsePayload.remoteIPAddress,
port: responsePayload.remotePort,
};
2022-06-13 09:16:25 +00:00
this.#statusText =
this.#parseStatusTextFromExtrInfo(extraInfo) ||
responsePayload.statusText;
2022-06-13 09:16:25 +00:00
this.#url = request.url();
this.#fromDiskCache = !!responsePayload.fromDiskCache;
this.#fromServiceWorker = !!responsePayload.fromServiceWorker;
2022-06-13 09:16:25 +00:00
this.#status = extraInfo ? extraInfo.statusCode : responsePayload.status;
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
}
2022-06-13 09:16:25 +00:00
this.#securityDetails = responsePayload.securityDetails
? new SecurityDetails(responsePayload.securityDetails)
: null;
2022-06-13 09:16:25 +00:00
this.#timing = responsePayload.timing || null;
}
2022-06-13 09:16:25 +00:00
#parseStatusTextFromExtrInfo(
extraInfo: Protocol.Network.ResponseReceivedExtraInfoEvent | null
): string | undefined {
2022-06-14 11:55:35 +00:00
if (!extraInfo || !extraInfo.headersText) {
return;
}
const firstLine = extraInfo.headersText.split('\r', 1)[0];
2022-06-14 11:55:35 +00:00
if (!firstLine) {
return;
}
const match = firstLine.match(/[^ ]* [^ ]* (.*)/);
2022-06-14 11:55:35 +00:00
if (!match) {
return;
}
const statusText = match[1];
2022-06-14 11:55:35 +00:00
if (!statusText) {
return;
}
return statusText;
}
override _resolveBody(err: Error | null): void {
if (err) {
2022-06-13 09:16:25 +00:00
return this.#bodyLoadedPromiseFulfill(err);
}
2022-06-13 09:16:25 +00:00
return this.#bodyLoadedPromiseFulfill();
}
override remoteAddress(): RemoteAddress {
2022-06-13 09:16:25 +00:00
return this.#remoteAddress;
}
override url(): string {
2022-06-13 09:16:25 +00:00
return this.#url;
}
override ok(): boolean {
// TODO: document === 0 case?
2022-06-13 09:16:25 +00:00
return this.#status === 0 || (this.#status >= 200 && this.#status <= 299);
}
override status(): number {
2022-06-13 09:16:25 +00:00
return this.#status;
}
override statusText(): string {
2022-06-13 09:16:25 +00:00
return this.#statusText;
}
override headers(): Record<string, string> {
2022-06-13 09:16:25 +00:00
return this.#headers;
}
override securityDetails(): SecurityDetails | null {
2022-06-13 09:16:25 +00:00
return this.#securityDetails;
}
override timing(): Protocol.Network.ResourceTiming | null {
2022-06-13 09:16:25 +00:00
return this.#timing;
}
override buffer(): Promise<Buffer> {
2022-06-13 09:16:25 +00:00
if (!this.#contentPromise) {
this.#contentPromise = this.#bodyLoadedPromise.then(async error => {
2022-06-14 11:55:35 +00:00
if (error) {
throw error;
}
try {
2022-06-13 09:16:25 +00:00
const response = await this.#client.send('Network.getResponseBody', {
requestId: this.#request._requestId,
});
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;
}
});
}
2022-06-13 09:16:25 +00:00
return this.#contentPromise;
}
override request(): HTTPRequest {
2022-06-13 09:16:25 +00:00
return this.#request;
}
override fromCache(): boolean {
2022-06-13 09:16:25 +00:00
return this.#fromDiskCache || this.#request._fromMemoryCache;
}
override fromServiceWorker(): boolean {
2022-06-13 09:16:25 +00:00
return this.#fromServiceWorker;
}
override frame(): Frame | null {
2022-06-13 09:16:25 +00:00
return this.#request.frame();
}
}