175 lines
28 KiB
JavaScript
175 lines
28 KiB
JavaScript
"use strict";
|
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
};
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
exports.createRequestOptions = void 0;
|
|
const lodash_1 = require("lodash");
|
|
const is_stream_1 = __importDefault(require("is-stream"));
|
|
const interfaces_1 = require("./interfaces");
|
|
const testcafe_hammerhead_1 = require("testcafe-hammerhead");
|
|
const content_types_1 = __importDefault(require("../../assets/content-types"));
|
|
const http_headers_1 = __importDefault(require("../../utils/http-headers"));
|
|
const types_1 = require("../../errors/types");
|
|
const runtime_1 = require("../../errors/runtime");
|
|
const actions_1 = require("../commands/actions");
|
|
const DEFAULT_ACCEPT = { [http_headers_1.default.accept]: `${content_types_1.default.json}, ${content_types_1.default.textPlain}, ${content_types_1.default.all}` };
|
|
const METHODS_WITH_CONTENT_TYPE = ['post', 'put', 'patch'];
|
|
const DEFAULT_REQUEST_METHOD = 'GET';
|
|
const DEFAULT_PROTOCOL = 'http:';
|
|
function setContentTypeIfNotExists(headers, value) {
|
|
if (!(0, lodash_1.isUndefined)(headers) && (0, lodash_1.isUndefined)(headers[http_headers_1.default.contentType]))
|
|
headers[http_headers_1.default.contentType] = value;
|
|
}
|
|
function typeOf(value) {
|
|
if (value === null)
|
|
return 'null';
|
|
if (value && typeof value === 'object')
|
|
return value.constructor.name.toLowerCase();
|
|
return typeof value;
|
|
}
|
|
function transformBody(headers, body) {
|
|
if (!body)
|
|
return Buffer.from('');
|
|
if (typeOf(body) === 'formdata' ||
|
|
typeOf(body) === 'file' ||
|
|
typeOf(body) === 'blob' ||
|
|
(0, lodash_1.isArrayBuffer)(body) ||
|
|
(0, lodash_1.isBuffer)(body) ||
|
|
(0, is_stream_1.default)(body))
|
|
return Buffer.from(body);
|
|
else if (ArrayBuffer.isView(body))
|
|
return Buffer.from(body.buffer);
|
|
else if (body instanceof URLSearchParams) {
|
|
setContentTypeIfNotExists(headers, `${content_types_1.default.urlencoded};charset=utf-8`);
|
|
return Buffer.from(body.toString());
|
|
}
|
|
else if ((0, lodash_1.isObject)(body) || headers && headers[http_headers_1.default.contentType] === content_types_1.default.json) {
|
|
setContentTypeIfNotExists(headers, content_types_1.default.json);
|
|
return Buffer.from(JSON.stringify(body));
|
|
}
|
|
else if (typeof body === 'string')
|
|
setContentTypeIfNotExists(headers, content_types_1.default.textPlain);
|
|
return body;
|
|
}
|
|
function getAuthString(auth) {
|
|
return 'Basic ' + Buffer.from(auth.username + ':' + auth.password, 'utf8').toString('base64');
|
|
}
|
|
function changeHeaderNamesToLowercase(headers) {
|
|
const lowerCaseHeaders = {};
|
|
Object.keys(headers).forEach(headerName => {
|
|
lowerCaseHeaders[headerName.toLowerCase()] = headers[headerName];
|
|
});
|
|
return lowerCaseHeaders;
|
|
}
|
|
async function prepareHeaders(headers, currentPageUrl, url, body, testRun, withCredentials, options) {
|
|
var _a;
|
|
const { host, origin } = url;
|
|
const preparedHeaders = Object.assign({}, DEFAULT_ACCEPT, changeHeaderNamesToLowercase(headers));
|
|
preparedHeaders[http_headers_1.default.host] = host;
|
|
preparedHeaders[http_headers_1.default.origin] = origin;
|
|
preparedHeaders[http_headers_1.default.contentLength] = body.length;
|
|
if (headers.method && METHODS_WITH_CONTENT_TYPE.includes(String(headers.method)))
|
|
preparedHeaders[http_headers_1.default.contentType] = content_types_1.default.urlencoded;
|
|
if (options.auth && withCredentials)
|
|
preparedHeaders[http_headers_1.default.authorization] = getAuthString(options.auth);
|
|
if ((_a = options.proxy) === null || _a === void 0 ? void 0 : _a.auth)
|
|
preparedHeaders[http_headers_1.default.proxyAuthorization] = getAuthString(options.proxy.auth);
|
|
if (withCredentials) {
|
|
const currentPageCookies = testRun.session.cookies.getHeader({
|
|
url: currentPageUrl.href,
|
|
hostname: currentPageUrl.hostname,
|
|
});
|
|
if (currentPageCookies)
|
|
preparedHeaders[http_headers_1.default.cookie] = currentPageCookies;
|
|
}
|
|
//NOTE: Additional header to recognize API requests in the hammerhead
|
|
preparedHeaders[http_headers_1.default.isApiRequest] = 'true';
|
|
return preparedHeaders;
|
|
}
|
|
async function prepareUrl(testRun, currentPageUrl, url, callsite) {
|
|
let preparedUrl;
|
|
try {
|
|
preparedUrl = url instanceof URL
|
|
? url
|
|
: new URL(url, currentPageUrl.hostname ? currentPageUrl.origin : void 0);
|
|
}
|
|
catch (e) {
|
|
throw new runtime_1.APIError(callsite, types_1.RUNTIME_ERRORS.requestUrlInvalidValueError, url);
|
|
}
|
|
return preparedUrl;
|
|
}
|
|
function prepareSearchParams(url, params) {
|
|
if (!params)
|
|
return url;
|
|
let searchParams;
|
|
if (params instanceof URLSearchParams)
|
|
searchParams = params;
|
|
else {
|
|
searchParams = new URLSearchParams();
|
|
for (const key in params) {
|
|
if (!params[key])
|
|
continue;
|
|
(0, lodash_1.castArray)(params[key]).forEach(v => {
|
|
searchParams.append(key, typeof v === 'object' ? JSON.stringify(v) : String(v));
|
|
});
|
|
}
|
|
}
|
|
return `${url}${url.includes('?') ? '&' : '?'}${searchParams.toString()}`;
|
|
}
|
|
function getProxyUrl(testRun, url, withCredentials) {
|
|
return testRun.executeCommand(new actions_1.GetProxyUrlCommand({
|
|
url: url,
|
|
options: { credentials: withCredentials ? interfaces_1.Credentials.include : interfaces_1.Credentials.omit },
|
|
}, testRun, true));
|
|
}
|
|
async function createRequestOptions(currentPageUrl, testRun, options, callsite) {
|
|
options.headers = options.headers || {};
|
|
const url = await prepareUrl(testRun, currentPageUrl, options.url, callsite);
|
|
const withCredentials = !currentPageUrl.host || (0, testcafe_hammerhead_1.sameOriginCheck)(currentPageUrl.href, url.href) || options.withCredentials || false;
|
|
const body = transformBody(options.headers, options.body);
|
|
const headers = await prepareHeaders(options.headers, currentPageUrl, url, body, testRun, withCredentials, options);
|
|
const proxyUrl = await getProxyUrl(testRun, url.href, withCredentials);
|
|
const proxyUrlObj = (0, testcafe_hammerhead_1.parseProxyUrl)(proxyUrl);
|
|
let auth = options.auth;
|
|
if (!auth && url.username && url.password) {
|
|
auth = {
|
|
username: url.username,
|
|
password: url.password,
|
|
};
|
|
}
|
|
const requestParams = {
|
|
method: options.method || DEFAULT_REQUEST_METHOD,
|
|
url: proxyUrl,
|
|
protocol: DEFAULT_PROTOCOL,
|
|
hostname: proxyUrlObj.proxy.hostname,
|
|
host: proxyUrlObj.proxy.hostname,
|
|
port: proxyUrlObj.proxy.port,
|
|
path: prepareSearchParams(proxyUrlObj.partAfterHost, options.params),
|
|
auth: auth && withCredentials ? `${auth.username}:${auth.password}` : void 0,
|
|
headers: headers,
|
|
credentials: withCredentials ? testRun.session.getAuthCredentials() : void 0,
|
|
body: body,
|
|
disableHttp2: testRun.session.isHttp2Disabled(),
|
|
requestTimeout: {
|
|
ajax: options.timeout,
|
|
page: options.timeout,
|
|
},
|
|
};
|
|
if (options.proxy) {
|
|
requestParams.externalProxySettings = {
|
|
host: options.proxy.host,
|
|
hostname: options.proxy.host,
|
|
port: options.proxy.port.toString(),
|
|
proxyAuth: options.proxy.auth ? `${options.proxy.auth.username}:${options.proxy.auth.password}` : void 0,
|
|
};
|
|
requestParams.protocol = url.protocol;
|
|
requestParams.host = url.host;
|
|
requestParams.hostname = url.hostname;
|
|
requestParams.port = url.port;
|
|
requestParams.path = prepareSearchParams(url.pathname + url.search, options.params);
|
|
}
|
|
return new testcafe_hammerhead_1.RequestOptions(requestParams);
|
|
}
|
|
exports.createRequestOptions = createRequestOptions;
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"create-request-options.js","sourceRoot":"","sources":["../../../src/test-run/request/create-request-options.ts"],"names":[],"mappings":"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","sourcesContent":["import { OutgoingHttpHeaders } from 'http';\nimport {\n    castArray,\n    isArrayBuffer,\n    isBuffer,\n    isObject,\n    isUndefined,\n} from 'lodash';\nimport isStream from 'is-stream';\nimport {\n    AuthOptions,\n    Credentials,\n    ExternalRequestOptions,\n    Params,\n} from './interfaces';\nimport {\n    RequestOptions,\n    parseProxyUrl,\n    RequestOptionsParams,\n    sameOriginCheck,\n} from 'testcafe-hammerhead';\nimport TestRun from '../index';\nimport CONTENT_TYPES from '../../assets/content-types';\nimport HTTP_HEADERS from '../../utils/http-headers';\nimport { RUNTIME_ERRORS } from '../../errors/types';\nimport { APIError } from '../../errors/runtime';\nimport { GetProxyUrlCommand } from '../commands/actions';\nimport { CallsiteRecord } from 'callsite-record';\n\nconst DEFAULT_ACCEPT            = { [HTTP_HEADERS.accept]: `${CONTENT_TYPES.json}, ${CONTENT_TYPES.textPlain}, ${CONTENT_TYPES.all}` };\nconst METHODS_WITH_CONTENT_TYPE = ['post', 'put', 'patch'];\nconst DEFAULT_REQUEST_METHOD    = 'GET';\nconst DEFAULT_PROTOCOL          = 'http:';\n\nfunction setContentTypeIfNotExists (headers: OutgoingHttpHeaders, value: string): void {\n    if (!isUndefined(headers) && isUndefined(headers[HTTP_HEADERS.contentType]))\n        headers[HTTP_HEADERS.contentType] = value;\n}\n\nfunction typeOf (value: unknown): string {\n    if (value === null)\n        return 'null';\n\n    if (value && typeof value === 'object')\n        return value.constructor.name.toLowerCase();\n\n    return typeof value;\n}\n\nfunction transformBody (headers: OutgoingHttpHeaders, body?: any): Buffer {\n    if (!body)\n        return Buffer.from('');\n\n    if (typeOf(body) === 'formdata' ||\n        typeOf(body) === 'file' ||\n        typeOf(body) === 'blob' ||\n        isArrayBuffer(body) ||\n        isBuffer(body) ||\n        isStream(body)\n    )\n        return Buffer.from(body);\n    else if (ArrayBuffer.isView(body))\n        return Buffer.from(body.buffer);\n\n    else if (body instanceof URLSearchParams) {\n        setContentTypeIfNotExists(headers, `${CONTENT_TYPES.urlencoded};charset=utf-8`);\n\n        return Buffer.from(body.toString());\n    }\n    else if (isObject(body) || headers && headers[HTTP_HEADERS.contentType] === CONTENT_TYPES.json) {\n        setContentTypeIfNotExists(headers, CONTENT_TYPES.json);\n\n        return Buffer.from(JSON.stringify(body));\n    }\n    else if (typeof body === 'string')\n        setContentTypeIfNotExists(headers, CONTENT_TYPES.textPlain);\n\n    return body;\n}\n\nfunction getAuthString (auth: AuthOptions): string {\n    return 'Basic ' + Buffer.from(auth.username + ':' + auth.password, 'utf8').toString('base64');\n}\n\nfunction changeHeaderNamesToLowercase (headers: OutgoingHttpHeaders): OutgoingHttpHeaders {\n    const lowerCaseHeaders: OutgoingHttpHeaders = {};\n\n    Object.keys(headers).forEach(headerName => {\n        lowerCaseHeaders[headerName.toLowerCase()] = headers[headerName];\n    });\n\n    return lowerCaseHeaders;\n}\n\nasync function prepareHeaders (headers: OutgoingHttpHeaders, currentPageUrl: URL, url: URL, body: Buffer, testRun: TestRun, withCredentials: boolean, options: ExternalRequestOptions): Promise<OutgoingHttpHeaders> {\n    const { host, origin } = url;\n\n    const preparedHeaders: OutgoingHttpHeaders = Object.assign({}, DEFAULT_ACCEPT, changeHeaderNamesToLowercase(headers));\n\n\n    preparedHeaders[HTTP_HEADERS.host] = host;\n    preparedHeaders[HTTP_HEADERS.origin] = origin;\n    preparedHeaders[HTTP_HEADERS.contentLength] = body.length;\n\n    if (headers.method && METHODS_WITH_CONTENT_TYPE.includes(String(headers.method)))\n        preparedHeaders[HTTP_HEADERS.contentType] = CONTENT_TYPES.urlencoded;\n\n    if (options.auth && withCredentials)\n        preparedHeaders[HTTP_HEADERS.authorization] = getAuthString(options.auth);\n\n    if (options.proxy?.auth)\n        preparedHeaders[HTTP_HEADERS.proxyAuthorization] = getAuthString(options.proxy.auth);\n\n    if (withCredentials) {\n        const currentPageCookies = testRun.session.cookies.getHeader({\n            url:      currentPageUrl.href,\n            hostname: currentPageUrl.hostname,\n        });\n\n        if (currentPageCookies)\n            preparedHeaders[HTTP_HEADERS.cookie] = currentPageCookies;\n    }\n\n    //NOTE: Additional header to recognize API requests in the hammerhead\n    preparedHeaders[HTTP_HEADERS.isApiRequest] = 'true';\n\n    return preparedHeaders;\n}\n\nasync function prepareUrl (testRun: TestRun, currentPageUrl: URL, url: string | URL, callsite: CallsiteRecord | null): Promise<URL> {\n    let preparedUrl: URL;\n\n    try {\n        preparedUrl = url instanceof URL\n            ? url\n            : new URL(url, currentPageUrl.hostname ? currentPageUrl.origin : void 0);\n    }\n    catch (e) {\n        throw new APIError(callsite, RUNTIME_ERRORS.requestUrlInvalidValueError, url);\n    }\n\n    return preparedUrl;\n}\n\nfunction prepareSearchParams (url: string, params?: Params): string {\n    if (!params)\n        return url;\n\n    let searchParams: URLSearchParams;\n\n    if (params instanceof URLSearchParams)\n        searchParams = params;\n    else {\n        searchParams = new URLSearchParams();\n\n        for (const key in params) {\n            if (!params[key])\n                continue;\n\n            castArray(params[key]).forEach(v => {\n                searchParams.append(key, typeof v === 'object' ? JSON.stringify(v) : String(v));\n            });\n        }\n    }\n\n    return `${url}${url.includes('?') ? '&' : '?'}${searchParams.toString()}`;\n}\n\nfunction getProxyUrl (testRun: TestRun, url: string, withCredentials?: boolean): Promise<string> {\n    return testRun.executeCommand(new GetProxyUrlCommand({\n        url:     url,\n        options: { credentials: withCredentials ? Credentials.include : Credentials.omit },\n    }, testRun, true)) as Promise<string>;\n}\n\nexport async function createRequestOptions (currentPageUrl: URL, testRun: TestRun, options: ExternalRequestOptions, callsite: CallsiteRecord | null): Promise<RequestOptions> {\n    options.headers = options.headers || {};\n\n    const url             = await prepareUrl(testRun, currentPageUrl, options.url, callsite);\n    const withCredentials = !currentPageUrl.host || sameOriginCheck(currentPageUrl.href, url.href) || options.withCredentials || false;\n    const body            = transformBody(options.headers, options.body);\n    const headers         = await prepareHeaders(options.headers, currentPageUrl, url, body, testRun, withCredentials, options);\n    const proxyUrl        = await getProxyUrl(testRun, url.href, withCredentials);\n    const proxyUrlObj     = parseProxyUrl(proxyUrl);\n    let auth              = options.auth;\n\n    if (!auth && url.username && url.password) {\n        auth = {\n            username: url.username,\n            password: url.password,\n        };\n    }\n\n    const requestParams: RequestOptionsParams = {\n        method:         options.method || DEFAULT_REQUEST_METHOD,\n        url:            proxyUrl,\n        protocol:       DEFAULT_PROTOCOL,\n        hostname:       proxyUrlObj.proxy.hostname,\n        host:           proxyUrlObj.proxy.hostname,\n        port:           proxyUrlObj.proxy.port,\n        path:           prepareSearchParams(proxyUrlObj.partAfterHost, options.params),\n        auth:           auth && withCredentials ? `${auth.username}:${auth.password}` : void 0,\n        headers:        headers,\n        credentials:    withCredentials ? testRun.session.getAuthCredentials() : void 0,\n        body:           body,\n        disableHttp2:   testRun.session.isHttp2Disabled(),\n        requestTimeout: {\n            ajax: options.timeout,\n            page: options.timeout,\n        },\n    };\n\n    if (options.proxy) {\n        requestParams.externalProxySettings = {\n            host:      options.proxy.host,\n            hostname:  options.proxy.host,\n            port:      options.proxy.port.toString(),\n            proxyAuth: options.proxy.auth ? `${options.proxy.auth.username}:${options.proxy.auth.password}` : void 0,\n        };\n\n        requestParams.protocol = url.protocol;\n        requestParams.host     = url.host;\n        requestParams.hostname = url.hostname;\n        requestParams.port     = url.port;\n        requestParams.path     = prepareSearchParams(url.pathname + url.search, options.params);\n    }\n\n    return new RequestOptions(requestParams);\n}\n"]}
|