testcafe
Version:
Automated browser testing for the modern web development stack.
126 lines • 19.4 kB
JavaScript
;
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.CdpCookieProvider = void 0;
const url_1 = require("url");
const base_1 = require("../test-run/cookies/base");
const match_collection_1 = __importDefault(require("../utils/match-collection"));
const get_active_client_1 = require("./utils/get-active-client");
const set_cookie_parser_1 = require("set-cookie-parser");
const lodash_1 = require("lodash");
const MAX_TIMESTAMP = 8640000000000000;
class CdpCookieProvider extends base_1.CookieProviderBase {
async _getCdpClient() {
const browserConnection = this.testRun.browserConnection;
return (0, get_active_client_1.getActiveClient)(browserConnection);
}
async initialize() {
return this.deleteCookies();
}
async getCookies(externalCookies, urls = []) {
const client = await this._getCdpClient();
const { cookies } = await client.Storage.getCookies({});
const parsedUrls = this._parseUrls(urls);
return (0, match_collection_1.default)(cookies, externalCookies, (cookie, cookieFilter) => {
const { domain, path } = cookieFilter;
if (domain && path || !parsedUrls.length)
return (0, lodash_1.isMatch)(cookie, cookieFilter);
for (const url of parsedUrls) {
if ((0, lodash_1.isMatch)(cookie, Object.assign({}, cookieFilter, { domain: url.domain, path: url.path })))
return true;
}
return false;
})
.map(this._cdpCookieToExternalCookie);
}
async setCookies(cookies, url) {
const client = await this._getCdpClient();
const { hostname = '', pathname = '/' } = url ? new url_1.URL(url) : {};
const cookiesArray = (0, lodash_1.castArray)(cookies);
const parsedCookies = this._isCookieOptionsArray(cookiesArray)
? cookiesArray
: this._parseSetCookieStrings(cookiesArray);
await client.Network.setCookies({
cookies: parsedCookies.map(cookie => this._cookieOptionToCdpCookieParam(cookie, hostname, pathname)),
});
}
async deleteCookies(cookies = [], urls = []) {
const client = await this._getCdpClient();
if (!cookies || !cookies.length)
return client.Network.clearBrowserCookies();
const parsedUrls = this._parseUrls(urls);
let existingCookies = await this.getCookies([]);
if (parsedUrls.length) {
existingCookies = existingCookies.filter(cookie => parsedUrls
.find(url => url.domain === cookie.domain && url.path === cookie.path));
}
existingCookies = (0, match_collection_1.default)(existingCookies, cookies);
for (const cookie of existingCookies) {
await client.Network.deleteCookies({
name: cookie.name || '',
domain: cookie.domain,
path: cookie.path,
});
}
return void 0;
}
async getCookieHeader(url) {
const [{ domain, path }] = this._parseUrls([url]);
const cookies = await this.getCookies([{ domain }]);
const filteredCookies = cookies.filter(c => this._includesPath(c.path || '/', path));
return filteredCookies.map(c => `${c.name}=${c.value}`).join(';');
}
_cdpCookieToExternalCookie(cookie) {
var _a;
return {
name: cookie.name,
value: cookie.value,
domain: cookie.domain,
maxAge: void 0,
path: cookie.path,
expires: void 0,
secure: cookie.secure,
httpOnly: cookie.httpOnly,
sameSite: (_a = cookie.sameSite) !== null && _a !== void 0 ? _a : 'none',
};
}
_cookieOptionToCdpCookieParam(cookie, hostname, pathname) {
var _a, _b, _c;
return {
name: cookie.name,
value: cookie.value,
domain: (_a = cookie.domain) !== null && _a !== void 0 ? _a : hostname,
path: (_b = cookie.path) !== null && _b !== void 0 ? _b : pathname,
secure: cookie.secure,
httpOnly: cookie.httpOnly,
sameSite: cookie.sameSite,
expires: ((_c = cookie.expires) === null || _c === void 0 ? void 0 : _c.getTime()) || MAX_TIMESTAMP,
};
}
_parseUrls(urls) {
return urls.map(url => {
const { hostname, pathname } = new url_1.URL(url);
return { domain: hostname, path: pathname };
});
}
_includesPath(cookiePath, urlPath) {
if (cookiePath === '/')
return true;
const cookieParts = cookiePath.split('/');
const urlParts = urlPath.split('/');
if (cookieParts.length > urlParts.length)
return false;
while (cookieParts.length) {
if (cookieParts.shift() !== urlParts.shift())
return false;
}
return true;
}
_parseSetCookieStrings(cookies) {
return (0, set_cookie_parser_1.parse)(cookies);
}
}
exports.CdpCookieProvider = CdpCookieProvider;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cookie-provider.js","sourceRoot":"","sources":["../../src/native-automation/cookie-provider.ts"],"names":[],"mappings":"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","sourcesContent":["import remoteChrome from 'chrome-remote-interface';\nimport { ExternalCookies } from 'testcafe-hammerhead';\nimport Protocol from 'devtools-protocol';\nimport Cookie = Protocol.Network.Cookie;\nimport { URL } from 'url';\nimport { CookieOptions } from '../test-run/commands/options';\nimport { CookieProvider, CookieProviderBase } from '../test-run/cookies/base';\nimport CookieParam = Protocol.Network.CookieParam;\nimport matchCollection from '../utils/match-collection';\nimport { getActiveClient } from './utils/get-active-client';\nimport { parse } from 'set-cookie-parser';\nimport { castArray, isMatch } from 'lodash';\n\ndeclare type CookieSameSite = 'Lax' | 'Strict' | 'None';\n\nconst MAX_TIMESTAMP = 8640000000000000;\n\nexport class CdpCookieProvider extends CookieProviderBase implements CookieProvider {\n    private async _getCdpClient (): Promise<remoteChrome.ProtocolApi> {\n        const browserConnection = this.testRun.browserConnection;\n\n        return getActiveClient(browserConnection);\n    }\n\n    async initialize (): Promise<void> {\n        return this.deleteCookies();\n    }\n\n    async getCookies (externalCookies: ExternalCookies[], urls: string[] = []): Promise<ExternalCookies[]> {\n        const client      = await this._getCdpClient();\n        const { cookies } = await client.Storage.getCookies({});\n        const parsedUrls  = this._parseUrls(urls);\n\n        return (matchCollection(cookies, externalCookies, (cookie: Protocol.Network.Cookie, cookieFilter: ExternalCookies) => {\n            const { domain, path } = cookieFilter;\n\n            if (domain && path || !parsedUrls.length)\n                return isMatch(cookie, cookieFilter);\n\n            for (const url of parsedUrls) {\n                if (isMatch(cookie, Object.assign({}, cookieFilter, { domain: url.domain, path: url.path })))\n                    return true;\n            }\n\n            return false;\n        }) as Cookie[])\n            .map(this._cdpCookieToExternalCookie);\n    }\n\n    async setCookies (cookies: string | string[] | CookieOptions[], url: string): Promise<void> {\n        const client                            = await this._getCdpClient();\n        const { hostname = '', pathname = '/' } = url ? new URL(url) : {};\n        const cookiesArray                      = castArray<string | CookieOptions>(cookies);\n\n        const parsedCookies = this._isCookieOptionsArray(cookiesArray)\n            ? cookiesArray\n            : this._parseSetCookieStrings(cookiesArray as string[]);\n\n        await client.Network.setCookies({\n            cookies: parsedCookies.map(cookie => this._cookieOptionToCdpCookieParam(cookie, hostname, pathname)),\n        });\n    }\n\n    async deleteCookies (cookies: CookieOptions[] = [], urls: string[] = []): Promise<void> {\n        const client = await this._getCdpClient();\n\n        if (!cookies || !cookies.length)\n            return client.Network.clearBrowserCookies();\n\n        const parsedUrls    = this._parseUrls(urls);\n        let existingCookies = await this.getCookies([]);\n\n        if (parsedUrls.length) {\n            existingCookies = existingCookies.filter(cookie => parsedUrls\n                .find(url => url.domain === cookie.domain && url.path === cookie.path));\n        }\n\n        existingCookies = matchCollection(existingCookies, cookies) as ExternalCookies[];\n\n        for (const cookie of existingCookies) {\n            await client.Network.deleteCookies({\n                name:   cookie.name || '',\n                domain: cookie.domain,\n                path:   cookie.path,\n            });\n        }\n\n        return void 0;\n    }\n\n    async getCookieHeader (url: string): Promise<string | null> {\n        const [{ domain, path }] = this._parseUrls([url]);\n        const cookies            = await this.getCookies([{ domain }]);\n        const filteredCookies    = cookies.filter(c => this._includesPath(c.path || '/', path));\n\n        return filteredCookies.map(c => `${ c.name }=${ c.value }`).join(';');\n    }\n\n    private _cdpCookieToExternalCookie (cookie: Cookie): ExternalCookies {\n        return {\n            name:     cookie.name,\n            value:    cookie.value,\n            domain:   cookie.domain,\n            maxAge:   void 0,\n            path:     cookie.path,\n            expires:  void 0,\n            secure:   cookie.secure,\n            httpOnly: cookie.httpOnly,\n            sameSite: cookie.sameSite ?? 'none',\n        } as unknown as ExternalCookies;\n    }\n\n    private _cookieOptionToCdpCookieParam (cookie: CookieOptions, hostname: string, pathname: string): CookieParam {\n        return {\n            name:     cookie.name,\n            value:    cookie.value,\n            domain:   cookie.domain ?? hostname,\n            path:     cookie.path ?? pathname,\n            secure:   cookie.secure,\n            httpOnly: cookie.httpOnly,\n            sameSite: cookie.sameSite as CookieSameSite,\n            expires:  cookie.expires?.getTime() || MAX_TIMESTAMP,\n        };\n    }\n\n    private _parseUrls (urls: string[]): { domain: string, path: string }[] {\n        return urls.map(url => {\n            const { hostname, pathname } = new URL(url);\n\n            return { domain: hostname, path: pathname };\n        });\n    }\n\n    private _includesPath (cookiePath: string, urlPath: string): boolean {\n        if (cookiePath === '/')\n            return true;\n\n        const cookieParts = cookiePath.split('/');\n        const urlParts    = urlPath.split('/');\n\n        if (cookieParts.length > urlParts.length)\n            return false;\n\n        while (cookieParts.length) {\n            if (cookieParts.shift() !== urlParts.shift())\n                return false;\n        }\n\n        return true;\n    }\n\n    private _parseSetCookieStrings (cookies: string[]): CookieOptions[] {\n        return parse(cookies) as CookieOptions[];\n    }\n}\n"]}