org.openqa.selenium.remote.codec.w3c.W3CHttpCommandCodec Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of selenium-remote-driver Show documentation
Show all versions of selenium-remote-driver Show documentation
Selenium automates browsers. That's it! What you do with that power is entirely up to you.
// 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.
package org.openqa.selenium.remote.codec.w3c;
import static org.openqa.selenium.remote.DriverCommand.ACCEPT_ALERT;
import static org.openqa.selenium.remote.DriverCommand.ACTIONS;
import static org.openqa.selenium.remote.DriverCommand.CLEAR_ACTIONS_STATE;
import static org.openqa.selenium.remote.DriverCommand.CLEAR_LOCAL_STORAGE;
import static org.openqa.selenium.remote.DriverCommand.CLEAR_SESSION_STORAGE;
import static org.openqa.selenium.remote.DriverCommand.DISMISS_ALERT;
import static org.openqa.selenium.remote.DriverCommand.EXECUTE_ASYNC_SCRIPT;
import static org.openqa.selenium.remote.DriverCommand.EXECUTE_SCRIPT;
import static org.openqa.selenium.remote.DriverCommand.FIND_CHILD_ELEMENT;
import static org.openqa.selenium.remote.DriverCommand.FIND_CHILD_ELEMENTS;
import static org.openqa.selenium.remote.DriverCommand.FIND_ELEMENT;
import static org.openqa.selenium.remote.DriverCommand.FIND_ELEMENTS;
import static org.openqa.selenium.remote.DriverCommand.FIND_ELEMENTS_FROM_SHADOW_ROOT;
import static org.openqa.selenium.remote.DriverCommand.FIND_ELEMENT_FROM_SHADOW_ROOT;
import static org.openqa.selenium.remote.DriverCommand.GET_ACTIVE_ELEMENT;
import static org.openqa.selenium.remote.DriverCommand.GET_ALERT_TEXT;
import static org.openqa.selenium.remote.DriverCommand.GET_AVAILABLE_LOG_TYPES;
import static org.openqa.selenium.remote.DriverCommand.GET_CURRENT_WINDOW_HANDLE;
import static org.openqa.selenium.remote.DriverCommand.GET_CURRENT_WINDOW_POSITION;
import static org.openqa.selenium.remote.DriverCommand.GET_CURRENT_WINDOW_SIZE;
import static org.openqa.selenium.remote.DriverCommand.GET_ELEMENT_ACCESSIBLE_NAME;
import static org.openqa.selenium.remote.DriverCommand.GET_ELEMENT_ARIA_ROLE;
import static org.openqa.selenium.remote.DriverCommand.GET_ELEMENT_ATTRIBUTE;
import static org.openqa.selenium.remote.DriverCommand.GET_ELEMENT_DOM_ATTRIBUTE;
import static org.openqa.selenium.remote.DriverCommand.GET_ELEMENT_DOM_PROPERTY;
import static org.openqa.selenium.remote.DriverCommand.GET_ELEMENT_LOCATION;
import static org.openqa.selenium.remote.DriverCommand.GET_ELEMENT_LOCATION_ONCE_SCROLLED_INTO_VIEW;
import static org.openqa.selenium.remote.DriverCommand.GET_ELEMENT_RECT;
import static org.openqa.selenium.remote.DriverCommand.GET_ELEMENT_SHADOW_ROOT;
import static org.openqa.selenium.remote.DriverCommand.GET_ELEMENT_SIZE;
import static org.openqa.selenium.remote.DriverCommand.GET_LOCAL_STORAGE_ITEM;
import static org.openqa.selenium.remote.DriverCommand.GET_LOCAL_STORAGE_KEYS;
import static org.openqa.selenium.remote.DriverCommand.GET_LOCAL_STORAGE_SIZE;
import static org.openqa.selenium.remote.DriverCommand.GET_LOG;
import static org.openqa.selenium.remote.DriverCommand.GET_PAGE_SOURCE;
import static org.openqa.selenium.remote.DriverCommand.GET_SESSION_STORAGE_ITEM;
import static org.openqa.selenium.remote.DriverCommand.GET_SESSION_STORAGE_KEYS;
import static org.openqa.selenium.remote.DriverCommand.GET_SESSION_STORAGE_SIZE;
import static org.openqa.selenium.remote.DriverCommand.GET_WINDOW_HANDLES;
import static org.openqa.selenium.remote.DriverCommand.IS_ELEMENT_DISPLAYED;
import static org.openqa.selenium.remote.DriverCommand.MAXIMIZE_CURRENT_WINDOW;
import static org.openqa.selenium.remote.DriverCommand.MINIMIZE_CURRENT_WINDOW;
import static org.openqa.selenium.remote.DriverCommand.PRINT_PAGE;
import static org.openqa.selenium.remote.DriverCommand.REMOVE_LOCAL_STORAGE_ITEM;
import static org.openqa.selenium.remote.DriverCommand.REMOVE_SESSION_STORAGE_ITEM;
import static org.openqa.selenium.remote.DriverCommand.SEND_KEYS_TO_ELEMENT;
import static org.openqa.selenium.remote.DriverCommand.SET_ALERT_VALUE;
import static org.openqa.selenium.remote.DriverCommand.SET_CURRENT_WINDOW_POSITION;
import static org.openqa.selenium.remote.DriverCommand.SET_CURRENT_WINDOW_SIZE;
import static org.openqa.selenium.remote.DriverCommand.SET_LOCAL_STORAGE_ITEM;
import static org.openqa.selenium.remote.DriverCommand.SET_SESSION_STORAGE_ITEM;
import static org.openqa.selenium.remote.DriverCommand.SET_TIMEOUT;
import static org.openqa.selenium.remote.DriverCommand.SUBMIT_ELEMENT;
import static org.openqa.selenium.remote.DriverCommand.UPLOAD_FILE;
import java.io.IOException;
import java.io.InputStream;
import java.io.UncheckedIOException;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.regex.Pattern;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import org.openqa.selenium.InvalidSelectorException;
import org.openqa.selenium.WebDriverException;
import org.openqa.selenium.remote.WebElementToJsonConverter;
import org.openqa.selenium.remote.codec.AbstractHttpCommandCodec;
/**
* A command codec that adheres to the W3C's WebDriver wire protocol.
*
* @see W3C WebDriver spec
*/
public class W3CHttpCommandCodec extends AbstractHttpCommandCodec {
private static final ConcurrentHashMap ATOM_SCRIPTS = new ConcurrentHashMap<>();
private static final Pattern CSS_ESCAPE =
Pattern.compile("([\\s'\"\\\\#.:;,!?+<>=~*^$|%&@`{}\\-\\/\\[\\]\\(\\)])");
public W3CHttpCommandCodec() {
String sessionId = "/session/:sessionId";
alias(GET_ELEMENT_ATTRIBUTE, EXECUTE_SCRIPT);
alias(GET_ELEMENT_LOCATION, GET_ELEMENT_RECT);
alias(GET_ELEMENT_LOCATION_ONCE_SCROLLED_INTO_VIEW, EXECUTE_SCRIPT);
alias(GET_ELEMENT_SIZE, GET_ELEMENT_RECT);
alias(IS_ELEMENT_DISPLAYED, EXECUTE_SCRIPT);
alias(SUBMIT_ELEMENT, EXECUTE_SCRIPT);
defineCommand(EXECUTE_SCRIPT, post(sessionId + "/execute/sync"));
defineCommand(EXECUTE_ASYNC_SCRIPT, post(sessionId + "/execute/async"));
alias(GET_PAGE_SOURCE, EXECUTE_SCRIPT);
alias(CLEAR_LOCAL_STORAGE, EXECUTE_SCRIPT);
alias(GET_LOCAL_STORAGE_KEYS, EXECUTE_SCRIPT);
alias(SET_LOCAL_STORAGE_ITEM, EXECUTE_SCRIPT);
alias(REMOVE_LOCAL_STORAGE_ITEM, EXECUTE_SCRIPT);
alias(GET_LOCAL_STORAGE_ITEM, EXECUTE_SCRIPT);
alias(GET_LOCAL_STORAGE_SIZE, EXECUTE_SCRIPT);
alias(CLEAR_SESSION_STORAGE, EXECUTE_SCRIPT);
alias(GET_SESSION_STORAGE_KEYS, EXECUTE_SCRIPT);
alias(SET_SESSION_STORAGE_ITEM, EXECUTE_SCRIPT);
alias(REMOVE_SESSION_STORAGE_ITEM, EXECUTE_SCRIPT);
alias(GET_SESSION_STORAGE_ITEM, EXECUTE_SCRIPT);
alias(GET_SESSION_STORAGE_SIZE, EXECUTE_SCRIPT);
String window = sessionId + "/window";
defineCommand(MAXIMIZE_CURRENT_WINDOW, post(window + "/maximize"));
defineCommand(MINIMIZE_CURRENT_WINDOW, post(window + "/minimize"));
defineCommand(GET_CURRENT_WINDOW_SIZE, get(window + "/rect"));
defineCommand(SET_CURRENT_WINDOW_SIZE, post(window + "/rect"));
alias(GET_CURRENT_WINDOW_POSITION, GET_CURRENT_WINDOW_SIZE);
alias(SET_CURRENT_WINDOW_POSITION, SET_CURRENT_WINDOW_SIZE);
defineCommand(GET_CURRENT_WINDOW_HANDLE, get(window));
defineCommand(GET_WINDOW_HANDLES, get(window + "/handles"));
String alert = sessionId + "/alert";
defineCommand(ACCEPT_ALERT, post(alert + "/accept"));
defineCommand(DISMISS_ALERT, post(alert + "/dismiss"));
defineCommand(GET_ALERT_TEXT, get(alert + "/text"));
defineCommand(SET_ALERT_VALUE, post(alert + "/text"));
defineCommand(PRINT_PAGE, post(sessionId + "/print"));
defineCommand(UPLOAD_FILE, post(sessionId + "/se/file"));
defineCommand(GET_ACTIVE_ELEMENT, get(sessionId + "/element/active"));
defineCommand(ACTIONS, post(sessionId + "/actions"));
defineCommand(CLEAR_ACTIONS_STATE, delete(sessionId + "/actions"));
String elementId = sessionId + "/element/:id";
defineCommand(GET_ELEMENT_DOM_PROPERTY, get(elementId + "/property/:name"));
defineCommand(GET_ELEMENT_DOM_ATTRIBUTE, get(elementId + "/attribute/:name"));
defineCommand(GET_ELEMENT_ARIA_ROLE, get(elementId + "/computedrole"));
defineCommand(GET_ELEMENT_ACCESSIBLE_NAME, get(elementId + "/computedlabel"));
defineCommand(GET_ELEMENT_SHADOW_ROOT, get(elementId + "/shadow"));
defineCommand(FIND_ELEMENT_FROM_SHADOW_ROOT, post(sessionId + "/shadow/:shadowId/element"));
defineCommand(FIND_ELEMENTS_FROM_SHADOW_ROOT, post(sessionId + "/shadow/:shadowId/elements"));
defineCommand(GET_LOG, post(sessionId + "/se/log"));
defineCommand(GET_AVAILABLE_LOG_TYPES, get(sessionId + "/se/log/types"));
}
@Override
protected Map amendParameters(String name, Map parameters) {
switch (name) {
case FIND_CHILD_ELEMENT:
case FIND_CHILD_ELEMENTS:
case FIND_ELEMENT:
case FIND_ELEMENTS:
String using = (String) parameters.get("using");
Object value = parameters.get("value");
if (value instanceof String) {
String stringValue = (String) value;
switch (using) {
case "class name":
if (stringValue.matches(".*\\s.*")) {
throw new InvalidSelectorException("Compound class names not permitted");
}
return amendLocatorToCssSelector(parameters, "." + cssEscape(stringValue));
case "id":
return amendLocatorToCssSelector(parameters, "#" + cssEscape(stringValue));
case "name":
return amendLocatorToCssSelector(parameters, "*[name='" + stringValue + "']");
default:
// Do nothing
break;
}
}
return parameters;
case GET_ELEMENT_ATTRIBUTE:
// Read the atom, wrap it, execute it.
return executeAtom(
"getAttribute.js", asElement(parameters.get("id")), parameters.get("name"));
case GET_ELEMENT_LOCATION_ONCE_SCROLLED_INTO_VIEW:
return toScript(
"var e = arguments[0]; e.scrollIntoView({behavior: 'instant', block: 'end', inline:"
+ " 'nearest'}); var rect = e.getBoundingClientRect(); return {'x': rect.left, 'y':"
+ " rect.top};",
asElement(parameters.get("id")));
case GET_PAGE_SOURCE:
return toScript(
"var source = document.documentElement.outerHTML; \n"
+ "if (!source) { source = new XMLSerializer().serializeToString(document); }\n"
+ "return source;");
case CLEAR_LOCAL_STORAGE:
return toScript("localStorage.clear()");
case GET_LOCAL_STORAGE_KEYS:
return toScript("return Object.keys(localStorage)");
case SET_LOCAL_STORAGE_ITEM:
return toScript(
"localStorage.setItem(arguments[0], arguments[1])",
parameters.get("key"),
parameters.get("value"));
case REMOVE_LOCAL_STORAGE_ITEM:
return toScript(
"var item = localStorage.getItem(arguments[0]); localStorage.removeItem(arguments[0]);"
+ " return item",
parameters.get("key"));
case GET_LOCAL_STORAGE_ITEM:
return toScript("return localStorage.getItem(arguments[0])", parameters.get("key"));
case GET_LOCAL_STORAGE_SIZE:
return toScript("return localStorage.length");
case CLEAR_SESSION_STORAGE:
return toScript("sessionStorage.clear()");
case GET_SESSION_STORAGE_KEYS:
return toScript("return Object.keys(sessionStorage)");
case SET_SESSION_STORAGE_ITEM:
return toScript(
"sessionStorage.setItem(arguments[0], arguments[1])",
parameters.get("key"),
parameters.get("value"));
case REMOVE_SESSION_STORAGE_ITEM:
return toScript(
"var item = sessionStorage.getItem(arguments[0]);"
+ " sessionStorage.removeItem(arguments[0]); return item",
parameters.get("key"));
case GET_SESSION_STORAGE_ITEM:
return toScript("return sessionStorage.getItem(arguments[0])", parameters.get("key"));
case GET_SESSION_STORAGE_SIZE:
return toScript("return sessionStorage.length");
case IS_ELEMENT_DISPLAYED:
return executeAtom("isDisplayed.js", asElement(parameters.get("id")));
case SEND_KEYS_TO_ELEMENT:
// When converted from JSON, this is a list, not an array
Object rawValue = parameters.get("value");
Stream source;
if (rawValue instanceof Collection) {
//noinspection unchecked
source = ((Collection) rawValue).stream();
} else {
source = Stream.of((CharSequence[]) rawValue);
}
String text = source.collect(Collectors.joining());
{
Map merged = new LinkedHashMap<>();
parameters.forEach(
(key, val) -> {
if ("text".equals(key) || "value".equals(key)) {
return;
}
merged.put(key, val);
});
merged.put("text", text);
merged.put("value", stringToUtf8Array(text));
return Map.copyOf(merged);
}
case SET_ALERT_VALUE:
return Map.of(
"text", parameters.get("text"),
"value", stringToUtf8Array((String) parameters.get("text")));
case SET_TIMEOUT:
String timeoutType = (String) parameters.get("type");
Number duration = (Number) parameters.get("ms");
if (timeoutType == null) {
// Assume a local end that Knows What To Do according to the spec
return parameters;
}
{
Map merged = new LinkedHashMap<>();
parameters.forEach(
(key, val) -> {
if (timeoutType.equals(key)) {
return;
}
merged.put(key, val);
});
merged.put(timeoutType, duration);
return Map.copyOf(merged);
}
case SUBMIT_ELEMENT:
return toScript(
"/* submitForm */var form = arguments[0];\n"
+ "while (form.nodeName != \"FORM\" && form.parentNode) {\n"
+ " form = form.parentNode;\n"
+ "}\n"
+ "if (!form) { throw Error('Unable to find containing form element'); }\n"
+ "if (!form.ownerDocument) { throw Error('Unable to find owning document'); }\n"
+ "var e = form.ownerDocument.createEvent('Event');\n"
+ "e.initEvent('submit', true, true);\n"
+ "if (form.dispatchEvent(e)) { HTMLFormElement.prototype.submit.call(form) }\n",
asElement(parameters.get("id")));
default:
return parameters;
}
}
private List stringToUtf8Array(String toConvert) {
List toReturn = new ArrayList<>();
int offset = 0;
while (offset < toConvert.length()) {
int next = toConvert.codePointAt(offset);
toReturn.add(new StringBuilder().appendCodePoint(next).toString());
offset += Character.charCount(next);
}
return toReturn;
}
private Map executeAtom(String atomFileName, Object... args) {
try {
String script =
ATOM_SCRIPTS.computeIfAbsent(
atomFileName,
(fileName) -> {
String scriptName = "/org/openqa/selenium/remote/" + atomFileName;
String rawFunction;
try (InputStream stream = getClass().getResourceAsStream(scriptName)) {
rawFunction = new String(stream.readAllBytes(), StandardCharsets.UTF_8);
} catch (IOException e) {
throw new UncheckedIOException(e);
}
String atomName = fileName.replace(".js", "");
return String.format(
"/* %s */return (%s).apply(null, arguments);", atomName, rawFunction);
});
return toScript(script, args);
} catch (UncheckedIOException e) {
throw new WebDriverException(e.getCause());
} catch (NullPointerException e) {
throw new WebDriverException(e);
}
}
private Map toScript(String script, Object... args) {
List