Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add Bluetooth device emulation support #630

Merged
merged 29 commits into from
Oct 4, 2024
Merged
Changes from 21 commits
Commits
Show all changes
29 commits
Select commit Hold shift + click to select a range
c413238
Initial cut
alexnj Aug 10, 2024
35e3421
Add semantics and examples
alexnj Aug 16, 2024
9e201a6
Review changes.
alexnj Aug 19, 2024
9bd0bf4
Curly brace for SimulateAdvertisementParameters
alexnj Aug 19, 2024
a5217c4
Add algorithm for adding a virtual peripheral
alexnj Sep 4, 2024
b67a242
Add remote algorithm for simulateAdvertisement
alexnj Sep 4, 2024
d99c313
Review changes
alexnj Sep 4, 2024
f2a3577
Review updates
alexnj Sep 5, 2024
3c9eec6
Reintroduce simulateCentral + review feedback
alexnj Sep 11, 2024
f8f6dc4
Rename simulateCentral to simulateAdapter
alexnj Sep 11, 2024
ee735fd
Rename virtual prefix to simulated + review changes
alexnj Sep 12, 2024
83d6e02
Review changes
alexnj Sep 12, 2024
7053802
Formatting for scanRecord definitions
alexnj Sep 12, 2024
6e0a6b3
Apply suggestions from code review
alexnj Sep 16, 2024
5c9b744
Review edits
alexnj Sep 16, 2024
e90b078
Update index.bs
alexnj Sep 16, 2024
977a282
Review edits + issues
alexnj Sep 16, 2024
4450610
Update changes from offline sync
alexnj Sep 17, 2024
2f92b5b
An attempt at resolving the correct event loop
alexnj Sep 17, 2024
4411dce
Add text to resolve navigator.bluetooth on navigable.
alexnj Sep 18, 2024
e34b094
Revise spec to adjust for bidi navigable resolution
alexnj Sep 18, 2024
8df4f2d
Review updates + CDDL changes to match the actual impl.
alexnj Sep 19, 2024
03050b9
More updates.
alexnj Sep 19, 2024
0a18c16
Update steps for global param for scan for devices algorithm
alexnj Sep 19, 2024
79256fa
Add a missing [=exists=].
alexnj Sep 19, 2024
dab2bf1
Correct lint errors + review changes
alexnj Sep 20, 2024
fa4b089
Review changes
alexnj Sep 20, 2024
b7a40ff
Review changes.
alexnj Sep 24, 2024
b373044
Update data type to bstr
alexnj Oct 2, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
226 changes: 223 additions & 3 deletions index.bs
Original file line number Diff line number Diff line change
Expand Up @@ -127,6 +127,12 @@ spec: html
spec: webidl
type: dfn
text: resolve
spec: webdriver
type: dfn
text: remote end steps
type: dfn
text: current browsing context

</pre>

<style>
Expand Down Expand Up @@ -1658,6 +1664,13 @@ following steps:
1. Let <var>nearbyDevices</var> be a set of <a>Bluetooth device</a>s, initially
equal to the set of devices that are connected (have an <a>ATT Bearer</a>)
to the UA.
1. Let |topLevelTraversable| be |global|'s [=Window/navigable=]'s [=navigable/top-level
alexnj marked this conversation as resolved.
Show resolved Hide resolved
traversable=].
1. Let |simulatedBluetoothDevices| be an empty [=set=].
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. If |topLevelTraversable| has a [=simulated Bluetooth adapter=], let |simulatedBluetoothDevices| be its set of [=simulated Bluetooth devices=].

Issue: Support asynchronous device discovery.

1. If the UA supports the LE transport, perform the <a>General Discovery
Procedure</a>, except that the UA may include devices that have no
<a>Discoverable Mode</a> flag set, and add the discovered <a>Bluetooth
Expand All @@ -1677,7 +1690,7 @@ following steps:
immutable device address.
1. Let <var>result</var> be a set of <a>Bluetooth device</a>s, initially empty.
1. For each <a>Bluetooth device</a> <var>device</var> in
<var>nearbyDevices</var>, do the following sub-steps:
<var>nearbyDevices</var> and <var>simulatedBluetoothDevices</var>, do the following sub-steps:
1. If <var>device</var>'s <a>supported physical transports</a> include LE
and its <a>Bluetooth Device Name</a> is partial or absent, the UA SHOULD
perform the <a>Name Discovery Procedure</a> to acquire a complete name.
Expand Down Expand Up @@ -1988,6 +2001,14 @@ steps <a>in parallel</a>:
Note: If the Web Bluetooth permission has been blocked by the user, the UA
may <a>resolve</a> |promise| with `false`.
</div>

1. Let |simulatedBluetoothAdapter| be [=this=]'s [=Window/navigable=]'s
alexnj marked this conversation as resolved.
Show resolved Hide resolved
[=navigable/top-level traversable=] <a>simulated Bluetooth adapter</a>.
1. If |simulatedBluetoothAdapter| is not empty,
1. If |simulatedBluetoothAdapter|["state"] is "absent", [=queue a task=] to [=resolve=] |promise| with `false`.
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. Otherwise, [=queue a task=] to [=resolve=] |promise| with `true`.
1. Abort these steps.

1. If the UA is running on a system that has a Bluetooth radio <a>queue a
task</a> to <a>resolve</a> |promise| with `true` regardless of the powered
state of the Bluetooth radio.
Expand Down Expand Up @@ -2495,8 +2516,7 @@ broadcasting, measured in dBm. This is used to compute the path loss as
in dBm. This is used to compute the path loss as <code>this.txPower -
this.rssi</code>.

<dfn>manufacturerData</dfn> maps <code>unsigned short</code> Company Identifier
Codes to {{DataView}}s.
<dfn>manufacturerData</dfn> maps <code>unsigned short</code> Company IdentifierCodes to list of unsigned octets.
alexnj marked this conversation as resolved.
Show resolved Hide resolved
alexnj marked this conversation as resolved.
Show resolved Hide resolved

<dfn>serviceData</dfn> maps {{UUID}}s to {{DataView}}s.
</div>
Expand Down Expand Up @@ -4827,6 +4847,10 @@ if the following steps return `blocked`:

# Extensions to the Navigator Interface # {#navigator-extensions}

Each {{Window}} has an <dfn>associated `Bluetooth`</dfn>, which is a {{Bluetooth}} object. Upon creation of the {{Window}} object, its <a>associated `Bluetooth`</a> must be set to a new {{Bluetooth}} object created in the {{Window}} object's [=relevant realm=].
alexnj marked this conversation as resolved.
Show resolved Hide resolved

{{Navigator}}'s {{bluetooth}} getter steps are to return [=this=]'s <a>associated `Bluetooth`</a>.
alexnj marked this conversation as resolved.
Show resolved Hide resolved

<xmp class="idl">
[SecureContext]
partial interface Navigator {
Expand All @@ -4850,6 +4874,14 @@ The <a>default allowlist</a> for this feature is <code>["self"]</code>.

For the purposes of user-agent automation and application testing, this document defines extensions to the [[WebDriver-BiDi]] specification.

The Web Bluetooth API and its extension specifications pose a challenge to test authors, as fully exercising those interfaces requires physical hardware devices that respond in predictable ways. To address this challenge this document defines a number of WebDriver-BiDi extension commands that allow defining and controlling simulated peripherals and advertisements that behave like physical device peripherals and their advertisements. These simulated peripherals and advertisements represent devices with particular properties and whose readings can be entirely defined by users.

Each [=navigable/top-level traversable=] may have a <dfn>simulated Bluetooth adapter</dfn>, that is a software defined Bluetooth adapter that has a set of discovered <a>simulated Bluetooth devices</a> and can assume roles like <a>Central</a>.

Each <a>simulated Bluetooth adapter</a> has a <dfn>simulated Bluetooth device mapping</dfn>, which is an <a>ordered map</a> of Bluetooth address to <a>simulated Bluetooth devices</a>.
alexnj marked this conversation as resolved.
Show resolved Hide resolved

A <dfn>simulated Bluetooth device</dfn> is a software defined [=Bluetooth device=] that behaves like a physical device, and may be attached to a <a>simulated Bluetooth adapter</a>.

Issue: CDDL snippetes use the "text" type instead of
"browsingContext.BrowsingContext" to allow indepedent programmatic
processing of CDDL snippets. Currently, other modules cannot be
Expand Down Expand Up @@ -4938,6 +4970,34 @@ To <dfn>serialize prompt devices</dfn> given [=device prompt=] |prompt|:

</div>

#### The bluetooth.ScanRecord Type #### {#bluetooth-scanrecord-type}

<pre highlight="cddl" class="cddl remote-cddl local-cddl">
bluetooth.ScanRecord = {
optional name: text,
optional uuids: BluetoothServiceUuid[],
optional appearance: integer,
alexnj marked this conversation as resolved.
Show resolved Hide resolved
optional manufacturerData: BluetoothManufacturerDataMap,
}
</pre>

A `bluetooth.ScanRecord` represents data of the advertisement packet sent by a [=Bluetooth device=].

<dl>
<dt><code>name</code></dt>
<dd>is {{BluetoothDevice}}'s local name, or a prefix of it.</dd>
alexnj marked this conversation as resolved.
Show resolved Hide resolved

<dt><code>uuids</code></dt>
<dd>lists the Service UUIDs that this scan record says the [=Bluetooth device=]'s GATT server supports.</dd>

<dt><code>appearance</code></dt>
<dd>is an <a>Appearance</a>, one of the values defined by the {{gap.appearance}} characteristic.</dd>

<dt><code>manufacturerData</code></dt>
<dd>maps <code>unsigned short</code> Company Identifier Codes to {{DataView}}s.</dd>
alexnj marked this conversation as resolved.
Show resolved Hide resolved
</dl>


### Errors ### {#bidi-errors}

This specification extends the set of [=error codes=] from
Expand Down Expand Up @@ -5013,6 +5073,166 @@ A [=local end=] could dismiss a prompt by sending the following message:
</pre>
</div>

#### The bluetooth.simulateAdapter Command #### {#bluetooth-simulateAdapter-command}

<pre highlight="cddl" class="cddl remote-cddl local-cddl">
bluetooth.simulateAdapter = (
method: "bluetooth.simulateAdapter",
params: bluetooth.SimulateAdapterParameters,
)

bluetooth.SimulateAdapterParameters = {
state: "absent" / "powered-off" / "powered-on"
}
</pre>

<div algorithm="remote end steps for bluetooth.simulateAdapter">
The [=remote end steps=] with command parameters |params| are:

1. Let contextId be params["context"].
alexnj marked this conversation as resolved.
Show resolved Hide resolved
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. Let |navigable| be the result of [=trying=] to [=get a navigable=] with |contextId|.
1. If |navigable| is not a [=top-level traversable=], return [=error=] with [=error code=] [=invalid argument=].
1. Let |state| be |params|[`"state"`].
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. Let |simulatedBluetoothAdapter| be a new [=simulated Bluetooth adapter=].
1. Set |simulatedBluetoothAdapter|'s state to |state|.
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. Set |navigable|'s <a>simulated Bluetooth adapter</a> to |simulatedBluetoothAdapter|.
1. Return [=success=] with data `null`.
alexnj marked this conversation as resolved.
Show resolved Hide resolved

</div>

<div class="example">
A [=local end=] could simulate an adapter by sending the following message:

<pre highlight="json">
{
"method": "bluetooth.simulateAdapter",
"state": "powered-on"
}
</pre>
</div>

#### The bluetooth.simulatePreconnectedPeripheral Command #### {#bluetooth-simulateconnectedperipheral-command}

<pre highlight="cddl" class="cddl remote-cddl local-cddl">
bluetooth.SimulatePreconnectedPeripheral = (
method: "bluetooth.simulatePreconnectedPeripheral",
params: bluetooth.SimulatePreconnectedPeripheralParameters,
)

bluetooth.SimulatePreconnectedPeripheralParameters = {
address: string,
name: string,
manufacturerData: BluetoothManufacturerDataMap,
knownServiceUuids: string[]
}
</pre>

<div algorithm="remote end steps for bluetooth.simulatePreconnectedPeripheral">
The [=remote end steps=] with command parameters |params| are:

1. Let |contextId| be params["context"].
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. Let |navigable| be the result of [=trying=] to [=get a navigable=] with |contextId|.
1. If |navigable| is not a [=top-level traversable=], return [=error=] with [=error code=] [=invalid argument=].
1. Let |simulatedBluetoothAdapter| be |navigable|'s <a>simulated Bluetooth adapter</a>.
1. If |simulatedBluetoothAdapter| is empty, return [=error=].
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. Let |deviceAddress| be |params|[`"address"`].
1. Let |deviceMapping| be |simulatedBluetoothAdapter|'s <a>simulated Bluetooth device mapping</a>.
1. If |deviceMapping|[|deviceAddress|] exists, return [=error=].
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. Let |simulatedBluetoothDevice| be a new [=simulated Bluetooth device=].
1. Set |simulatedBluetoothDevice|'s name to |params|[`"name"`].
1. Set |simulatedBluetoothDevice|'s address to |params|[`"address"`].
1. Set |simulatedBluetoothDevice|'s manufacturer data to |params|[`"manufacturerData"`].
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. Set |simulatedBluetoothDevice|'s known service UUIDs to |params|[`"knownServiceUuids"`].
1. Set |deviceMapping|[|deviceAddress|] to |simulatedBluetoothDevice|.
1. Return [=success=] with data `null`.
alexnj marked this conversation as resolved.
Show resolved Hide resolved

</div>

<div class="example">
A [=local end=] could simulate a preconnected peripheral by sending the following message:

<pre highlight="json">
{
"method": "bluetooth.simulatePreconnectedPeripheral",
"address": "09:09:09:09:09:09",
"name": "Some Device",
"manufacturerData": { 17: [0, 255, 1, 1, 127] },
"knownServiceUuids": [
"12345678-1234-5678-9abc-def123456789",
]
}
</pre>
</div>

#### The bluetooth.simulateAdvertisement Command #### {#bluetooth-simulateadvertisement-command}

<pre highlight="cddl" class="cddl remote-cddl local-cddl">
bluetooth.SimulateAdvertisement = (
method: "bluetooth.simulateAdvertisement",
params: bluetooth.SimulateAdvertisementParameters,
)

bluetooth.SimulateAdvertisementParameters = {
scanEntry: bluetooth.SimulateAdvertisementScanEntryParameters
}

bluetooth.SimulateAdvertisementScanEntryParameters = {
deviceAddress: string,
rssi: integer,
scanRecord: bluetooth.ScanRecord
}

</pre>

<div algorithm="remote end steps for bluetooth.simulateAdvertisement">
The [=remote end steps=] with command parameters |params| are:

1. Let |contextId| be params["context"].
alexnj marked this conversation as resolved.
Show resolved Hide resolved
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. Let |topLevelNavigable| be the result of [=trying=] to [=get a navigable=] with |contextId|.
1. If |topLevelNavigable| is not a [=top-level traversable=], return [=error=] with [=error code=] [=invalid argument=].
1. Let |scanEntry| be |params|[`"scanEntry"`].
1. Let |deviceAddress| be |scanEntry|[`"deviceAddress"`].
1. Let |simulatedBluetoothAdapter| be |topLevelNavigable|'s <a>simulated Bluetooth adapter</a>.
1. If |simulatedBluetoothAdapter| is empty, return [=error=].
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. Let |deviceMapping| be |simulatedBluetoothAdapter|'s <a>simulated Bluetooth device mapping</a>.
1. If |deviceMapping|[|deviceAddress|] exists, let |simulatedDevice| be |deviceMapping|[|deviceAddress|]. Otherwise, let |simulatedDevice| be a new <a>simulated Bluetooth device</a> with |deviceAddress| and set |deviceMapping|[|deviceAddress|] to |simulatedDevice|.
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. If the [=navigable/top-level traversable=] is currently executing the
alexnj marked this conversation as resolved.
Show resolved Hide resolved
[=scan for devices=] algorithm, insert <var>simulatedDevice</var> into
the <em>simulatedBluetoothDevices</em> variable within that algorithm.

Issue: Inserting data into variables from another algorithm could be improved.
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. Let |navigables| be the <a>inclusive descendant navigables</a> of |topLevelNavigable|'s <a>active document</a>.
1. For each |navigable| of |navigables|:
1. Let |document| be |navigable|'s <a>active document</a>.
1. <a>Queue a task</a> on |document|'s <a>relevant settings object</a>'s <a>responsible event loop</a> to do the following sub-steps:
1. Let |simulatedDeviceInstance| be the {{BluetoothDevice}} the |navigable|'s `navigator.bluetooth.[[deviceInstanceMap]]` corresponding to |simulatedDevice|.
alexnj marked this conversation as resolved.
Show resolved Hide resolved
alexnj marked this conversation as resolved.
Show resolved Hide resolved
1. If |simulatedDeviceInstance|.{{[[watchAdvertisementsState]]}} is `not-watching`, abort these sub-steps.
1. <a>Fire an `advertisementreceived` event</a> for the advertising event represented by |scanEntry|[`"scanRecord"`], at |simulatedDeviceInstance|.
1. Return [=success=] with data `null`.

</div>

<div class="example">
A [=local end=] could simulate a device advertisement by sending the following message:

<pre highlight="json">
{
"method": "bluetooth.simulateAdvertisement",
"entry": {
"deviceAddress": "08:08:08:08:08:08",
"rssi": -10,
"scanRecord": {
"name": "Heart Rate",
"uuids": ["0000180d-0000-1000-8000-00805f9b34fb"],
"manufacturerData": { 17: [0, 255, 1, 1, 127] },
"appearance": 1,
"txPower": 1
}
}
}
</pre>
</div>

### Events ### {#bidi-events}

#### The bluetooth.requestDevicePromptUpdated Event #### {#bluetooth-requestdevicepromptupdated-event}
Expand Down