mirror of
https://github.com/microsoft/playwright.git
synced 2024-12-11 04:30:17 +03:00
350 lines
13 KiB
TypeScript
350 lines
13 KiB
TypeScript
/**
|
|
* Copyright 2018 Google Inc. All rights reserved.
|
|
* Modifications 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.
|
|
*/
|
|
|
|
import fs from 'fs';
|
|
import url from 'url';
|
|
import { expect, test as it } from './pageTest';
|
|
|
|
it('should work @smoke', async ({ page, server }) => {
|
|
server.setRoute('/empty.html', (req, res) => {
|
|
res.setHeader('foo', 'bar');
|
|
res.setHeader('BaZ', 'bAz');
|
|
res.end();
|
|
});
|
|
const response = await page.goto(server.EMPTY_PAGE);
|
|
expect((await response.allHeaders())['foo']).toBe('bar');
|
|
expect((await response.allHeaders())['baz']).toBe('bAz');
|
|
expect((await response.allHeaders())['BaZ']).toBe(undefined);
|
|
});
|
|
|
|
it('should return multiple header value', async ({ page, server, browserName, platform }) => {
|
|
it.skip(browserName === 'webkit' && platform === 'win32', 'libcurl does not support non-set-cookie multivalue headers');
|
|
|
|
server.setRoute('/headers', (req, res) => {
|
|
// Headers array is only supported since Node v14.14.0 so we write directly to the socket.
|
|
// res.writeHead(200, ['name-a', 'v1','name-b', 'v4','Name-a', 'v2', 'name-A', 'v3']);
|
|
const conn = res.connection;
|
|
conn.write('HTTP/1.1 200 OK\r\n');
|
|
conn.write('Name-A: v1\r\n');
|
|
conn.write('Name-a: v2\r\n');
|
|
conn.write('name-A: v3\r\n');
|
|
conn.write('\r\n');
|
|
conn.uncork();
|
|
conn.end();
|
|
});
|
|
const response = await page.goto(`${server.PREFIX}/headers`);
|
|
expect(response.status()).toBe(200);
|
|
expect(response.headers()['name-a']).toBe('v1, v2, v3');
|
|
});
|
|
|
|
it('should return text', async ({ page, server }) => {
|
|
const response = await page.goto(server.PREFIX + '/simple.json');
|
|
expect(await response.text()).toBe('{"foo": "bar"}\n');
|
|
});
|
|
|
|
it('should return uncompressed text', async ({ page, server }) => {
|
|
server.enableGzip('/simple.json');
|
|
const response = await page.goto(server.PREFIX + '/simple.json');
|
|
expect(response.headers()['content-encoding']).toBe('gzip');
|
|
expect(await response.text()).toBe('{"foo": "bar"}\n');
|
|
});
|
|
|
|
it('should throw when requesting body of redirected response', async ({ page, server }) => {
|
|
server.setRedirect('/foo.html', '/empty.html');
|
|
const response = await page.goto(server.PREFIX + '/foo.html');
|
|
const redirectedFrom = response.request().redirectedFrom();
|
|
expect(redirectedFrom).toBeTruthy();
|
|
const redirected = await redirectedFrom.response();
|
|
expect(redirected.status()).toBe(302);
|
|
let error = null;
|
|
await redirected.text().catch(e => error = e);
|
|
expect(error.message).toContain('Response body is unavailable for redirect responses');
|
|
});
|
|
|
|
it('should wait until response completes', async ({ page, server }) => {
|
|
await page.goto(server.EMPTY_PAGE);
|
|
// Setup server to trap request.
|
|
let serverResponse = null;
|
|
server.setRoute('/get', (req, res) => {
|
|
serverResponse = res;
|
|
// In Firefox, |fetch| will be hanging until it receives |Content-Type| header
|
|
// from server.
|
|
res.setHeader('Content-Type', 'text/plain; charset=utf-8');
|
|
res.write('hello ');
|
|
});
|
|
// Setup page to trap response.
|
|
let requestFinished = false;
|
|
page.on('requestfinished', r => requestFinished = requestFinished || r.url().includes('/get'));
|
|
// send request and wait for server response
|
|
const [pageResponse] = await Promise.all([
|
|
page.waitForEvent('response'),
|
|
page.evaluate(() => fetch('./get', { method: 'GET' })),
|
|
server.waitForRequest('/get'),
|
|
]);
|
|
|
|
expect(serverResponse).toBeTruthy();
|
|
expect(pageResponse).toBeTruthy();
|
|
expect(pageResponse.status()).toBe(200);
|
|
expect(requestFinished).toBe(false);
|
|
|
|
const responseText = pageResponse.text();
|
|
// Write part of the response and wait for it to be flushed.
|
|
await new Promise(x => serverResponse.write('wor', x));
|
|
// Finish response.
|
|
await new Promise(x => serverResponse.end('ld!', x));
|
|
expect(await responseText).toBe('hello world!');
|
|
});
|
|
|
|
it('should reject response.finished if page closes', async ({ page, server }) => {
|
|
await page.goto(server.EMPTY_PAGE);
|
|
server.setRoute('/get', (req, res) => {
|
|
// In Firefox, |fetch| will be hanging until it receives |Content-Type| header
|
|
// from server.
|
|
res.setHeader('Content-Type', 'text/plain; charset=utf-8');
|
|
res.write('hello ');
|
|
});
|
|
// send request and wait for server response
|
|
const [pageResponse] = await Promise.all([
|
|
page.waitForEvent('response'),
|
|
page.evaluate(() => fetch('./get', { method: 'GET' })),
|
|
]);
|
|
|
|
const finishPromise = pageResponse.finished().catch(e => e);
|
|
await page.close();
|
|
const error = await finishPromise;
|
|
expect(error.message).toContain('closed');
|
|
});
|
|
|
|
it('should reject response.finished if context closes', async ({ page, server }) => {
|
|
await page.goto(server.EMPTY_PAGE);
|
|
server.setRoute('/get', (req, res) => {
|
|
// In Firefox, |fetch| will be hanging until it receives |Content-Type| header
|
|
// from server.
|
|
res.setHeader('Content-Type', 'text/plain; charset=utf-8');
|
|
res.write('hello ');
|
|
});
|
|
// send request and wait for server response
|
|
const [pageResponse] = await Promise.all([
|
|
page.waitForEvent('response'),
|
|
page.evaluate(() => fetch('./get', { method: 'GET' })),
|
|
]);
|
|
|
|
const finishPromise = pageResponse.finished().catch(e => e);
|
|
await page.context().close();
|
|
const error = await finishPromise;
|
|
expect(error.message).toContain('closed');
|
|
});
|
|
|
|
it('should return json', async ({ page, server }) => {
|
|
const response = await page.goto(server.PREFIX + '/simple.json');
|
|
expect(await response.json()).toEqual({ foo: 'bar' });
|
|
});
|
|
|
|
it('should return body', async ({ page, server, asset }) => {
|
|
const response = await page.goto(server.PREFIX + '/pptr.png');
|
|
const imageBuffer = fs.readFileSync(asset('pptr.png'));
|
|
const responseBuffer = await response.body();
|
|
expect(responseBuffer.equals(imageBuffer)).toBe(true);
|
|
});
|
|
|
|
it('should return body with compression', async ({ page, server, asset }) => {
|
|
server.enableGzip('/pptr.png');
|
|
const response = await page.goto(server.PREFIX + '/pptr.png');
|
|
const imageBuffer = fs.readFileSync(asset('pptr.png'));
|
|
const responseBuffer = await response.body();
|
|
expect(responseBuffer.equals(imageBuffer)).toBe(true);
|
|
});
|
|
|
|
it('should return status text', async ({ page, server }) => {
|
|
server.setRoute('/cool', (req, res) => {
|
|
res.writeHead(200, 'cool!');
|
|
res.end();
|
|
});
|
|
const response = await page.goto(server.PREFIX + '/cool');
|
|
expect(response.statusText()).toBe('cool!');
|
|
});
|
|
|
|
it('should report all headers', async ({ page, server, browserName, platform, isElectron, browserMajorVersion }) => {
|
|
it.skip(isElectron && browserMajorVersion < 99, 'This needs Chromium >= 99');
|
|
it.skip(browserName === 'webkit' && platform === 'win32', 'libcurl does not support non-set-cookie multivalue headers');
|
|
|
|
const expectedHeaders = {
|
|
'header-a': ['value-a', 'value-a-1', 'value-a-2'],
|
|
'header-b': ['value-b'],
|
|
};
|
|
server.setRoute('/headers', (req, res) => {
|
|
res.writeHead(200, expectedHeaders);
|
|
res.end();
|
|
});
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
const [response] = await Promise.all([
|
|
page.waitForResponse('**/*'),
|
|
page.evaluate(() => fetch('/headers'))
|
|
]);
|
|
const headers = await response.headersArray();
|
|
const actualHeaders = {};
|
|
for (const { name, value } of headers) {
|
|
if (!actualHeaders[name])
|
|
actualHeaders[name] = [];
|
|
actualHeaders[name].push(value);
|
|
}
|
|
delete actualHeaders['Keep-Alive'];
|
|
delete actualHeaders['keep-alive'];
|
|
delete actualHeaders['Connection'];
|
|
delete actualHeaders['connection'];
|
|
delete actualHeaders['Date'];
|
|
delete actualHeaders['date'];
|
|
delete actualHeaders['Transfer-Encoding'];
|
|
delete actualHeaders['transfer-encoding'];
|
|
expect(actualHeaders).toEqual(expectedHeaders);
|
|
});
|
|
|
|
it('should report multiple set-cookie headers', async ({ page, server, isElectron, browserMajorVersion }) => {
|
|
it.skip(isElectron && browserMajorVersion < 99, 'This needs Chromium >= 99');
|
|
|
|
server.setRoute('/headers', (req, res) => {
|
|
res.writeHead(200, {
|
|
'Set-Cookie': ['a=b', 'c=d']
|
|
});
|
|
res.write('\r\n');
|
|
res.end();
|
|
});
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
const [response] = await Promise.all([
|
|
page.waitForResponse('**/*'),
|
|
page.evaluate(() => fetch('/headers'))
|
|
]);
|
|
const headers = await response.headersArray();
|
|
const cookies = headers.filter(({ name }) => name.toLowerCase() === 'set-cookie').map(({ value }) => value);
|
|
expect(cookies).toEqual(['a=b', 'c=d']);
|
|
expect(await response.headerValue('not-there')).toEqual(null);
|
|
expect(await response.headerValue('set-cookie')).toEqual('a=b\nc=d');
|
|
expect(await response.headerValues('set-cookie')).toEqual(['a=b', 'c=d']);
|
|
});
|
|
|
|
it('should behave the same way for headers and allHeaders', async ({ page, server, browserName, platform }) => {
|
|
it.skip(browserName === 'webkit' && platform === 'win32', 'libcurl does not support non-set-cookie multivalue headers');
|
|
server.setRoute('/headers', (req, res) => {
|
|
const headers = {
|
|
'Set-Cookie': ['a=b', 'c=d'],
|
|
'header-a': ['a=b', 'c=d'],
|
|
'Name-A': 'v1',
|
|
'name-b': 'v4',
|
|
'Name-a': 'v2',
|
|
'name-A': 'v3',
|
|
};
|
|
// Chromium does not report set-cookie headers immediately, so they are missing from .headers()
|
|
if (browserName === 'chromium')
|
|
delete headers['Set-Cookie'];
|
|
|
|
res.writeHead(200, headers);
|
|
res.write('\r\n');
|
|
res.end();
|
|
});
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
const [response] = await Promise.all([
|
|
page.waitForResponse('**/*'),
|
|
page.evaluate(() => fetch('/headers'))
|
|
]);
|
|
const allHeaders = await response.allHeaders();
|
|
expect(response.headers()).toEqual(allHeaders);
|
|
expect(allHeaders['header-a']).toEqual('a=b, c=d');
|
|
expect(allHeaders['name-a']).toEqual('v1, v2, v3');
|
|
expect(allHeaders['name-b']).toEqual('v4');
|
|
});
|
|
|
|
it('should provide a Response with a file URL', async ({ page, asset, isAndroid, isElectron, isWindows, browserName, browserMajorVersion, mode }) => {
|
|
it.skip(isAndroid, 'No files on Android');
|
|
it.skip(browserName === 'firefox', 'Firefox does return null for file:// URLs');
|
|
it.skip(mode.startsWith('service'));
|
|
|
|
const fileurl = url.pathToFileURL(asset('frames/two-frames.html')).href;
|
|
const response = await page.goto(fileurl);
|
|
if (isElectron || (browserName === 'chromium' && browserMajorVersion >= 99) || (browserName === 'webkit' && isWindows))
|
|
expect(response.status()).toBe(200);
|
|
else
|
|
expect(response.status()).toBe(0);
|
|
expect(response.ok()).toBe(true);
|
|
});
|
|
|
|
it('should return set-cookie header after route.fulfill', async ({ page, server, browserName }) => {
|
|
it.fail(browserName === 'webkit' || browserName === 'chromium', 'https://github.com/microsoft/playwright/issues/11035');
|
|
await page.route('**/*', async route => {
|
|
await route.fulfill({
|
|
status: 200,
|
|
headers: {
|
|
'set-cookie': 'a=b'
|
|
},
|
|
contentType: 'text/plain',
|
|
body: ''
|
|
});
|
|
});
|
|
const response = await page.goto(server.EMPTY_PAGE);
|
|
const headers = await response.allHeaders();
|
|
expect(headers['set-cookie']).toBe('a=b');
|
|
});
|
|
|
|
it('should return headers after route.fulfill', async ({ page, server }) => {
|
|
await page.route('**/*', async route => {
|
|
await route.fulfill({
|
|
status: 200,
|
|
headers: {
|
|
'foo': 'bar',
|
|
'content-language': 'en'
|
|
},
|
|
contentType: 'text/plain',
|
|
body: 'done'
|
|
});
|
|
});
|
|
const response = await page.goto(server.EMPTY_PAGE);
|
|
expect(await response.allHeaders()).toEqual({
|
|
'foo': 'bar',
|
|
'content-type': 'text/plain',
|
|
'content-length': '4',
|
|
'content-language': 'en'
|
|
});
|
|
});
|
|
|
|
it('should report if request was fromServiceWorker', async ({ page, server, isAndroid, isElectron }) => {
|
|
it.skip(isAndroid || isElectron);
|
|
{
|
|
const res = await page.goto(server.PREFIX + '/serviceworkers/fetch/sw.html');
|
|
expect(res.fromServiceWorker()).toBe(false);
|
|
}
|
|
await page.evaluate(() => window['activationPromise']);
|
|
{
|
|
const [res] = await Promise.all([
|
|
page.waitForResponse(/example\.txt/),
|
|
page.evaluate(() => fetch('/example.txt')),
|
|
]);
|
|
expect(res.fromServiceWorker()).toBe(true);
|
|
}
|
|
});
|
|
|
|
it('should return body for prefetch script', async ({ page, server, browserName }) => {
|
|
it.skip(browserName === 'webkit', 'No prefetch in WebKit: https://caniuse.com/link-rel-prefetch');
|
|
const [response] = await Promise.all([
|
|
page.waitForResponse('**/prefetch.js'),
|
|
page.goto(server.PREFIX + '/prefetch.html')
|
|
]);
|
|
const body = await response.body();
|
|
expect(body.toString()).toBe('// Scripts will be pre-fetched');
|
|
});
|