UNPKG

24.3 kBJavaScriptView Raw
1// Licensed to the Software Freedom Conservancy (SFC) under one
2// or more contributor license agreements. See the NOTICE file
3// distributed with this work for additional information
4// regarding copyright ownership. The SFC licenses this file
5// to you under the Apache License, Version 2.0 (the
6// "License"); you may not use this file except in compliance
7// with the License. You may obtain a copy of the License at
8//
9// http://www.apache.org/licenses/LICENSE-2.0
10//
11// Unless required by applicable law or agreed to in writing,
12// software distributed under the License is distributed on an
13// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
14// KIND, either express or implied. See the License for the
15// specific language governing permissions and limitations
16// under the License.
17
18/**
19 * @fileoverview Defines an environment agnostic {@linkplain cmd.Executor
20 * command executor} that communicates with a remote end using JSON over HTTP.
21 *
22 * Clients should implement the {@link Client} interface, which is used by
23 * the {@link Executor} to send commands to the remote end.
24 */
25
26'use strict'
27
28const path = require('path')
29const cmd = require('./command')
30const error = require('./error')
31const logging = require('./logging')
32const promise = require('./promise')
33const { Session } = require('./session')
34const { WebElement } = require('./webdriver')
35
36const getAttribute = requireAtom(
37 'get-attribute.js',
38 '//javascript/node/selenium-webdriver/lib/atoms:get-attribute.js'
39)
40const isDisplayed = requireAtom(
41 'is-displayed.js',
42 '//javascript/node/selenium-webdriver/lib/atoms:is-displayed.js'
43)
44const findElements = requireAtom(
45 'find-elements.js',
46 '//javascript/node/selenium-webdriver/lib/atoms:find-elements.js'
47)
48
49/**
50 * @param {string} module
51 * @param {string} bazelTarget
52 * @return {!Function}
53 */
54function requireAtom(module, bazelTarget) {
55 try {
56 return require('./atoms/' + module)
57 } catch (ex) {
58 try {
59 const file = bazelTarget.slice(2).replace(':', '/')
60 console.log(`../../../bazel-bin/${file}`)
61 return require(path.resolve(`../../../bazel-bin/${file}`))
62 } catch (ex2) {
63 console.log(ex2)
64 throw Error(
65 `Failed to import atoms module ${module}. If running in dev mode, you` +
66 ` need to run \`bazel build ${bazelTarget}\` from the project` +
67 `root: ${ex}`
68 )
69 }
70 }
71}
72
73/**
74 * Converts a headers map to a HTTP header block string.
75 * @param {!Map<string, string>} headers The map to convert.
76 * @return {string} The headers as a string.
77 */
78function headersToString(headers) {
79 const ret = []
80 headers.forEach(function (value, name) {
81 ret.push(`${name.toLowerCase()}: ${value}`)
82 })
83 return ret.join('\n')
84}
85
86/**
87 * Represents a HTTP request message. This class is a "partial" request and only
88 * defines the path on the server to send a request to. It is each client's
89 * responsibility to build the full URL for the final request.
90 * @final
91 */
92class Request {
93 /**
94 * @param {string} method The HTTP method to use for the request.
95 * @param {string} path The path on the server to send the request to.
96 * @param {Object=} opt_data This request's non-serialized JSON payload data.
97 */
98 constructor(method, path, opt_data) {
99 this.method = /** string */ method
100 this.path = /** string */ path
101 this.data = /** Object */ opt_data
102 this.headers = /** !Map<string, string> */ new Map([
103 ['Accept', 'application/json; charset=utf-8'],
104 ])
105 }
106
107 /** @override */
108 toString() {
109 let ret = `${this.method} ${this.path} HTTP/1.1\n`
110 ret += headersToString(this.headers) + '\n\n'
111 if (this.data) {
112 ret += JSON.stringify(this.data)
113 }
114 return ret
115 }
116}
117
118/**
119 * Represents a HTTP response message.
120 * @final
121 */
122class Response {
123 /**
124 * @param {number} status The response code.
125 * @param {!Object<string>} headers The response headers. All header names
126 * will be converted to lowercase strings for consistent lookups.
127 * @param {string} body The response body.
128 */
129 constructor(status, headers, body) {
130 this.status = /** number */ status
131 this.body = /** string */ body
132 this.headers = /** !Map<string, string>*/ new Map()
133 for (let header in headers) {
134 this.headers.set(header.toLowerCase(), headers[header])
135 }
136 }
137
138 /** @override */
139 toString() {
140 let ret = `HTTP/1.1 ${this.status}\n${headersToString(this.headers)}\n\n`
141 if (this.body) {
142 ret += this.body
143 }
144 return ret
145 }
146}
147
148/** @enum {!Function} */
149const Atom = {
150 GET_ATTRIBUTE: getAttribute,
151 IS_DISPLAYED: isDisplayed,
152 FIND_ELEMENTS: findElements,
153}
154
155const LOG = logging.getLogger('webdriver.http')
156
157function post(path) {
158 return resource('POST', path)
159}
160function del(path) {
161 return resource('DELETE', path)
162}
163function get(path) {
164 return resource('GET', path)
165}
166function resource(method, path) {
167 return { method: method, path: path }
168}
169
170/** @typedef {{method: string, path: string}} */
171var CommandSpec // eslint-disable-line
172
173/** @typedef {function(!cmd.Command): !cmd.Command} */
174var CommandTransformer // eslint-disable-line
175
176class InternalTypeError extends TypeError {}
177
178/**
179 * @param {!cmd.Command} command The initial command.
180 * @param {Atom} atom The name of the atom to execute.
181 * @return {!cmd.Command} The transformed command to execute.
182 */
183function toExecuteAtomCommand(command, atom, ...params) {
184 if (typeof atom !== 'function') {
185 throw new InternalTypeError('atom is not a function: ' + typeof atom)
186 }
187
188 return new cmd.Command(cmd.Name.EXECUTE_SCRIPT)
189 .setParameter('sessionId', command.getParameter('sessionId'))
190 .setParameter('script', `return (${atom}).apply(null, arguments)`)
191 .setParameter(
192 'args',
193 params.map((param) => command.getParameter(param))
194 )
195}
196
197/** @const {!Map<string, CommandSpec>} */
198const COMMAND_MAP = new Map([
199 [cmd.Name.GET_SERVER_STATUS, get('/status')],
200 [cmd.Name.NEW_SESSION, post('/session')],
201 [cmd.Name.GET_SESSIONS, get('/sessions')],
202 [cmd.Name.QUIT, del('/session/:sessionId')],
203 [cmd.Name.CLOSE, del('/session/:sessionId/window')],
204 [
205 cmd.Name.GET_CURRENT_WINDOW_HANDLE,
206 get('/session/:sessionId/window_handle'),
207 ],
208 [cmd.Name.GET_WINDOW_HANDLES, get('/session/:sessionId/window_handles')],
209 [cmd.Name.GET_CURRENT_URL, get('/session/:sessionId/url')],
210 [cmd.Name.GET, post('/session/:sessionId/url')],
211 [cmd.Name.GO_BACK, post('/session/:sessionId/back')],
212 [cmd.Name.GO_FORWARD, post('/session/:sessionId/forward')],
213 [cmd.Name.REFRESH, post('/session/:sessionId/refresh')],
214 [cmd.Name.ADD_COOKIE, post('/session/:sessionId/cookie')],
215 [cmd.Name.GET_ALL_COOKIES, get('/session/:sessionId/cookie')],
216 [cmd.Name.DELETE_ALL_COOKIES, del('/session/:sessionId/cookie')],
217 [cmd.Name.DELETE_COOKIE, del('/session/:sessionId/cookie/:name')],
218 [cmd.Name.FIND_ELEMENT, post('/session/:sessionId/element')],
219 [cmd.Name.FIND_ELEMENTS, post('/session/:sessionId/elements')],
220 [cmd.Name.GET_ACTIVE_ELEMENT, post('/session/:sessionId/element/active')],
221 [
222 cmd.Name.FIND_CHILD_ELEMENT,
223 post('/session/:sessionId/element/:id/element'),
224 ],
225 [
226 cmd.Name.FIND_CHILD_ELEMENTS,
227 post('/session/:sessionId/element/:id/elements'),
228 ],
229 [cmd.Name.CLEAR_ELEMENT, post('/session/:sessionId/element/:id/clear')],
230 [cmd.Name.CLICK_ELEMENT, post('/session/:sessionId/element/:id/click')],
231 [
232 cmd.Name.SEND_KEYS_TO_ELEMENT,
233 post('/session/:sessionId/element/:id/value'),
234 ],
235 [cmd.Name.SUBMIT_ELEMENT, post('/session/:sessionId/element/:id/submit')],
236 [cmd.Name.GET_ELEMENT_TEXT, get('/session/:sessionId/element/:id/text')],
237 [
238 cmd.Name.GET_COMPUTED_ROLE,
239 get('/session/:sessionId/element/:id/computedrole'),
240 ],
241 [
242 cmd.Name.GET_COMPUTED_LABEL,
243 get('/session/:sessionId/element/:id/computedlabel'),
244 ],
245 [cmd.Name.GET_ELEMENT_TAG_NAME, get('/session/:sessionId/element/:id/name')],
246 [
247 cmd.Name.IS_ELEMENT_SELECTED,
248 get('/session/:sessionId/element/:id/selected'),
249 ],
250 [cmd.Name.IS_ELEMENT_ENABLED, get('/session/:sessionId/element/:id/enabled')],
251 [
252 cmd.Name.IS_ELEMENT_DISPLAYED,
253 get('/session/:sessionId/element/:id/displayed'),
254 ],
255 [
256 cmd.Name.GET_ELEMENT_LOCATION,
257 get('/session/:sessionId/element/:id/location'),
258 ],
259 [cmd.Name.GET_ELEMENT_SIZE, get('/session/:sessionId/element/:id/size')],
260 [
261 cmd.Name.GET_ELEMENT_ATTRIBUTE,
262 get('/session/:sessionId/element/:id/attribute/:name'),
263 ],
264 [
265 cmd.Name.GET_ELEMENT_PROPERTY,
266 get('/session/:sessionId/element/:id/property/:name'),
267 ],
268 [
269 cmd.Name.GET_ELEMENT_VALUE_OF_CSS_PROPERTY,
270 get('/session/:sessionId/element/:id/css/:propertyName'),
271 ],
272 [
273 cmd.Name.TAKE_ELEMENT_SCREENSHOT,
274 get('/session/:sessionId/element/:id/screenshot'),
275 ],
276 [cmd.Name.SWITCH_TO_WINDOW, post('/session/:sessionId/window')],
277 [
278 cmd.Name.MAXIMIZE_WINDOW,
279 post('/session/:sessionId/window/current/maximize'),
280 ],
281 [
282 cmd.Name.GET_WINDOW_POSITION,
283 get('/session/:sessionId/window/current/position'),
284 ],
285 [
286 cmd.Name.SET_WINDOW_POSITION,
287 post('/session/:sessionId/window/current/position'),
288 ],
289 [cmd.Name.GET_WINDOW_SIZE, get('/session/:sessionId/window/current/size')],
290 [cmd.Name.SET_WINDOW_SIZE, post('/session/:sessionId/window/current/size')],
291 [cmd.Name.SWITCH_TO_FRAME, post('/session/:sessionId/frame')],
292 [cmd.Name.SWITCH_TO_FRAME_PARENT, post('/session/:sessionId/frame/parent')],
293 [cmd.Name.GET_PAGE_SOURCE, get('/session/:sessionId/source')],
294 [cmd.Name.GET_TITLE, get('/session/:sessionId/title')],
295 [cmd.Name.EXECUTE_SCRIPT, post('/session/:sessionId/execute')],
296 [cmd.Name.EXECUTE_ASYNC_SCRIPT, post('/session/:sessionId/execute_async')],
297 [cmd.Name.SCREENSHOT, get('/session/:sessionId/screenshot')],
298 [cmd.Name.GET_TIMEOUT, get('/session/:sessionId/timeouts')],
299 [cmd.Name.SET_TIMEOUT, post('/session/:sessionId/timeouts')],
300 [cmd.Name.ACCEPT_ALERT, post('/session/:sessionId/accept_alert')],
301 [cmd.Name.DISMISS_ALERT, post('/session/:sessionId/dismiss_alert')],
302 [cmd.Name.GET_ALERT_TEXT, get('/session/:sessionId/alert_text')],
303 [cmd.Name.SET_ALERT_TEXT, post('/session/:sessionId/alert_text')],
304 [cmd.Name.GET_LOG, post('/session/:sessionId/log')],
305 [cmd.Name.GET_AVAILABLE_LOG_TYPES, get('/session/:sessionId/log/types')],
306 [cmd.Name.GET_SESSION_LOGS, post('/logs')],
307 [cmd.Name.UPLOAD_FILE, post('/session/:sessionId/se/file')],
308])
309
310/** @const {!Map<string, (CommandSpec|CommandTransformer)>} */
311const W3C_COMMAND_MAP = new Map([
312 // Server status.
313 [cmd.Name.GET_SERVER_STATUS, get('/status')],
314 // Session management.
315 [cmd.Name.NEW_SESSION, post('/session')],
316 [cmd.Name.QUIT, del('/session/:sessionId')],
317 [cmd.Name.GET_TIMEOUT, get('/session/:sessionId/timeouts')],
318 [cmd.Name.SET_TIMEOUT, post('/session/:sessionId/timeouts')],
319 // Navigation.
320 [cmd.Name.GET_CURRENT_URL, get('/session/:sessionId/url')],
321 [cmd.Name.GET, post('/session/:sessionId/url')],
322 [cmd.Name.GO_BACK, post('/session/:sessionId/back')],
323 [cmd.Name.GO_FORWARD, post('/session/:sessionId/forward')],
324 [cmd.Name.REFRESH, post('/session/:sessionId/refresh')],
325 // Page inspection.
326 [cmd.Name.GET_PAGE_SOURCE, get('/session/:sessionId/source')],
327 [cmd.Name.GET_TITLE, get('/session/:sessionId/title')],
328 // Script execution.
329 [cmd.Name.EXECUTE_SCRIPT, post('/session/:sessionId/execute/sync')],
330 [cmd.Name.EXECUTE_ASYNC_SCRIPT, post('/session/:sessionId/execute/async')],
331 // Frame selection.
332 [cmd.Name.SWITCH_TO_FRAME, post('/session/:sessionId/frame')],
333 [cmd.Name.SWITCH_TO_FRAME_PARENT, post('/session/:sessionId/frame/parent')],
334 // Window management.
335 [cmd.Name.GET_CURRENT_WINDOW_HANDLE, get('/session/:sessionId/window')],
336 [cmd.Name.CLOSE, del('/session/:sessionId/window')],
337 [cmd.Name.SWITCH_TO_WINDOW, post('/session/:sessionId/window')],
338 [cmd.Name.SWITCH_TO_NEW_WINDOW, post('/session/:sessionId/window/new')],
339 [cmd.Name.GET_WINDOW_HANDLES, get('/session/:sessionId/window/handles')],
340 [cmd.Name.GET_WINDOW_RECT, get('/session/:sessionId/window/rect')],
341 [cmd.Name.SET_WINDOW_RECT, post('/session/:sessionId/window/rect')],
342 [cmd.Name.MAXIMIZE_WINDOW, post('/session/:sessionId/window/maximize')],
343 [cmd.Name.MINIMIZE_WINDOW, post('/session/:sessionId/window/minimize')],
344 [cmd.Name.FULLSCREEN_WINDOW, post('/session/:sessionId/window/fullscreen')],
345 // Actions.
346 [cmd.Name.ACTIONS, post('/session/:sessionId/actions')],
347 [cmd.Name.CLEAR_ACTIONS, del('/session/:sessionId/actions')],
348 // Locating elements.
349 [cmd.Name.GET_ACTIVE_ELEMENT, get('/session/:sessionId/element/active')],
350 [cmd.Name.FIND_ELEMENT, post('/session/:sessionId/element')],
351 [cmd.Name.FIND_ELEMENTS, post('/session/:sessionId/elements')],
352 [
353 cmd.Name.FIND_ELEMENTS_RELATIVE,
354 (cmd) => {
355 return toExecuteAtomCommand(cmd, Atom.FIND_ELEMENTS, 'args')
356 },
357 ],
358 [
359 cmd.Name.FIND_CHILD_ELEMENT,
360 post('/session/:sessionId/element/:id/element'),
361 ],
362 [
363 cmd.Name.FIND_CHILD_ELEMENTS,
364 post('/session/:sessionId/element/:id/elements'),
365 ],
366 // Element interaction.
367 [cmd.Name.GET_ELEMENT_TAG_NAME, get('/session/:sessionId/element/:id/name')],
368 [
369 cmd.Name.GET_ELEMENT_PROPERTY,
370 get('/session/:sessionId/element/:id/property/:name'),
371 ],
372 [
373 cmd.Name.GET_ELEMENT_VALUE_OF_CSS_PROPERTY,
374 get('/session/:sessionId/element/:id/css/:propertyName'),
375 ],
376 [cmd.Name.GET_ELEMENT_RECT, get('/session/:sessionId/element/:id/rect')],
377 [cmd.Name.CLEAR_ELEMENT, post('/session/:sessionId/element/:id/clear')],
378 [cmd.Name.CLICK_ELEMENT, post('/session/:sessionId/element/:id/click')],
379 [
380 cmd.Name.SEND_KEYS_TO_ELEMENT,
381 post('/session/:sessionId/element/:id/value'),
382 ],
383 [cmd.Name.GET_ELEMENT_TEXT, get('/session/:sessionId/element/:id/text')],
384 [
385 cmd.Name.GET_COMPUTED_ROLE,
386 get('/session/:sessionId/element/:id/computedrole'),
387 ],
388 [
389 cmd.Name.GET_COMPUTED_LABEL,
390 get('/session/:sessionId/element/:id/computedlabel'),
391 ],
392 [cmd.Name.IS_ELEMENT_ENABLED, get('/session/:sessionId/element/:id/enabled')],
393 [
394 cmd.Name.GET_ELEMENT_ATTRIBUTE,
395 (cmd) => {
396 return toExecuteAtomCommand(cmd, Atom.GET_ATTRIBUTE, 'id', 'name')
397 },
398 ],
399 [
400 cmd.Name.IS_ELEMENT_DISPLAYED,
401 (cmd) => {
402 return toExecuteAtomCommand(cmd, Atom.IS_DISPLAYED, 'id')
403 },
404 ],
405 // Cookie management.
406 [cmd.Name.GET_ALL_COOKIES, get('/session/:sessionId/cookie')],
407 [cmd.Name.ADD_COOKIE, post('/session/:sessionId/cookie')],
408 [cmd.Name.DELETE_ALL_COOKIES, del('/session/:sessionId/cookie')],
409 [cmd.Name.GET_COOKIE, get('/session/:sessionId/cookie/:name')],
410 [cmd.Name.DELETE_COOKIE, del('/session/:sessionId/cookie/:name')],
411 // Alert management.
412 [cmd.Name.ACCEPT_ALERT, post('/session/:sessionId/alert/accept')],
413 [cmd.Name.DISMISS_ALERT, post('/session/:sessionId/alert/dismiss')],
414 [cmd.Name.GET_ALERT_TEXT, get('/session/:sessionId/alert/text')],
415 [cmd.Name.SET_ALERT_TEXT, post('/session/:sessionId/alert/text')],
416 // Screenshots.
417 [cmd.Name.SCREENSHOT, get('/session/:sessionId/screenshot')],
418 [
419 cmd.Name.TAKE_ELEMENT_SCREENSHOT,
420 get('/session/:sessionId/element/:id/screenshot'),
421 ],
422 // print page.
423 [cmd.Name.PRINT_PAGE, post('/session/:sessionId/print')],
424 // Log extensions.
425 [cmd.Name.GET_LOG, post('/session/:sessionId/se/log')],
426 [cmd.Name.GET_AVAILABLE_LOG_TYPES, get('/session/:sessionId/se/log/types')],
427])
428
429/**
430 * Handles sending HTTP messages to a remote end.
431 *
432 * @interface
433 */
434class Client {
435 /**
436 * Sends a request to the server. The client will automatically follow any
437 * redirects returned by the server, fulfilling the returned promise with the
438 * final response.
439 *
440 * @param {!Request} httpRequest The request to send.
441 * @return {!Promise<Response>} A promise that will be fulfilled with the
442 * server's response.
443 */
444 send(httpRequest) {} // eslint-disable-line
445}
446
447/**
448 * @param {Map<string, CommandSpec>} customCommands
449 * A map of custom command definitions.
450 * @param {boolean} w3c Whether to use W3C command mappings.
451 * @param {!cmd.Command} command The command to resolve.
452 * @return {!Request} A promise that will resolve with the
453 * command to execute.
454 */
455function buildRequest(customCommands, w3c, command) {
456 LOG.finest(() => `Translating command: ${command.getName()}`)
457 let spec = customCommands && customCommands.get(command.getName())
458 if (spec) {
459 return toHttpRequest(spec)
460 }
461
462 if (w3c) {
463 spec = W3C_COMMAND_MAP.get(command.getName())
464 if (typeof spec === 'function') {
465 LOG.finest(() => `Transforming command for W3C: ${command.getName()}`)
466 let newCommand = spec(command)
467 return buildRequest(customCommands, w3c, newCommand)
468 } else if (spec) {
469 return toHttpRequest(spec)
470 }
471 }
472
473 spec = COMMAND_MAP.get(command.getName())
474 if (spec) {
475 return toHttpRequest(spec)
476 }
477 throw new error.UnknownCommandError(
478 'Unrecognized command: ' + command.getName()
479 )
480
481 /**
482 * @param {CommandSpec} resource
483 * @return {!Request}
484 */
485 function toHttpRequest(resource) {
486 LOG.finest(() => `Building HTTP request: ${JSON.stringify(resource)}`)
487 let parameters = command.getParameters()
488 let path = buildPath(resource.path, parameters)
489 return new Request(resource.method, path, parameters)
490 }
491}
492
493const CLIENTS =
494 /** !WeakMap<!Executor, !(Client|IThenable<!Client>)> */ new WeakMap()
495
496/**
497 * A command executor that communicates with the server using JSON over HTTP.
498 *
499 * By default, each instance of this class will use the legacy wire protocol
500 * from [Selenium project][json]. The executor will automatically switch to the
501 * [W3C wire protocol][w3c] if the remote end returns a compliant response to
502 * a new session command.
503 *
504 * [json]: https://github.com/SeleniumHQ/selenium/wiki/JsonWireProtocol
505 * [w3c]: https://w3c.github.io/webdriver/webdriver-spec.html
506 *
507 * @implements {cmd.Executor}
508 */
509class Executor {
510 /**
511 * @param {!(Client|IThenable<!Client>)} client The client to use for sending
512 * requests to the server, or a promise-like object that will resolve to
513 * to the client.
514 */
515 constructor(client) {
516 CLIENTS.set(this, client)
517
518 /**
519 * Whether this executor should use the W3C wire protocol. The executor
520 * will automatically switch if the remote end sends a compliant response
521 * to a new session command, however, this property may be directly set to
522 * `true` to force the executor into W3C mode.
523 * @type {boolean}
524 */
525 this.w3c = false
526
527 /** @private {Map<string, CommandSpec>} */
528 this.customCommands_ = null
529
530 /** @private {!logging.Logger} */
531 this.log_ = logging.getLogger('webdriver.http.Executor')
532 }
533
534 /**
535 * Defines a new command for use with this executor. When a command is sent,
536 * the {@code path} will be preprocessed using the command's parameters; any
537 * path segments prefixed with ":" will be replaced by the parameter of the
538 * same name. For example, given "/person/:name" and the parameters
539 * "{name: 'Bob'}", the final command path will be "/person/Bob".
540 *
541 * @param {string} name The command name.
542 * @param {string} method The HTTP method to use when sending this command.
543 * @param {string} path The path to send the command to, relative to
544 * the WebDriver server's command root and of the form
545 * "/path/:variable/segment".
546 */
547 defineCommand(name, method, path) {
548 if (!this.customCommands_) {
549 this.customCommands_ = new Map()
550 }
551 this.customCommands_.set(name, { method, path })
552 }
553
554 /** @override */
555 async execute(command) {
556 let request = buildRequest(this.customCommands_, this.w3c, command)
557 this.log_.finer(() => `>>> ${request.method} ${request.path}`)
558
559 let client = CLIENTS.get(this)
560 if (promise.isPromise(client)) {
561 client = await client
562 CLIENTS.set(this, client)
563 }
564
565 let response = await client.send(request)
566 this.log_.finer(() => `>>>\n${request}\n<<<\n${response}`)
567
568 let httpResponse = /** @type {!Response} */ (response)
569 let { isW3C, value } = parseHttpResponse(command, httpResponse)
570
571 if (command.getName() === cmd.Name.NEW_SESSION) {
572 if (!value || !value.sessionId) {
573 throw new error.WebDriverError(
574 `Unable to parse new session response: ${response.body}`
575 )
576 }
577
578 // The remote end is a W3C compliant server if there is no `status`
579 // field in the response.
580 if (command.getName() === cmd.Name.NEW_SESSION) {
581 this.w3c = this.w3c || isW3C
582 }
583
584 // No implementations use the `capabilities` key yet...
585 let capabilities = value.capabilities || value.value
586 return new Session(
587 /** @type {{sessionId: string}} */ (value).sessionId,
588 capabilities
589 )
590 }
591
592 return typeof value === 'undefined' ? null : value
593 }
594}
595
596/**
597 * @param {string} str .
598 * @return {?} .
599 */
600function tryParse(str) {
601 try {
602 return JSON.parse(str)
603 } catch (ignored) {
604 // Do nothing.
605 }
606}
607
608/**
609 * Callback used to parse {@link Response} objects from a
610 * {@link HttpClient}.
611 *
612 * @param {!cmd.Command} command The command the response is for.
613 * @param {!Response} httpResponse The HTTP response to parse.
614 * @return {{isW3C: boolean, value: ?}} An object describing the parsed
615 * response. This object will have two fields: `isW3C` indicates whether
616 * the response looks like it came from a remote end that conforms with the
617 * W3C WebDriver spec, and `value`, the actual response value.
618 * @throws {WebDriverError} If the HTTP response is an error.
619 */
620function parseHttpResponse(command, httpResponse) {
621 if (httpResponse.status < 200) {
622 // This should never happen, but throw the raw response so users report it.
623 throw new error.WebDriverError(`Unexpected HTTP response:\n${httpResponse}`)
624 }
625
626 let parsed = tryParse(httpResponse.body)
627 if (parsed && typeof parsed === 'object') {
628 let value = parsed.value
629 let isW3C =
630 value !== null &&
631 typeof value === 'object' &&
632 typeof parsed.status === 'undefined'
633
634 if (!isW3C) {
635 error.checkLegacyResponse(parsed)
636
637 // Adjust legacy new session responses to look like W3C to simplify
638 // later processing.
639 if (command.getName() === cmd.Name.NEW_SESSION) {
640 value = parsed
641 }
642 } else if (httpResponse.status > 399) {
643 error.throwDecodedError(value)
644 }
645
646 return { isW3C, value }
647 }
648
649 if (parsed !== undefined) {
650 return { isW3C: false, value: parsed }
651 }
652
653 let value = httpResponse.body.replace(/\r\n/g, '\n')
654
655 // 404 represents an unknown command; anything else > 399 is a generic unknown
656 // error.
657 if (httpResponse.status == 404) {
658 throw new error.UnsupportedOperationError(command.getName() + ': ' + value)
659 } else if (httpResponse.status >= 400) {
660 throw new error.WebDriverError(value)
661 }
662
663 return { isW3C: false, value: value || null }
664}
665
666/**
667 * Builds a fully qualified path using the given set of command parameters. Each
668 * path segment prefixed with ':' will be replaced by the value of the
669 * corresponding parameter. All parameters spliced into the path will be
670 * removed from the parameter map.
671 * @param {string} path The original resource path.
672 * @param {!Object<*>} parameters The parameters object to splice into the path.
673 * @return {string} The modified path.
674 */
675function buildPath(path, parameters) {
676 let pathParameters = path.match(/\/:(\w+)\b/g)
677 if (pathParameters) {
678 for (let i = 0; i < pathParameters.length; ++i) {
679 let key = pathParameters[i].substring(2) // Trim the /:
680 if (key in parameters) {
681 let value = parameters[key]
682 if (WebElement.isId(value)) {
683 // When inserting a WebElement into the URL, only use its ID value,
684 // not the full JSON.
685 value = WebElement.extractId(value)
686 }
687 path = path.replace(pathParameters[i], '/' + value)
688 delete parameters[key]
689 } else {
690 throw new error.InvalidArgumentError(
691 'Missing required parameter: ' + key
692 )
693 }
694 }
695 }
696 return path
697}
698
699// PUBLIC API
700
701exports.Executor = Executor
702exports.Client = Client
703exports.Request = Request
704exports.Response = Response
705exports.buildPath = buildPath // Exported for testing.