aboutsummaryrefslogtreecommitdiffstats
path: root/components
diff options
context:
space:
mode:
Diffstat (limited to 'components')
-rw-r--r--components/script/Cargo.toml1
-rw-r--r--components/script/dom/window.rs2
-rw-r--r--components/script/webdriver_handlers.rs71
-rw-r--r--components/script_traits/Cargo.toml1
-rw-r--r--components/script_traits/webdriver_msg.rs6
-rw-r--r--components/webdriver_server/lib.rs10
6 files changed, 82 insertions, 9 deletions
diff --git a/components/script/Cargo.toml b/components/script/Cargo.toml
index cb03c00b873..2473dba7c9c 100644
--- a/components/script/Cargo.toml
+++ b/components/script/Cargo.toml
@@ -111,6 +111,7 @@ url = "1.6"
utf-8 = "0.7"
uuid = {version = "0.7", features = ["v4"]}
xml5ever = {version = "0.14"}
+webdriver = "0.40"
webrender_api = {git = "https://github.com/servo/webrender", features = ["ipc"]}
webvr_traits = {path = "../webvr_traits"}
webxr-api = {git = "https://github.com/servo/webxr", features = ["ipc"]}
diff --git a/components/script/dom/window.rs b/components/script/dom/window.rs
index b33a5c5a126..09247103d0c 100644
--- a/components/script/dom/window.rs
+++ b/components/script/dom/window.rs
@@ -964,7 +964,7 @@ impl WindowMethods for Window {
#[allow(unsafe_code)]
fn WebdriverCallback(&self, cx: JSContext, val: HandleValue) {
- let rv = unsafe { jsval_to_webdriver(*cx, val) };
+ let rv = unsafe { jsval_to_webdriver(*cx, &self.globalscope, val) };
let opt_chan = self.webdriver_script_chan.borrow_mut().take();
if let Some(chan) = opt_chan {
chan.send(rv).unwrap();
diff --git a/components/script/webdriver_handlers.rs b/components/script/webdriver_handlers.rs
index 874f707a739..182f110c0c8 100644
--- a/components/script/webdriver_handlers.rs
+++ b/components/script/webdriver_handlers.rs
@@ -12,7 +12,10 @@ use crate::dom::bindings::codegen::Bindings::NodeBinding::NodeMethods;
use crate::dom::bindings::codegen::Bindings::WindowBinding::WindowMethods;
use crate::dom::bindings::codegen::Bindings::XMLSerializerBinding::XMLSerializerMethods;
use crate::dom::bindings::conversions::{
- get_property_jsval, ConversionResult, FromJSValConvertible, StringificationBehavior,
+ get_property, get_property_jsval, is_array_like, root_from_object,
+};
+use crate::dom::bindings::conversions::{
+ ConversionBehavior, ConversionResult, FromJSValConvertible, StringificationBehavior,
};
use crate::dom::bindings::error::throw_dom_exception;
use crate::dom::bindings::inheritance::Castable;
@@ -29,12 +32,13 @@ use crate::dom::node::{window_from_node, Node, ShadowIncluding};
use crate::dom::nodelist::NodeList;
use crate::dom::window::Window;
use crate::dom::xmlserializer::XMLSerializer;
+use crate::script_runtime::JSContext as SafeJSContext;
use crate::script_thread::Documents;
use cookie::Cookie;
use euclid::default::{Point2D, Rect, Size2D};
use hyper_serde::Serde;
use ipc_channel::ipc::{self, IpcSender};
-use js::jsapi::JSContext;
+use js::jsapi::{JSAutoRealm, JSContext};
use js::jsval::UndefinedValue;
use js::rust::HandleValue;
use msg::constellation_msg::BrowsingContextId;
@@ -47,6 +51,7 @@ use script_traits::webdriver_msg::{
WebDriverFrameId, WebDriverJSError, WebDriverJSResult, WebDriverJSValue,
};
use servo_url::ServoUrl;
+use webdriver::common::WebElement;
fn find_node_by_unique_id(
documents: &Documents,
@@ -106,9 +111,15 @@ fn first_matching_link(
}
#[allow(unsafe_code)]
-pub unsafe fn jsval_to_webdriver(cx: *mut JSContext, val: HandleValue) -> WebDriverJSResult {
+pub unsafe fn jsval_to_webdriver(
+ cx: *mut JSContext,
+ global_scope: &GlobalScope,
+ val: HandleValue,
+) -> WebDriverJSResult {
if val.get().is_undefined() {
Ok(WebDriverJSValue::Undefined)
+ } else if val.get().is_null() {
+ Ok(WebDriverJSValue::Null)
} else if val.get().is_boolean() {
Ok(WebDriverJSValue::Boolean(val.get().to_boolean()))
} else if val.get().is_double() || val.get().is_int32() {
@@ -128,8 +139,52 @@ pub unsafe fn jsval_to_webdriver(cx: *mut JSContext, val: HandleValue) -> WebDri
_ => unreachable!(),
};
Ok(WebDriverJSValue::String(String::from(string)))
- } else if val.get().is_null() {
- Ok(WebDriverJSValue::Null)
+ } else if val.get().is_object() {
+ rooted!(in(cx) let object = match FromJSValConvertible::from_jsval(cx, val, ()).unwrap() {
+ ConversionResult::Success(object) => object,
+ _ => unreachable!(),
+ });
+ let _ac = JSAutoRealm::new(cx, *object);
+
+ if let Ok(element) = root_from_object::<HTMLElement>(*object, cx) {
+ return Ok(WebDriverJSValue::Element(WebElement(
+ element.upcast::<Node>().unique_id(),
+ )));
+ }
+
+ if !is_array_like(cx, val) {
+ return Err(WebDriverJSError::UnknownType);
+ }
+
+ let mut result: Vec<WebDriverJSValue> = Vec::new();
+
+ let length =
+ match get_property::<u32>(cx, object.handle(), "length", ConversionBehavior::Default) {
+ Ok(length) => match length {
+ Some(length) => length,
+ _ => return Err(WebDriverJSError::UnknownType),
+ },
+ Err(error) => {
+ throw_dom_exception(SafeJSContext::from_ptr(cx), global_scope, error);
+ return Err(WebDriverJSError::JSError);
+ },
+ };
+
+ for i in 0..length {
+ rooted!(in(cx) let mut item = UndefinedValue());
+ match get_property_jsval(cx, object.handle(), &i.to_string(), item.handle_mut()) {
+ Ok(_) => match jsval_to_webdriver(cx, global_scope, item.handle()) {
+ Ok(converted_item) => result.push(converted_item),
+ err @ Err(_) => return err,
+ },
+ Err(error) => {
+ throw_dom_exception(SafeJSContext::from_ptr(cx), global_scope, error);
+ return Err(WebDriverJSError::JSError);
+ },
+ }
+ }
+
+ Ok(WebDriverJSValue::ArrayLike(result))
} else {
Err(WebDriverJSError::UnknownType)
}
@@ -149,7 +204,7 @@ pub fn handle_execute_script(
window
.upcast::<GlobalScope>()
.evaluate_js_on_global_with_result(&eval, rval.handle_mut());
- jsval_to_webdriver(*cx, rval.handle())
+ jsval_to_webdriver(*cx, &window.upcast::<GlobalScope>(), rval.handle())
};
reply.send(result).unwrap();
@@ -734,7 +789,9 @@ pub fn handle_get_property(
property.handle_mut(),
)
} {
- Ok(_) => match unsafe { jsval_to_webdriver(*cx, property.handle()) } {
+ Ok(_) => match unsafe {
+ jsval_to_webdriver(*cx, &node.reflector().global(), property.handle())
+ } {
Ok(property) => Ok(property),
Err(_) => Ok(WebDriverJSValue::Undefined),
},
diff --git a/components/script_traits/Cargo.toml b/components/script_traits/Cargo.toml
index 61e1dda7410..bd1011daad2 100644
--- a/components/script_traits/Cargo.toml
+++ b/components/script_traits/Cargo.toml
@@ -38,6 +38,7 @@ servo_url = {path = "../url"}
style_traits = {path = "../style_traits", features = ["servo"]}
time = "0.1.12"
url = "1.2"
+webdriver = "0.40"
webrender_api = {git = "https://github.com/servo/webrender", features = ["ipc"]}
webvr_traits = {path = "../webvr_traits"}
webxr-api = {git = "https://github.com/servo/webxr", features = ["ipc"]}
diff --git a/components/script_traits/webdriver_msg.rs b/components/script_traits/webdriver_msg.rs
index 7849a8580ef..c33a5ae302d 100644
--- a/components/script_traits/webdriver_msg.rs
+++ b/components/script_traits/webdriver_msg.rs
@@ -10,6 +10,7 @@ use hyper_serde::Serde;
use ipc_channel::ipc::IpcSender;
use msg::constellation_msg::BrowsingContextId;
use servo_url::ServoUrl;
+use webdriver::common::WebElement;
#[derive(Debug, Deserialize, Serialize)]
pub enum WebDriverScriptCommand {
@@ -66,13 +67,16 @@ pub enum WebDriverJSValue {
Null,
Boolean(bool),
Number(f64),
- String(String), // TODO: Object and WebElement
+ String(String),
+ Element(WebElement),
+ ArrayLike(Vec<WebDriverJSValue>),
}
#[derive(Debug, Deserialize, Serialize)]
pub enum WebDriverJSError {
Timeout,
UnknownType,
+ JSError,
/// Occurs when handler received an event message for a layout channel that is not
/// associated with the current script thread
BrowsingContextNotFound,
diff --git a/components/webdriver_server/lib.rs b/components/webdriver_server/lib.rs
index 9d6ab3355d1..fd141879ebd 100644
--- a/components/webdriver_server/lib.rs
+++ b/components/webdriver_server/lib.rs
@@ -224,6 +224,12 @@ impl Serialize for SendableWebDriverJSValue {
WebDriverJSValue::Boolean(x) => serializer.serialize_bool(x),
WebDriverJSValue::Number(x) => serializer.serialize_f64(x),
WebDriverJSValue::String(ref x) => serializer.serialize_str(&x),
+ WebDriverJSValue::Element(ref x) => x.serialize(serializer),
+ WebDriverJSValue::ArrayLike(ref x) => x
+ .iter()
+ .map(|element| SendableWebDriverJSValue(element.clone()))
+ .collect::<Vec<SendableWebDriverJSValue>>()
+ .serialize(serializer),
}
}
}
@@ -1397,6 +1403,10 @@ impl Handler {
ErrorStatus::UnsupportedOperation,
"Unsupported return type",
)),
+ Err(WebDriverJSError::JSError) => Err(WebDriverError::new(
+ ErrorStatus::JavascriptError,
+ "JS evaluation raised an exception",
+ )),
Err(WebDriverJSError::BrowsingContextNotFound) => Err(WebDriverError::new(
ErrorStatus::JavascriptError,
"Pipeline id not found in browsing context",