-
Notifications
You must be signed in to change notification settings - Fork 0
/
phantomjs.js
282 lines (249 loc) · 9.19 KB
/
phantomjs.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
// Licensed to the Software Freedom Conservancy (SFC) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The SFC licenses this file
// to you 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.
/**
* @fileoverview Defines a {@linkplain Driver WebDriver} client for the
* PhantomJS web browser. By default, it is expected that the PhantomJS
* executable can be located on your
* [PATH](https://en.wikipedia.org/wiki/PATH_(variable))
*
* __Using a Custom PhantomJS Binary__
*
* If you have PhantomJS.exe placed somewhere other than the root of your
* working directory, you can build a custom Capability and attach the
* executable's location to the Capability
*
* For example, if you're using the
* [phantomjs-prebuilt](https://www.npmjs.com/package/phantomjs-prebuilt) module
* from npm:
*
* //setup custom phantomJS capability
* var phantomjs_exe = require('phantomjs-prebuilt').path;
* var customPhantom = selenium.Capabilities.phantomjs();
* customPhantom.set("phantomjs.binary.path", phantomjs_exe);
* //build custom phantomJS driver
* var driver = new selenium.Builder().
* withCapabilities(customPhantom).
* build();
*
*/
'use strict';
const fs = require('fs');
const http = require('./http'),
io = require('./io'),
capabilities = require('./lib/capabilities'),
command = require('./lib/command'),
logging = require('./lib/logging'),
promise = require('./lib/promise'),
webdriver = require('./lib/webdriver'),
portprober = require('./net/portprober'),
remote = require('./remote');
/**
* Name of the PhantomJS executable.
* @type {string}
* @const
*/
const PHANTOMJS_EXE =
process.platform === 'win32' ? 'phantomjs.exe' : 'phantomjs';
/**
* Capability that designates the location of the PhantomJS executable to use.
* @type {string}
* @const
*/
const BINARY_PATH_CAPABILITY = 'phantomjs.binary.path';
/**
* Capability that designates the CLI arguments to pass to PhantomJS.
* @type {string}
* @const
*/
const CLI_ARGS_CAPABILITY = 'phantomjs.cli.args';
/**
* Custom command names supported by PhantomJS.
* @enum {string}
*/
const Command = {
EXECUTE_PHANTOM_SCRIPT: 'executePhantomScript'
};
/**
* Finds the PhantomJS executable.
* @param {string=} opt_exe Path to the executable to use.
* @return {string} The located executable.
* @throws {Error} If the executable cannot be found on the PATH, or if the
* provided executable path does not exist.
*/
function findExecutable(opt_exe) {
var exe = opt_exe || io.findInPath(PHANTOMJS_EXE, true);
if (!exe) {
throw Error(
'The PhantomJS executable could not be found on the current PATH. ' +
'Please download the latest version from ' +
'http://phantomjs.org/download.html and ensure it can be found on ' +
'your PATH. For more information, see ' +
'https://github.com/ariya/phantomjs/wiki');
}
if (!fs.existsSync(exe)) {
throw Error('File does not exist: ' + exe);
}
return exe;
}
/**
* Maps WebDriver logging level name to those recognised by PhantomJS.
* @const {!Map<string, string>}
*/
const WEBDRIVER_TO_PHANTOMJS_LEVEL = new Map([
[logging.Level.ALL.name, 'DEBUG'],
[logging.Level.DEBUG.name, 'DEBUG'],
[logging.Level.INFO.name, 'INFO'],
[logging.Level.WARNING.name, 'WARN'],
[logging.Level.SEVERE.name, 'ERROR']]);
/**
* Creates a command executor with support for PhantomJS' custom commands.
* @param {!Promise<string>} url The server's URL.
* @return {!command.Executor} The new command executor.
*/
function createExecutor(url) {
let client = url.then(url => new http.HttpClient(url));
let executor = new http.Executor(client);
executor.defineCommand(
Command.EXECUTE_PHANTOM_SCRIPT,
'POST', '/session/:sessionId/phantom/execute');
return executor;
}
/**
* Creates a new WebDriver client for PhantomJS.
*/
class Driver extends webdriver.WebDriver {
/**
* Creates a new PhantomJS session.
*
* @param {capabilities.Capabilities=} opt_capabilities The desired
* capabilities.
* @param {promise.ControlFlow=} opt_flow The control flow to use,
* or {@code null} to use the currently active flow.
* @param {string=} opt_logFile Path to the log file for the phantomjs
* executable's output. For convenience, this may be set at runtime with
* the `SELENIUM_PHANTOMJS_LOG` environment variable.
* @return {!Driver} A new driver reference.
*/
static createSession(opt_capabilities, opt_flow, opt_logFile) {
// TODO: add an Options class for consistency with the other driver types.
var caps = opt_capabilities || capabilities.Capabilities.phantomjs();
var exe = findExecutable(caps.get(BINARY_PATH_CAPABILITY));
var args = [];
var logPrefs = caps.get(capabilities.Capability.LOGGING_PREFS);
if (logPrefs instanceof logging.Preferences) {
logPrefs = logPrefs.toJSON();
}
if (logPrefs && logPrefs[logging.Type.DRIVER]) {
let level = WEBDRIVER_TO_PHANTOMJS_LEVEL.get(
logPrefs[logging.Type.DRIVER]);
if (level) {
args.push('--webdriver-loglevel=' + level);
}
}
opt_logFile = process.env['SELENIUM_PHANTOMJS_LOG'] || opt_logFile;
if (typeof opt_logFile === 'string') {
args.push('--webdriver-logfile=' + opt_logFile);
}
var proxy = caps.get(capabilities.Capability.PROXY);
if (proxy) {
switch (proxy.proxyType) {
case 'manual':
if (proxy.httpProxy) {
args.push(
'--proxy-type=http',
'--proxy=' + proxy.httpProxy);
console.log(args);
}
break;
case 'pac':
throw Error('PhantomJS does not support Proxy PAC files');
case 'system':
args.push('--proxy-type=system');
break;
case 'direct':
args.push('--proxy-type=none');
break;
}
}
args = args.concat(caps.get(CLI_ARGS_CAPABILITY) || []);
var port = portprober.findFreePort();
var service = new remote.DriverService(exe, {
port: port,
args: Promise.resolve(port).then(function(port) {
args.push('--webdriver=' + port);
return args;
})
});
var executor = createExecutor(service.start());
return /** @type {!Driver} */(webdriver.WebDriver.createSession(
executor, caps, opt_flow, this, () => service.kill()));
}
/**
* This function is a no-op as file detectors are not supported by this
* implementation.
* @override
*/
setFileDetector() {}
/**
* Executes a PhantomJS fragment. This method is similar to
* {@link #executeScript}, except it exposes the
* <a href="http://phantomjs.org/api/">PhantomJS API</a> to the injected
* script.
*
* <p>The injected script will execute in the context of PhantomJS's
* {@code page} variable. If a page has not been loaded before calling this
* method, one will be created.</p>
*
* <p>Be sure to wrap callback definitions in a try/catch block, as failures
* may cause future WebDriver calls to fail.</p>
*
* <p>Certain callbacks are used by GhostDriver (the PhantomJS WebDriver
* implementation) and overriding these may cause the script to fail. It is
* recommended that you check for existing callbacks before defining your own.
* </p>
*
* As with {@link #executeScript}, the injected script may be defined as
* a string for an anonymous function body (e.g. "return 123;"), or as a
* function. If a function is provided, it will be decompiled to its original
* source. Note that injecting functions is provided as a convenience to
* simplify defining complex scripts. Care must be taken that the function
* only references variables that will be defined in the page's scope and
* that the function does not override {@code Function.prototype.toString}
* (overriding toString() will interfere with how the function is
* decompiled.
*
* @param {(string|!Function)} script The script to execute.
* @param {...*} var_args The arguments to pass to the script.
* @return {!promise.Thenable<T>} A promise that resolve to the
* script's return value.
* @template T
*/
executePhantomJS(script, var_args) {
if (typeof script === 'function') {
script = 'return (' + script + ').apply(this, arguments);';
}
var args = arguments.length > 1
? Array.prototype.slice.call(arguments, 1) : [];
return this.schedule(
new command.Command(Command.EXECUTE_PHANTOM_SCRIPT)
.setParameter('script', script)
.setParameter('args', args),
'Driver.executePhantomJS()');
}
}
// PUBLIC API
exports.Driver = Driver;