Implement Page.uploadFile (#61)

This patch implements `Page.uploadFile` method to support file upload inputs.
This commit is contained in:
JoelEinbinder 2017-07-10 11:21:46 -07:00 committed by Andrey Lushnikov
parent 739c1566a9
commit da0cde1b45
8 changed files with 75 additions and 8 deletions

View File

@ -33,6 +33,7 @@
* [page.setViewportSize(size)](#pagesetviewportsizesize) * [page.setViewportSize(size)](#pagesetviewportsizesize)
* [page.title()](#pagetitle) * [page.title()](#pagetitle)
* [page.type()](#pagetype) * [page.type()](#pagetype)
* [page.uploadFile(selector, ...filePaths)](#pageuploadfileselector-filepaths)
* [page.url()](#pageurl) * [page.url()](#pageurl)
* [page.userAgent()](#pageuseragent) * [page.userAgent()](#pageuseragent)
* [page.viewportSize()](#pageviewportsize) * [page.viewportSize()](#pageviewportsize)
@ -236,6 +237,11 @@ Pages could be closed by `page.close()` method.
#### page.type() #### page.type()
#### page.uploadFile(selector, ...filePaths)
- `selector` <[string]> A query selector to a file input
- `...filePaths` <[string]> Sets the value of the file input these paths
- returns: <[Promise]> Promise which resolves when the value is set.
#### page.url() #### page.url()
- returns: <[Promise]<[string]>> Promise which resolves with the current page url. - returns: <[Promise]<[string]>> Promise which resolves with the current page url.

View File

@ -108,7 +108,10 @@ class NetworkManager extends EventEmitter {
* @param {!Object} event * @param {!Object} event
*/ */
_onResponseReceived(event) { _onResponseReceived(event) {
let request = this._idToRequest.get(event.requestId) || null; let request = this._idToRequest.get(event.requestId);
// FileUpload sends a response without a matching request.
if (!request)
return;
let response = new Response(request, event.response, this._getResponseBody.bind(this, event.requestId)); let response = new Response(request, event.response, this._getResponseBody.bind(this, event.requestId));
request._response = response; request._response = response;
this.emit(NetworkManager.Events.Response, response); this.emit(NetworkManager.Events.Response, response);

View File

@ -493,10 +493,13 @@ class Page extends EventEmitter {
* @param {!Promise<number>} * @param {!Promise<number>}
*/ */
async _querySelector(selector) { async _querySelector(selector) {
return (await this._client.send('DOM.querySelector', { let {nodeId} = await this._client.send('DOM.querySelector', {
nodeId: await this._rootNodeId(), nodeId: await this._rootNodeId(),
selector selector
})).nodeId; });
if (!nodeId)
throw new Error('No node found for selector: ' + selector);
return nodeId;
} }
/** /**
@ -569,6 +572,18 @@ class Page extends EventEmitter {
waitFor(selector) { waitFor(selector) {
return this.mainFrame().waitFor(selector); return this.mainFrame().waitFor(selector);
} }
/**
* @param {string} selector
* @param {!Array<string>} filePaths
* @return {!Promise}
*/
async uploadFile(selector, ...filePaths) {
await this._client.send('DOM.setFileInputFiles', {
nodeId: await this._querySelector(selector),
files: filePaths
});
}
} }
/** @enum {string} */ /** @enum {string} */

View File

@ -332,6 +332,17 @@ class WebPage {
await(this._page.setContent(html)); await(this._page.setContent(html));
} }
/**
* @param {string} selector
* @param {(string|!Array<string>)} files
*/
uploadFile(selector, files) {
if (typeof files === 'string')
await(this._page.uploadFile(selector, files));
else
await(this._page.uploadFile(selector, ...files));
}
/** /**
* @param {string} html * @param {string} html
* @param {function()=} callback * @param {function()=} callback

View File

@ -0,0 +1 @@
contents of the file

View File

@ -0,0 +1,9 @@
<!DOCTYPE html>
<html>
<head>
<title>File upload test</title>
</head>
<body>
<input type="file">
</body>
</html>

View File

@ -544,6 +544,15 @@ describe('Puppeteer', function() {
await page.click('button'); await page.click('button');
expect(await page.evaluate(() => result)).toBe('Clicked'); expect(await page.evaluate(() => result)).toBe('Clicked');
})); }));
it('should fail to click a missing button', SX(async function() {
await page.navigate(STATIC_PREFIX + '/input/button.html');
try {
await page.click('button.does-not-exist');
fail('Clicking the button did not throw.');
} catch (error) {
expect(error.message).toBe('No node found for selector: button.does-not-exist');
}
}));
it('should type into the textarea', SX(async function() { it('should type into the textarea', SX(async function() {
await page.navigate(STATIC_PREFIX + '/input/textarea.html'); await page.navigate(STATIC_PREFIX + '/input/textarea.html');
await page.focus('textarea'); await page.focus('textarea');
@ -557,6 +566,21 @@ describe('Puppeteer', function() {
await page.click('button'); await page.click('button');
expect(await page.evaluate(() => result)).toBe('Clicked'); expect(await page.evaluate(() => result)).toBe('Clicked');
})); }));
it('should upload the file', SX(async function(){
await page.navigate(STATIC_PREFIX + '/input/fileupload.html');
await page.uploadFile('input', __dirname + '/assets/file-to-upload.txt');
expect(await page.evaluate(() => {
let input = document.querySelector('input');
return input.files[0].name;
})).toBe('file-to-upload.txt');
expect(await page.evaluate(() => {
let input = document.querySelector('input');
let reader = new FileReader();
let promise = new Promise(fulfill => reader.onload = fulfill);
reader.readAsText(input.files[0]);
return promise.then(() => reader.result);
})).toBe('contents of the file');
}));
}); });
describe('Page.setUserAgent', function() { describe('Page.setUserAgent', function() {
it('should work', SX(async function() { it('should work', SX(async function() {

View File

@ -1,11 +1,9 @@
//! unsupported
// Note: uses various files in module/webpage as things to be uploaded. // Note: uses various files in module/webpage as things to be uploaded.
// Which files they are doesn't matter. // Which files they are doesn't matter.
var page; var page;
setup(function () { setup(function () {
page = new WebPage(); page = require('webpage').create();
page.content = page.content =
'<input type="file" id="file">\n' + '<input type="file" id="file">\n' +
'<input type="file" id="file2" multiple>\n' + '<input type="file" id="file2" multiple>\n' +
@ -21,7 +19,7 @@ function test_one_elt(id, names) {
var elt = document.getElementById(id); var elt = document.getElementById(id);
var rv = []; var rv = [];
for (var i = 0; i < elt.files.length; i++) { for (var i = 0; i < elt.files.length; i++) {
rv.push(elt.files[i].fileName); rv.push(elt.files[i].name);
} }
return rv; return rv;
}, id); }, id);
@ -32,7 +30,7 @@ generate_tests(test_one_elt, [
["single upload single file", "file", ["file-upload.js"]], ["single upload single file", "file", ["file-upload.js"]],
["multiple upload single file", "file2", ["file-upload.js"]], ["multiple upload single file", "file2", ["file-upload.js"]],
["multiple upload multiple file", "file3", ["file-upload.js", "object.js"]], ["multiple upload multiple file", "file3", ["file-upload.js", "object.js"]],
], { expected_fail: true }); ], { expected_fail: false });
async_test(function () { async_test(function () {
page.onFilePicker = this.step_func(function (oldFile) { page.onFilePicker = this.step_func(function (oldFile) {