jam-cloud/jam-ui/node_modules/@playwright/test/lib/server/fetch.js

497 lines
16 KiB
JavaScript

"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.GlobalFetchRequest = exports.BrowserContextFetchRequest = exports.FetchRequest = void 0;
var http = _interopRequireWildcard(require("http"));
var https = _interopRequireWildcard(require("https"));
var _httpsProxyAgent = require("https-proxy-agent");
var _stream = require("stream");
var _url = _interopRequireDefault(require("url"));
var _zlib = _interopRequireDefault(require("zlib"));
var _timeoutSettings = require("../utils/timeoutSettings");
var _utils = require("../utils/utils");
var _browserContext = require("./browserContext");
var _formData = require("./formData");
var _instrumentation = require("./instrumentation");
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function (nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); }
function _interopRequireWildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
/**
* 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.
*/
class FetchRequest extends _instrumentation.SdkObject {
static findResponseBody(guid) {
for (const request of FetchRequest.allInstances) {
const body = request.fetchResponses.get(guid);
if (body) return body;
}
return undefined;
}
constructor(parent) {
super(parent, 'fetchRequest');
this.fetchResponses = new Map();
FetchRequest.allInstances.add(this);
}
_disposeImpl() {
FetchRequest.allInstances.delete(this);
this.fetchResponses.clear();
this.emit(FetchRequest.Events.Dispose);
}
_storeResponseBody(body) {
const uid = (0, _utils.createGuid)();
this.fetchResponses.set(uid, body);
return uid;
}
async fetch(params) {
try {
var _params$method;
const headers = {};
const defaults = this._defaultOptions();
headers['user-agent'] = defaults.userAgent;
headers['accept'] = '*/*';
headers['accept-encoding'] = 'gzip,deflate,br';
if (defaults.extraHTTPHeaders) {
for (const {
name,
value
} of defaults.extraHTTPHeaders) headers[name.toLowerCase()] = value;
}
if (params.headers) {
for (const [name, value] of Object.entries(params.headers)) headers[name.toLowerCase()] = value;
}
const method = ((_params$method = params.method) === null || _params$method === void 0 ? void 0 : _params$method.toUpperCase()) || 'GET';
const proxy = defaults.proxy;
let agent;
if (proxy) {
// TODO: support bypass proxy
const proxyOpts = _url.default.parse(proxy.server);
if (proxy.username) proxyOpts.auth = `${proxy.username}:${proxy.password || ''}`;
agent = new _httpsProxyAgent.HttpsProxyAgent(proxyOpts);
}
const timeout = defaults.timeoutSettings.timeout(params);
const deadline = timeout && (0, _utils.monotonicTime)() + timeout;
const options = {
method,
headers,
agent,
maxRedirects: 20,
timeout,
deadline
}; // rejectUnauthorized = undefined is treated as true in node 12.
if (defaults.ignoreHTTPSErrors) options.rejectUnauthorized = false;
const requestUrl = new URL(params.url, defaults.baseURL);
if (params.params) {
for (const [name, value] of Object.entries(params.params)) requestUrl.searchParams.set(name, value);
}
let postData;
if (['POST', 'PUSH', 'PATCH'].includes(method)) postData = params.formData ? serilizeFormData(params.formData, headers) : params.postData;else if (params.postData || params.formData) throw new Error(`Method ${method} does not accept post data`);
const fetchResponse = await this._sendRequest(requestUrl, options, postData);
const fetchUid = this._storeResponseBody(fetchResponse.body);
if (params.failOnStatusCode && (fetchResponse.status < 200 || fetchResponse.status >= 400)) return {
error: `${fetchResponse.status} ${fetchResponse.statusText}`
};
return {
fetchResponse: { ...fetchResponse,
fetchUid
}
};
} catch (e) {
return {
error: String(e)
};
}
}
async _updateCookiesFromHeader(responseUrl, setCookie) {
const url = new URL(responseUrl); // https://datatracker.ietf.org/doc/html/rfc6265#section-5.1.4
const defaultPath = '/' + url.pathname.substr(1).split('/').slice(0, -1).join('/');
const cookies = [];
for (const header of setCookie) {
// Decode cookie value?
const cookie = parseCookie(header);
if (!cookie) continue;
if (!cookie.domain) cookie.domain = url.hostname;
if (!canSetCookie(cookie.domain, url.hostname)) continue; // https://datatracker.ietf.org/doc/html/rfc6265#section-5.2.4
if (!cookie.path || !cookie.path.startsWith('/')) cookie.path = defaultPath;
cookies.push(cookie);
}
if (cookies.length) await this._addCookies(cookies);
}
async _updateRequestCookieHeader(url, options) {
if (options.headers['cookie'] !== undefined) return;
const cookies = await this._cookies(url.toString());
if (cookies.length) {
const valueArray = cookies.map(c => `${c.name}=${c.value}`);
options.headers['cookie'] = valueArray.join('; ');
}
}
async _sendRequest(url, options, postData) {
await this._updateRequestCookieHeader(url, options);
return new Promise((fulfill, reject) => {
const requestConstructor = (url.protocol === 'https:' ? https : http).request;
const request = requestConstructor(url, options, async response => {
if (response.headers['set-cookie']) await this._updateCookiesFromHeader(response.url || url.toString(), response.headers['set-cookie']);
if (redirectStatus.includes(response.statusCode)) {
if (!options.maxRedirects) {
reject(new Error('Max redirect count exceeded'));
request.abort();
return;
}
const headers = { ...options.headers
};
delete headers[`cookie`]; // HTTP-redirect fetch step 13 (https://fetch.spec.whatwg.org/#http-redirect-fetch)
const status = response.statusCode;
let method = options.method;
if ((status === 301 || status === 302) && method === 'POST' || status === 303 && !['GET', 'HEAD'].includes(method)) {
method = 'GET';
postData = undefined;
delete headers[`content-encoding`];
delete headers[`content-language`];
delete headers[`content-location`];
delete headers[`content-type`];
}
const redirectOptions = {
method,
headers,
agent: options.agent,
maxRedirects: options.maxRedirects - 1,
timeout: options.timeout,
deadline: options.deadline
}; // HTTP-redirect fetch step 4: If locationURL is null, then return response.
if (response.headers.location) {
const locationURL = new URL(response.headers.location, url);
fulfill(this._sendRequest(locationURL, redirectOptions, postData));
request.abort();
return;
}
}
if (response.statusCode === 401 && !options.headers['authorization']) {
const auth = response.headers['www-authenticate'];
const credentials = this._defaultOptions().httpCredentials;
if (auth !== null && auth !== void 0 && auth.trim().startsWith('Basic ') && credentials) {
const {
username,
password
} = credentials;
const encoded = Buffer.from(`${username || ''}:${password || ''}`).toString('base64');
options.headers['authorization'] = `Basic ${encoded}`;
fulfill(this._sendRequest(url, options, postData));
request.abort();
return;
}
}
response.on('aborted', () => reject(new Error('aborted')));
let body = response;
let transform;
const encoding = response.headers['content-encoding'];
if (encoding === 'gzip' || encoding === 'x-gzip') {
transform = _zlib.default.createGunzip({
flush: _zlib.default.constants.Z_SYNC_FLUSH,
finishFlush: _zlib.default.constants.Z_SYNC_FLUSH
});
} else if (encoding === 'br') {
transform = _zlib.default.createBrotliDecompress();
} else if (encoding === 'deflate') {
transform = _zlib.default.createInflate();
}
if (transform) {
body = (0, _stream.pipeline)(response, transform, e => {
if (e) reject(new Error(`failed to decompress '${encoding}' encoding: ${e}`));
});
}
const chunks = [];
body.on('data', chunk => chunks.push(chunk));
body.on('end', () => {
const body = Buffer.concat(chunks);
fulfill({
url: response.url || url.toString(),
status: response.statusCode || 0,
statusText: response.statusMessage || '',
headers: toHeadersArray(response.rawHeaders),
body
});
});
body.on('error', reject);
});
request.on('error', reject);
if (options.deadline) {
const rejectOnTimeout = () => {
reject(new Error(`Request timed out after ${options.timeout}ms`));
request.abort();
};
const remaining = options.deadline - (0, _utils.monotonicTime)();
if (remaining <= 0) {
rejectOnTimeout();
return;
}
request.setTimeout(remaining, rejectOnTimeout);
}
if (postData) request.write(postData);
request.end();
});
}
}
exports.FetchRequest = FetchRequest;
FetchRequest.Events = {
Dispose: 'dispose'
};
FetchRequest.allInstances = new Set();
class BrowserContextFetchRequest extends FetchRequest {
constructor(context) {
super(context);
this._context = void 0;
this._context = context;
context.once(_browserContext.BrowserContext.Events.Close, () => this._disposeImpl());
}
dispose() {
this.fetchResponses.clear();
}
_defaultOptions() {
return {
userAgent: this._context._options.userAgent || this._context._browser.userAgent(),
extraHTTPHeaders: this._context._options.extraHTTPHeaders,
httpCredentials: this._context._options.httpCredentials,
proxy: this._context._options.proxy || this._context._browser.options.proxy,
timeoutSettings: this._context._timeoutSettings,
ignoreHTTPSErrors: this._context._options.ignoreHTTPSErrors,
baseURL: this._context._options.baseURL
};
}
async _addCookies(cookies) {
await this._context.addCookies(cookies);
}
async _cookies(url) {
return await this._context.cookies(url);
}
}
exports.BrowserContextFetchRequest = BrowserContextFetchRequest;
class GlobalFetchRequest extends FetchRequest {
constructor(playwright, options) {
super(playwright);
this._options = void 0;
const timeoutSettings = new _timeoutSettings.TimeoutSettings();
if (options.timeout !== undefined) timeoutSettings.setDefaultTimeout(options.timeout);
const proxy = options.proxy;
if (proxy !== null && proxy !== void 0 && proxy.server) {
let url = proxy === null || proxy === void 0 ? void 0 : proxy.server.trim();
if (!/^\w+:\/\//.test(url)) url = 'http://' + url;
proxy.server = url;
}
this._options = {
baseURL: options.baseURL,
userAgent: options.userAgent || '',
extraHTTPHeaders: options.extraHTTPHeaders,
ignoreHTTPSErrors: !!options.ignoreHTTPSErrors,
httpCredentials: options.httpCredentials,
proxy,
timeoutSettings
};
}
dispose() {
this._disposeImpl();
}
_defaultOptions() {
return this._options;
}
async _addCookies(cookies) {}
async _cookies(url) {
return [];
}
}
exports.GlobalFetchRequest = GlobalFetchRequest;
function toHeadersArray(rawHeaders) {
const result = [];
for (let i = 0; i < rawHeaders.length; i += 2) result.push({
name: rawHeaders[i],
value: rawHeaders[i + 1]
});
return result;
}
const redirectStatus = [301, 302, 303, 307, 308];
function canSetCookie(cookieDomain, hostname) {
// TODO: check public suffix list?
hostname = '.' + hostname;
if (!cookieDomain.startsWith('.')) cookieDomain = '.' + cookieDomain;
return hostname.endsWith(cookieDomain);
}
function parseCookie(header) {
const pairs = header.split(';').filter(s => s.trim().length > 0).map(p => p.split('=').map(s => s.trim()));
if (!pairs.length) return null;
const [name, value] = pairs[0];
const cookie = {
name,
value,
domain: '',
path: '',
expires: -1,
httpOnly: false,
secure: false,
sameSite: 'Lax' // None for non-chromium
};
for (let i = 1; i < pairs.length; i++) {
const [name, value] = pairs[i];
switch (name.toLowerCase()) {
case 'expires':
const expiresMs = +new Date(value);
if (isFinite(expiresMs)) cookie.expires = expiresMs / 1000;
break;
case 'max-age':
const maxAgeSec = parseInt(value, 10);
if (isFinite(maxAgeSec)) cookie.expires = Date.now() / 1000 + maxAgeSec;
break;
case 'domain':
cookie.domain = value || '';
break;
case 'path':
cookie.path = value || '';
break;
case 'secure':
cookie.secure = true;
break;
case 'httponly':
cookie.httpOnly = true;
break;
}
}
return cookie;
}
function serilizeFormData(data, headers) {
const contentType = headers['content-type'] || 'application/json';
if (contentType === 'application/json') {
var _contentType, _headers$_contentType;
const json = JSON.stringify(data);
(_headers$_contentType = headers[_contentType = 'content-type']) !== null && _headers$_contentType !== void 0 ? _headers$_contentType : headers[_contentType] = contentType;
return Buffer.from(json, 'utf8');
} else if (contentType === 'application/x-www-form-urlencoded') {
const searchParams = new URLSearchParams();
for (const [name, value] of Object.entries(data)) searchParams.append(name, String(value));
return Buffer.from(searchParams.toString(), 'utf8');
} else if (contentType === 'multipart/form-data') {
const formData = new _formData.MultipartFormData();
for (const [name, value] of Object.entries(data)) {
if ((0, _utils.isFilePayload)(value)) {
const payload = value;
formData.addFileField(name, payload);
} else if (value !== undefined) {
formData.addField(name, String(value));
}
}
headers['content-type'] = formData.contentTypeHeader();
return formData.finish();
} else {
throw new Error(`Cannot serialize data using content type: ${contentType}`);
}
}