mirror of
				https://github.com/microsoft/playwright.git
				synced 2025-06-26 21:40:17 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			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();
 | |
| });
 | 
