<!-- Please describe your changes on the following line: --> The previous implementation differed from the spec, because there was three maps instead of one. With this, they will be merged into one. Also this map has been moved from bluetooth to bluetoothDevice, because its make more sense to store it there. There is an issue about it [here](https://github.com/WebBluetoothCG/web-bluetooth/issues/330). --- <!-- Thank you for contributing to Servo! Please replace each `[ ]` by `[X]` when the step is complete, and replace `__` with appropriate data: --> - [X] `./mach build -d` does not report any errors - [X] `./mach test-tidy` does not report any errors <!-- Either: --> - [X] There are tests for these changes <!-- Pull requests that do not address these steps are welcome, but they will require additional verification as part of the review process. --> Source-Repo: https://github.com/servo/servo Source-Revision: 2e1c40ec360ece7165fec11810158bf925d322dc
392 lines
16 KiB
Rust
392 lines
16 KiB
Rust
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
use bluetooth_traits::{BluetoothRequest, BluetoothResponse};
|
|
use bluetooth_traits::blocklist::{Blocklist, uuid_is_blocklisted};
|
|
use dom::bindings::cell::DOMRefCell;
|
|
use dom::bindings::codegen::Bindings::BluetoothCharacteristicPropertiesBinding::
|
|
BluetoothCharacteristicPropertiesMethods;
|
|
use dom::bindings::codegen::Bindings::BluetoothDeviceBinding::BluetoothDeviceMethods;
|
|
use dom::bindings::codegen::Bindings::BluetoothRemoteGATTCharacteristicBinding;
|
|
use dom::bindings::codegen::Bindings::BluetoothRemoteGATTCharacteristicBinding::
|
|
BluetoothRemoteGATTCharacteristicMethods;
|
|
use dom::bindings::codegen::Bindings::BluetoothRemoteGATTServerBinding::BluetoothRemoteGATTServerMethods;
|
|
use dom::bindings::codegen::Bindings::BluetoothRemoteGATTServiceBinding::BluetoothRemoteGATTServiceMethods;
|
|
use dom::bindings::codegen::Bindings::EventHandlerBinding::EventHandlerNonNull;
|
|
use dom::bindings::error::Error::{self, InvalidModification, Network, NotSupported, Security};
|
|
use dom::bindings::inheritance::Castable;
|
|
use dom::bindings::js::{JS, MutHeap, Root};
|
|
use dom::bindings::reflector::{Reflectable, reflect_dom_object};
|
|
use dom::bindings::str::{ByteString, DOMString};
|
|
use dom::bluetooth::{AsyncBluetoothListener, response_async};
|
|
use dom::bluetoothcharacteristicproperties::BluetoothCharacteristicProperties;
|
|
use dom::bluetoothremotegattservice::BluetoothRemoteGATTService;
|
|
use dom::bluetoothuuid::{BluetoothDescriptorUUID, BluetoothUUID};
|
|
use dom::eventtarget::EventTarget;
|
|
use dom::globalscope::GlobalScope;
|
|
use dom::promise::Promise;
|
|
use ipc_channel::ipc::IpcSender;
|
|
use js::jsapi::JSContext;
|
|
use std::rc::Rc;
|
|
|
|
// Maximum length of an attribute value.
|
|
// https://www.bluetooth.org/DocMan/handlers/DownloadDoc.ashx?doc_id=286439 (Vol. 3, page 2169)
|
|
pub const MAXIMUM_ATTRIBUTE_LENGTH: usize = 512;
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#bluetoothremotegattcharacteristic
|
|
#[dom_struct]
|
|
pub struct BluetoothRemoteGATTCharacteristic {
|
|
eventtarget: EventTarget,
|
|
service: MutHeap<JS<BluetoothRemoteGATTService>>,
|
|
uuid: DOMString,
|
|
properties: MutHeap<JS<BluetoothCharacteristicProperties>>,
|
|
value: DOMRefCell<Option<ByteString>>,
|
|
instance_id: String,
|
|
}
|
|
|
|
impl BluetoothRemoteGATTCharacteristic {
|
|
pub fn new_inherited(service: &BluetoothRemoteGATTService,
|
|
uuid: DOMString,
|
|
properties: &BluetoothCharacteristicProperties,
|
|
instance_id: String)
|
|
-> BluetoothRemoteGATTCharacteristic {
|
|
BluetoothRemoteGATTCharacteristic {
|
|
eventtarget: EventTarget::new_inherited(),
|
|
service: MutHeap::new(service),
|
|
uuid: uuid,
|
|
properties: MutHeap::new(properties),
|
|
value: DOMRefCell::new(None),
|
|
instance_id: instance_id,
|
|
}
|
|
}
|
|
|
|
pub fn new(global: &GlobalScope,
|
|
service: &BluetoothRemoteGATTService,
|
|
uuid: DOMString,
|
|
properties: &BluetoothCharacteristicProperties,
|
|
instanceID: String)
|
|
-> Root<BluetoothRemoteGATTCharacteristic> {
|
|
reflect_dom_object(box BluetoothRemoteGATTCharacteristic::new_inherited(service,
|
|
uuid,
|
|
properties,
|
|
instanceID),
|
|
global,
|
|
BluetoothRemoteGATTCharacteristicBinding::Wrap)
|
|
}
|
|
|
|
fn get_bluetooth_thread(&self) -> IpcSender<BluetoothRequest> {
|
|
self.global().as_window().bluetooth_thread()
|
|
}
|
|
|
|
fn get_instance_id(&self) -> String {
|
|
self.instance_id.clone()
|
|
}
|
|
}
|
|
|
|
impl BluetoothRemoteGATTCharacteristicMethods for BluetoothRemoteGATTCharacteristic {
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-properties
|
|
fn Properties(&self) -> Root<BluetoothCharacteristicProperties> {
|
|
self.properties.get()
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-service
|
|
fn Service(&self) -> Root<BluetoothRemoteGATTService> {
|
|
self.service.get()
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-uuid
|
|
fn Uuid(&self) -> DOMString {
|
|
self.uuid.clone()
|
|
}
|
|
|
|
#[allow(unrooted_must_root)]
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-getdescriptor
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#getgattchildren
|
|
fn GetDescriptor(&self, descriptor: BluetoothDescriptorUUID) -> Rc<Promise> {
|
|
let p = Promise::new(&self.global());
|
|
let p_cx = p.global().get_cx();
|
|
|
|
// Step 1.
|
|
let uuid = match BluetoothUUID::descriptor(descriptor) {
|
|
Ok(uuid) => uuid.to_string(),
|
|
Err(e) => {
|
|
p.reject_error(p_cx, e);
|
|
return p;
|
|
}
|
|
};
|
|
|
|
// Step 2.
|
|
if uuid_is_blocklisted(uuid.as_ref(), Blocklist::All) {
|
|
p.reject_error(p_cx, Security);
|
|
return p;
|
|
}
|
|
|
|
// Step 3 - 4.
|
|
if !self.Service().Device().Gatt().Connected() {
|
|
p.reject_error(p_cx, Network);
|
|
return p;
|
|
}
|
|
|
|
// TODO: Step 5: Implement representedService internal slot for BluetoothRemoteGATTService.
|
|
|
|
// Note: Steps 6 - 7 are implemented in components/bluetooth/lib.rs in get_descriptor function
|
|
// and in handle_response function.
|
|
let sender = response_async(&p, self);
|
|
self.get_bluetooth_thread().send(
|
|
BluetoothRequest::GetDescriptor(self.get_instance_id(), uuid, sender)).unwrap();
|
|
return p;
|
|
}
|
|
|
|
#[allow(unrooted_must_root)]
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-getdescriptors
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#getgattchildren
|
|
fn GetDescriptors(&self,
|
|
descriptor: Option<BluetoothDescriptorUUID>)
|
|
-> Rc<Promise> {
|
|
let p = Promise::new(&self.global());
|
|
let p_cx = p.global().get_cx();
|
|
let mut uuid: Option<String> = None;
|
|
if let Some(d) = descriptor {
|
|
// Step 1.
|
|
uuid = match BluetoothUUID::descriptor(d) {
|
|
Ok(uuid) => Some(uuid.to_string()),
|
|
Err(e) => {
|
|
p.reject_error(p_cx, e);
|
|
return p;
|
|
}
|
|
};
|
|
if let Some(ref uuid) = uuid {
|
|
// Step 2.
|
|
if uuid_is_blocklisted(uuid.as_ref(), Blocklist::All) {
|
|
p.reject_error(p_cx, Security);
|
|
return p;
|
|
}
|
|
}
|
|
};
|
|
|
|
// Step 3 - 4.
|
|
if !self.Service().Device().Gatt().Connected() {
|
|
p.reject_error(p_cx, Network);
|
|
return p;
|
|
}
|
|
|
|
// TODO: Step 5: Implement representedService internal slot for BluetoothRemoteGATTService.
|
|
|
|
// Note: Steps 6 - 7 are implemented in components/bluetooth/lib.rs in get_descriptors function
|
|
// and in handle_response function.
|
|
let sender = response_async(&p, self);
|
|
self.get_bluetooth_thread().send(
|
|
BluetoothRequest::GetDescriptors(self.get_instance_id(), uuid, sender)).unwrap();
|
|
return p;
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-value
|
|
fn GetValue(&self) -> Option<ByteString> {
|
|
self.value.borrow().clone()
|
|
}
|
|
|
|
#[allow(unrooted_must_root)]
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-readvalue
|
|
fn ReadValue(&self) -> Rc<Promise> {
|
|
let p = Promise::new(&self.global());
|
|
let p_cx = p.global().get_cx();
|
|
|
|
// Step 1.
|
|
if uuid_is_blocklisted(self.uuid.as_ref(), Blocklist::Reads) {
|
|
p.reject_error(p_cx, Security);
|
|
return p;
|
|
}
|
|
|
|
// Step 2.
|
|
if !self.Service().Device().Gatt().Connected() {
|
|
p.reject_error(p_cx, Network);
|
|
return p;
|
|
}
|
|
|
|
// TODO: Step 3 - 4: Implement representedCharacteristic internal slot for BluetoothRemoteGATTCharacteristic.
|
|
|
|
// TODO: Step 5: Implement the `connection-checking-wrapper` algorithm for BluetoothRemoteGATTServer.
|
|
|
|
// Step 5.1.
|
|
if !self.Properties().Read() {
|
|
p.reject_error(p_cx, NotSupported);
|
|
return p;
|
|
}
|
|
|
|
// Note: Remaining substeps of Step 5 are implemented in components/bluetooth/lib.rs in readValue function
|
|
// and in handle_response function.
|
|
let sender = response_async(&p, self);
|
|
self.get_bluetooth_thread().send(
|
|
BluetoothRequest::ReadValue(self.get_instance_id(), sender)).unwrap();
|
|
return p;
|
|
}
|
|
|
|
#[allow(unrooted_must_root)]
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-writevalue
|
|
fn WriteValue(&self, value: Vec<u8>) -> Rc<Promise> {
|
|
let p = Promise::new(&self.global());
|
|
let p_cx = p.global().get_cx();
|
|
|
|
// Step 1.
|
|
if uuid_is_blocklisted(self.uuid.as_ref(), Blocklist::Writes) {
|
|
p.reject_error(p_cx, Security);
|
|
return p;
|
|
}
|
|
|
|
// Step 2 - 3.
|
|
if value.len() > MAXIMUM_ATTRIBUTE_LENGTH {
|
|
p.reject_error(p_cx, InvalidModification);
|
|
return p;
|
|
}
|
|
|
|
// Step 4.
|
|
if !self.Service().Device().Gatt().Connected() {
|
|
p.reject_error(p_cx, Network);
|
|
return p;
|
|
}
|
|
|
|
// TODO: Step 5 - 6: Implement representedCharacteristic internal slot for BluetoothRemoteGATTCharacteristic.
|
|
|
|
// TODO: Step 7: Implement the `connection-checking-wrapper` algorithm for BluetoothRemoteGATTServer.
|
|
|
|
// Step 7.1.
|
|
if !(self.Properties().Write() ||
|
|
self.Properties().WriteWithoutResponse() ||
|
|
self.Properties().AuthenticatedSignedWrites()) {
|
|
p.reject_error(p_cx, NotSupported);
|
|
return p;
|
|
}
|
|
|
|
// Note: Remaining substeps of Step 7 are implemented in components/bluetooth/lib.rs in writeValue function
|
|
// and in handle_response function.
|
|
let sender = response_async(&p, self);
|
|
self.get_bluetooth_thread().send(
|
|
BluetoothRequest::WriteValue(self.get_instance_id(), value, sender)).unwrap();
|
|
return p;
|
|
}
|
|
|
|
#[allow(unrooted_must_root)]
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-startnotifications
|
|
fn StartNotifications(&self) -> Rc<Promise> {
|
|
let p = Promise::new(&self.global());
|
|
let p_cx = p.global().get_cx();
|
|
|
|
// Step 1.
|
|
if uuid_is_blocklisted(self.uuid.as_ref(), Blocklist::Reads) {
|
|
p.reject_error(p_cx, Security);
|
|
return p;
|
|
}
|
|
|
|
// TODO: Step 2 - 3: Implement representedCharacteristic internal slot for BluetoothRemoteGATTCharacteristic.
|
|
|
|
// Step 4.
|
|
if !(self.Properties().Notify() ||
|
|
self.Properties().Indicate()) {
|
|
p.reject_error(p_cx, NotSupported);
|
|
return p;
|
|
}
|
|
|
|
// TODO: Step 5: Implement `active notification context set` for BluetoothRemoteGATTCharacteristic.
|
|
|
|
// Step 6.
|
|
if !self.Service().Device().Gatt().Connected() {
|
|
p.reject_error(p_cx, Network);
|
|
return p;
|
|
}
|
|
|
|
// Note: Steps 7 - 11 are implemented in components/bluetooth/lib.rs in enable_notification function
|
|
// and in handle_response function.
|
|
let sender = response_async(&p, self);
|
|
self.get_bluetooth_thread().send(
|
|
BluetoothRequest::EnableNotification(self.get_instance_id(),
|
|
true,
|
|
sender)).unwrap();
|
|
return p;
|
|
}
|
|
|
|
#[allow(unrooted_must_root)]
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-stopnotifications
|
|
fn StopNotifications(&self) -> Rc<Promise> {
|
|
let p = Promise::new(&self.global());
|
|
let sender = response_async(&p, self);
|
|
|
|
// TODO: Step 1 - 4: Implement representedCharacteristic internal slot and
|
|
// `active notification context set` for BluetoothRemoteGATTCharacteristic,
|
|
|
|
// Note: Part of Step 4 and Step 5 are implemented in components/bluetooth/lib.rs in enable_notification
|
|
// function and in handle_response function.
|
|
self.get_bluetooth_thread().send(
|
|
BluetoothRequest::EnableNotification(self.get_instance_id(),
|
|
false,
|
|
sender)).unwrap();
|
|
return p;
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-characteristiceventhandlers-oncharacteristicvaluechanged
|
|
event_handler!(characteristicvaluechanged, GetOncharacteristicvaluechanged, SetOncharacteristicvaluechanged);
|
|
}
|
|
|
|
impl AsyncBluetoothListener for BluetoothRemoteGATTCharacteristic {
|
|
fn handle_response(&self, response: BluetoothResponse, promise_cx: *mut JSContext, promise: &Rc<Promise>) {
|
|
let device = self.Service().Device();
|
|
match response {
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-getdescriptor
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#getgattchildren
|
|
// Step 7.
|
|
BluetoothResponse::GetDescriptor(descriptor) => {
|
|
let bt_descriptor = device.get_or_create_descriptor(&descriptor, &self);
|
|
promise.resolve_native(promise_cx, &bt_descriptor);
|
|
},
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-getdescriptors
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#getgattchildren
|
|
// Step 7.
|
|
BluetoothResponse::GetDescriptors(descriptors_vec) => {
|
|
let mut descriptors = vec!();
|
|
for descriptor in descriptors_vec {
|
|
let bt_descriptor = device.get_or_create_descriptor(&descriptor, &self);
|
|
descriptors.push(bt_descriptor);
|
|
}
|
|
promise.resolve_native(promise_cx, &descriptors);
|
|
},
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-readvalue
|
|
BluetoothResponse::ReadValue(result) => {
|
|
// TODO: Step 5.5.1: Implement activeAlgorithms internal slot for BluetoothRemoteGATTServer.
|
|
|
|
// Step 5.5.2.
|
|
// TODO(#5014): Replace ByteString with ArrayBuffer when it is implemented.
|
|
let value = ByteString::new(result);
|
|
*self.value.borrow_mut() = Some(value.clone());
|
|
|
|
// Step 5.5.3.
|
|
self.upcast::<EventTarget>().fire_bubbling_event(atom!("characteristicvaluechanged"));
|
|
|
|
// Step 5.5.4.
|
|
promise.resolve_native(promise_cx, &value);
|
|
},
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-writevalue
|
|
BluetoothResponse::WriteValue(result) => {
|
|
// TODO: Step 7.5.1: Implement activeAlgorithms internal slot for BluetoothRemoteGATTServer.
|
|
|
|
// Step 7.5.2.
|
|
// TODO(#5014): Replace ByteString with an ArrayBuffer wrapped in a DataView.
|
|
*self.value.borrow_mut() = Some(ByteString::new(result));
|
|
|
|
// Step 7.5.3.
|
|
promise.resolve_native(promise_cx, &());
|
|
},
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-startnotifications
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-stopnotifications
|
|
BluetoothResponse::EnableNotification(_result) => {
|
|
// (StartNotification) TODO: Step 10: Implement `active notification context set`
|
|
// for BluetoothRemoteGATTCharacteristic.
|
|
|
|
// (StartNotification) Step 11.
|
|
// (StopNotification) Step 5.
|
|
promise.resolve_native(promise_cx, self);
|
|
},
|
|
_ => promise.reject_error(promise_cx, Error::Type("Something went wrong...".to_owned())),
|
|
}
|
|
}
|
|
}
|