mirror of
https://github.com/microsoft/playwright.git
synced 2025-01-05 19:04:43 +03:00
700b310150
Fixes #26833
450 lines
15 KiB
TypeScript
450 lines
15 KiB
TypeScript
/**
|
|
* Copyright (c) Microsoft Corporation. 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 type { LookupAddress } from 'dns';
|
|
import fs from 'fs';
|
|
import type { APIRequestContext } from 'playwright-core';
|
|
import { expect, playwrightTest } from '../config/browserTest';
|
|
|
|
export type GlobalFetchFixtures = {
|
|
request: APIRequestContext;
|
|
};
|
|
|
|
const it = playwrightTest.extend<GlobalFetchFixtures>({
|
|
request: async ({ playwright }, use) => {
|
|
const request = await playwright.request.newContext({ ignoreHTTPSErrors: true });
|
|
await use(request);
|
|
await request.dispose();
|
|
},
|
|
});
|
|
|
|
type PromiseArg<T> = T extends Promise<infer R> ? R : never;
|
|
type StorageStateType = PromiseArg<ReturnType<APIRequestContext['storageState']>>;
|
|
|
|
it.skip(({ mode }) => mode !== 'default');
|
|
|
|
const __testHookLookup = (hostname: string): LookupAddress[] => {
|
|
if (hostname === 'localhost' || hostname.endsWith('one.com') || hostname.endsWith('two.com'))
|
|
return [{ address: '127.0.0.1', family: 4 }];
|
|
else
|
|
throw new Error(`Failed to resolve hostname: ${hostname}`);
|
|
};
|
|
|
|
it('should store cookie from Set-Cookie header', async ({ request, server }) => {
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', ['a=b', 'c=d; max-age=3600; domain=b.one.com; path=/input', 'e=f; domain=b.one.com; path=/input/subfolder']);
|
|
res.end();
|
|
});
|
|
await request.get(`http://a.b.one.com:${server.PORT}/setcookie.html`, { __testHookLookup } as any);
|
|
const [serverRequest] = await Promise.all([
|
|
server.waitForRequest('/input/button.html'),
|
|
request.get(`http://b.one.com:${server.PORT}/input/button.html`, { __testHookLookup } as any)
|
|
]);
|
|
expect(serverRequest.headers.cookie).toBe('c=d');
|
|
});
|
|
|
|
it('should filter outgoing cookies by path', async ({ request, server }) => {
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', ['a=v; path=/input/subfolder', 'b=v; path=/input', 'c=v;']);
|
|
res.end();
|
|
});
|
|
await request.get(`${server.PREFIX}/setcookie.html`);
|
|
const [serverRequest] = await Promise.all([
|
|
server.waitForRequest('/input/button.html'),
|
|
request.get(`${server.PREFIX}/input/button.html`)
|
|
]);
|
|
expect(serverRequest.headers.cookie).toBe('b=v; c=v');
|
|
});
|
|
|
|
it('should filter outgoing cookies by domain', async ({ request, server }) => {
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', ['a=v; domain=one.com', 'b=v; domain=.b.one.com', 'c=v; domain=other.com']);
|
|
res.end();
|
|
});
|
|
await request.get(`http://a.b.one.com:${server.PORT}/setcookie.html`, { __testHookLookup } as any);
|
|
const [serverRequest] = await Promise.all([
|
|
server.waitForRequest('/empty.html'),
|
|
request.get(`http://www.b.one.com:${server.PORT}/empty.html`, { __testHookLookup } as any)
|
|
]);
|
|
expect(serverRequest.headers.cookie).toBe('a=v; b=v');
|
|
|
|
const [serverRequest2] = await Promise.all([
|
|
server.waitForRequest('/empty.html'),
|
|
request.get(`http://two.com:${server.PORT}/empty.html`, { __testHookLookup } as any)
|
|
]);
|
|
expect(serverRequest2.headers.cookie).toBeFalsy();
|
|
});
|
|
|
|
it('should do case-insensitive match of cookie domain', async ({ request, server }) => {
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', ['a=v; domain=One.com', 'b=v; domain=.B.oNe.com']);
|
|
res.end();
|
|
});
|
|
await request.get(`http://a.b.one.com:${server.PORT}/setcookie.html`, { __testHookLookup } as any);
|
|
const [serverRequest] = await Promise.all([
|
|
server.waitForRequest('/empty.html'),
|
|
request.get(`http://www.b.one.com:${server.PORT}/empty.html`, { __testHookLookup } as any)
|
|
]);
|
|
expect(serverRequest.headers.cookie).toBe('a=v; b=v');
|
|
});
|
|
|
|
it('should do case-insensitive match of request domain', async ({ request, server }) => {
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', ['a=v; domain=one.com', 'b=v; domain=.b.one.com']);
|
|
res.end();
|
|
});
|
|
await request.get(`http://a.b.one.com:${server.PORT}/setcookie.html`, { __testHookLookup } as any);
|
|
const [serverRequest] = await Promise.all([
|
|
server.waitForRequest('/empty.html'),
|
|
request.get(`http://WWW.B.ONE.COM:${server.PORT}/empty.html`, { __testHookLookup } as any)
|
|
]);
|
|
expect(serverRequest.headers.cookie).toBe('a=v; b=v');
|
|
});
|
|
|
|
it('should send secure cookie over https', async ({ request, server, httpsServer }) => {
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', ['a=v; secure', 'b=v']);
|
|
res.end();
|
|
});
|
|
await request.get(`${server.PREFIX}/setcookie.html`);
|
|
const [serverRequest] = await Promise.all([
|
|
httpsServer.waitForRequest('/empty.html'),
|
|
request.get(httpsServer.EMPTY_PAGE)
|
|
]);
|
|
expect(serverRequest.headers.cookie).toBe('a=v; b=v');
|
|
});
|
|
|
|
it('should send secure cookie over http for localhost', async ({ request, server }) => {
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', ['a=v; secure', 'b=v']);
|
|
res.end();
|
|
});
|
|
await request.get(`${server.PREFIX}/setcookie.html`);
|
|
const [serverRequest] = await Promise.all([
|
|
server.waitForRequest('/empty.html'),
|
|
request.get(server.EMPTY_PAGE)
|
|
]);
|
|
expect(serverRequest.headers.cookie).toBe('a=v; b=v');
|
|
});
|
|
|
|
it('should send not expired cookies', async ({ request, server }) => {
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
const tomorrow = new Date();
|
|
tomorrow.setDate(tomorrow.getDate() + 1);
|
|
res.setHeader('Set-Cookie', ['a=v', `b=v; expires=${tomorrow.toUTCString()}`]);
|
|
res.end();
|
|
});
|
|
await request.get(`${server.PREFIX}/setcookie.html`);
|
|
const [serverRequest] = await Promise.all([
|
|
server.waitForRequest('/empty.html'),
|
|
request.get(server.EMPTY_PAGE)
|
|
]);
|
|
expect(serverRequest.headers.cookie).toBe('a=v; b=v');
|
|
});
|
|
|
|
it('should remove expired cookies', async ({ request, server }) => {
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', ['a=v', `b=v; expires=${new Date().toUTCString()}`]);
|
|
res.end();
|
|
});
|
|
await request.get(`${server.PREFIX}/setcookie.html`);
|
|
const [serverRequest] = await Promise.all([
|
|
server.waitForRequest('/empty.html'),
|
|
request.get(server.EMPTY_PAGE)
|
|
]);
|
|
expect(serverRequest.headers.cookie).toBe('a=v');
|
|
});
|
|
|
|
it('should remove cookie with negative max-age', async ({ request, server }) => {
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', ['a=v; max-age=100000', `b=v; max-age=100000`, 'c=v']);
|
|
res.end();
|
|
});
|
|
server.setRoute('/removecookie.html', (req, res) => {
|
|
const maxAge = -2 * Date.now();
|
|
res.setHeader('Set-Cookie', [`a=v; max-age=${maxAge}`, `b=v; max-age=-1`]);
|
|
res.end();
|
|
});
|
|
await request.get(`${server.PREFIX}/setcookie.html`);
|
|
await request.get(`${server.PREFIX}/removecookie.html`);
|
|
const [serverRequest] = await Promise.all([
|
|
server.waitForRequest('/empty.html'),
|
|
request.get(server.EMPTY_PAGE)
|
|
]);
|
|
expect(serverRequest.headers.cookie).toBe('c=v');
|
|
});
|
|
|
|
it('should remove cookie with expires far in the past', async ({ request, server }) => {
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', ['a=v; max-age=1000000']);
|
|
res.end();
|
|
});
|
|
server.setRoute('/removecookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', [`a=v; expires=1 Jan 1000 00:00:00 +0000 (UTC)`]);
|
|
res.end();
|
|
});
|
|
await request.get(`${server.PREFIX}/setcookie.html`);
|
|
await request.get(`${server.PREFIX}/removecookie.html`);
|
|
const [serverRequest] = await Promise.all([
|
|
server.waitForRequest('/empty.html'),
|
|
request.get(server.EMPTY_PAGE)
|
|
]);
|
|
expect(serverRequest.headers.cookie).toBeFalsy();
|
|
});
|
|
|
|
it('should store cookie from Set-Cookie header even if it contains equal signs', async ({ request, server }) => {
|
|
it.info().annotations.push({ type: 'issue', description: 'https://github.com/microsoft/playwright/issues/11612' });
|
|
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', ['f=value == value=; secure; httpOnly; path=/some=value']);
|
|
res.end();
|
|
});
|
|
|
|
await request.get(`http://a.b.one.com:${server.PORT}/setcookie.html`, { __testHookLookup } as any);
|
|
const state = await request.storageState();
|
|
expect(state).toEqual({
|
|
'cookies': [
|
|
{
|
|
domain: 'a.b.one.com',
|
|
expires: -1,
|
|
name: 'f',
|
|
path: '/some=value',
|
|
sameSite: 'Lax',
|
|
httpOnly: true,
|
|
secure: true,
|
|
value: 'value == value=',
|
|
}
|
|
],
|
|
'origins': []
|
|
});
|
|
});
|
|
|
|
it('should override cookie from Set-Cookie header', async ({ request, server }) => {
|
|
const tomorrow = new Date();
|
|
tomorrow.setDate(tomorrow.getDate() + 1);
|
|
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', [`a=old; expires=${tomorrow.toUTCString()}`]);
|
|
res.end();
|
|
});
|
|
|
|
const dayAfterTomorrow = new Date(tomorrow);
|
|
dayAfterTomorrow.setDate(tomorrow.getDate() + 1);
|
|
const dayAfterTomorrowInSeconds = Math.floor(dayAfterTomorrow.valueOf() / 1000);
|
|
server.setRoute('/updatecookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', [`a=new; expires=${dayAfterTomorrow.toUTCString()}`]);
|
|
res.end();
|
|
});
|
|
|
|
await request.get(`${server.PREFIX}/setcookie.html`);
|
|
await request.get(`${server.PREFIX}/updatecookie.html`);
|
|
|
|
const state = await request.storageState();
|
|
|
|
expect(state.cookies).toHaveLength(1);
|
|
expect(state.cookies[0].name).toBe(`a`);
|
|
expect(state.cookies[0].value).toBe(`new`);
|
|
expect(state.cookies[0].expires).toBe(dayAfterTomorrowInSeconds);
|
|
});
|
|
|
|
it('should override cookie from Set-Cookie header even if it expired', async ({ request, server }) => {
|
|
const tomorrow = new Date();
|
|
tomorrow.setDate(tomorrow.getDate() + 1);
|
|
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', [`a=ok`, `b=ok; expires=${tomorrow.toUTCString()}`]);
|
|
res.end();
|
|
});
|
|
|
|
server.setRoute('/unsetsetcookie.html', (req, res) => {
|
|
const pastDateString = new Date(1970, 0, 1, 0, 0, 0, 0).toUTCString();
|
|
res.setHeader('Set-Cookie', [`a=; expires=${pastDateString}`, `b=; expires=${pastDateString}`]);
|
|
res.end();
|
|
});
|
|
|
|
await request.get(`${server.PREFIX}/setcookie.html`);
|
|
await request.get(`${server.PREFIX}/unsetsetcookie.html`);
|
|
|
|
const [serverRequest] = await Promise.all([
|
|
server.waitForRequest('/empty.html'),
|
|
request.get(server.EMPTY_PAGE)
|
|
]);
|
|
|
|
expect(serverRequest.headers.cookie).toBeFalsy();
|
|
});
|
|
|
|
it('should export cookies to storage state', async ({ request, server }) => {
|
|
const expires = new Date('12/31/2100 PST');
|
|
server.setRoute('/setcookie.html', (req, res) => {
|
|
res.setHeader('Set-Cookie', ['a=b', `c=d; expires=${expires.toUTCString()}; domain=b.one.com; path=/input`, 'e=f; domain=b.one.com; path=/input/subfolder']);
|
|
res.end();
|
|
});
|
|
await request.get(`http://a.b.one.com:${server.PORT}/setcookie.html`, { __testHookLookup } as any);
|
|
const state = await request.storageState();
|
|
expect(state).toEqual({
|
|
'cookies': [
|
|
{
|
|
'name': 'a',
|
|
'value': 'b',
|
|
'domain': 'a.b.one.com',
|
|
'path': '/',
|
|
'expires': -1,
|
|
'httpOnly': false,
|
|
'secure': false,
|
|
'sameSite': 'Lax'
|
|
},
|
|
{
|
|
'name': 'c',
|
|
'value': 'd',
|
|
'domain': '.b.one.com',
|
|
'path': '/input',
|
|
'expires': +expires / 1000,
|
|
'httpOnly': false,
|
|
'secure': false,
|
|
'sameSite': 'Lax'
|
|
},
|
|
{
|
|
'name': 'e',
|
|
'value': 'f',
|
|
'domain': '.b.one.com',
|
|
'path': '/input/subfolder',
|
|
'expires': -1,
|
|
'httpOnly': false,
|
|
'secure': false,
|
|
'sameSite': 'Lax'
|
|
}
|
|
],
|
|
'origins': []
|
|
});
|
|
});
|
|
|
|
it('should preserve local storage on import/export of storage state', async ({ playwright, server }) => {
|
|
const storageState: StorageStateType = {
|
|
cookies: [
|
|
{
|
|
'name': 'a',
|
|
'value': 'b',
|
|
'domain': 'a.b.one.com',
|
|
'path': '/',
|
|
'expires': -1,
|
|
'httpOnly': false,
|
|
'secure': false,
|
|
'sameSite': 'Lax'
|
|
}
|
|
],
|
|
origins: [
|
|
{
|
|
origin: 'https://www.example.com',
|
|
localStorage: [{
|
|
name: 'name1',
|
|
value: 'value1'
|
|
}]
|
|
},
|
|
]
|
|
};
|
|
const request = await playwright.request.newContext({ storageState });
|
|
await request.get(server.EMPTY_PAGE);
|
|
const exportedState = await request.storageState();
|
|
expect(exportedState).toEqual(storageState);
|
|
await request.dispose();
|
|
});
|
|
|
|
it('should send cookies from storage state', async ({ playwright, server }) => {
|
|
const expires = new Date('12/31/2099 PST');
|
|
const storageState: StorageStateType = {
|
|
'cookies': [
|
|
{
|
|
'name': 'a',
|
|
'value': 'b',
|
|
'domain': 'a.b.one.com',
|
|
'path': '/',
|
|
'expires': -1,
|
|
'httpOnly': false,
|
|
'secure': false,
|
|
'sameSite': 'Lax'
|
|
},
|
|
{
|
|
'name': 'c',
|
|
'value': 'd',
|
|
'domain': '.b.one.com',
|
|
'path': '/first/',
|
|
'expires': +expires / 1000,
|
|
'httpOnly': false,
|
|
'secure': false,
|
|
'sameSite': 'Lax'
|
|
},
|
|
{
|
|
'name': 'e',
|
|
'value': 'f',
|
|
'domain': '.b.one.com',
|
|
'path': '/first/second',
|
|
'expires': -1,
|
|
'httpOnly': false,
|
|
'secure': false,
|
|
'sameSite': 'Lax'
|
|
}
|
|
],
|
|
'origins': []
|
|
};
|
|
const request = await playwright.request.newContext({ storageState });
|
|
const [serverRequest] = await Promise.all([
|
|
server.waitForRequest('/first/second/third/not_found.html'),
|
|
request.get(`http://www.a.b.one.com:${server.PORT}/first/second/third/not_found.html`, { __testHookLookup } as any)
|
|
]);
|
|
expect(serverRequest.headers.cookie).toBe('c=d; e=f');
|
|
});
|
|
|
|
it('storage state should round-trip through file', async ({ playwright, server }, testInfo) => {
|
|
const storageState: StorageStateType = {
|
|
'cookies': [
|
|
{
|
|
'name': 'a',
|
|
'value': 'b',
|
|
'domain': 'a.b.one.com',
|
|
'path': '/',
|
|
'expires': -1,
|
|
'httpOnly': false,
|
|
'secure': false,
|
|
'sameSite': 'Lax'
|
|
}
|
|
],
|
|
'origins': []
|
|
};
|
|
|
|
const request1 = await playwright.request.newContext({ storageState });
|
|
const path = testInfo.outputPath('storage-state.json');
|
|
const state1 = await request1.storageState({ path });
|
|
expect(state1).toEqual(storageState);
|
|
|
|
const written = await fs.promises.readFile(path, 'utf8');
|
|
expect(JSON.stringify(state1, undefined, 2)).toBe(written);
|
|
|
|
const request2 = await playwright.request.newContext({ storageState: path });
|
|
const state2 = await request2.storageState();
|
|
expect(state2).toEqual(storageState);
|
|
});
|
|
|
|
it('should work with empty storage state', async ({ playwright, server }, testInfo) => {
|
|
const storageState = testInfo.outputPath('storage-state.json');
|
|
await fs.promises.writeFile(storageState, '{}');
|
|
const request1 = await playwright.request.newContext({ storageState });
|
|
|
|
const state1 = await request1.storageState();
|
|
expect(state1).toEqual({ cookies: [], origins: [] });
|
|
await expect(await request1.get(server.EMPTY_PAGE)).toBeOK();
|
|
await request1.dispose();
|
|
});
|